| // Copyright 2016 The PDFium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "core/fpdfdoc/cpdf_dest.h" |
| |
| #include <algorithm> |
| #include <iterator> |
| #include <utility> |
| |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_document.h" |
| #include "core/fpdfapi/parser/cpdf_name.h" |
| #include "core/fpdfapi/parser/cpdf_number.h" |
| #include "core/fpdfdoc/cpdf_nametree.h" |
| |
| namespace { |
| |
| // These arrays are indexed by the PDFDEST_VIEW_* constants. |
| |
| // Last element is a sentinel. |
| const char* const kZoomModes[] = {"Unknown", "XYZ", "Fit", "FitH", "FitV", |
| "FitR", "FitB", "FitBH", "FitBV", nullptr}; |
| |
| constexpr uint8_t kZoomModeMaxParamCount[] = {0, 3, 0, 1, 1, 4, 0, 1, 1, 0}; |
| |
| static_assert(std::size(kZoomModes) == std::size(kZoomModeMaxParamCount), |
| "Zoom mode count Mismatch"); |
| |
| } // namespace |
| |
| CPDF_Dest::CPDF_Dest(RetainPtr<const CPDF_Array> pArray) |
| : m_pArray(std::move(pArray)) {} |
| |
| CPDF_Dest::CPDF_Dest(const CPDF_Dest& that) = default; |
| |
| CPDF_Dest::~CPDF_Dest() = default; |
| |
| // static |
| CPDF_Dest CPDF_Dest::Create(CPDF_Document* pDoc, |
| RetainPtr<const CPDF_Object> pDest) { |
| if (!pDest) |
| return CPDF_Dest(nullptr); |
| |
| if (pDest->IsString() || pDest->IsName()) |
| return CPDF_Dest(CPDF_NameTree::LookupNamedDest(pDoc, pDest->GetString())); |
| |
| return CPDF_Dest(ToArray(pDest)); |
| } |
| |
| int CPDF_Dest::GetDestPageIndex(CPDF_Document* pDoc) const { |
| if (!m_pArray) |
| return -1; |
| |
| RetainPtr<const CPDF_Object> pPage = m_pArray->GetDirectObjectAt(0); |
| if (!pPage) |
| return -1; |
| |
| if (pPage->IsNumber()) |
| return pPage->GetInteger(); |
| |
| if (!pPage->IsDictionary()) |
| return -1; |
| |
| return pDoc->GetPageIndex(pPage->GetObjNum()); |
| } |
| |
| std::vector<float> CPDF_Dest::GetScrollPositionArray() const { |
| std::vector<float> result; |
| if (m_pArray) { |
| // Skip over index 0 which contains destination page details, and index 1 |
| // which contains a parameter that describes the rest of the array. |
| for (size_t i = 2; i < m_pArray->size(); i++) |
| result.push_back(m_pArray->GetFloatAt(i)); |
| } |
| return result; |
| } |
| |
| int CPDF_Dest::GetZoomMode() const { |
| if (!m_pArray) |
| return 0; |
| |
| RetainPtr<const CPDF_Object> pArray = m_pArray->GetDirectObjectAt(1); |
| if (!pArray) |
| return 0; |
| |
| ByteString mode = pArray->GetString(); |
| for (int i = 1; kZoomModes[i]; ++i) { |
| if (mode == kZoomModes[i]) |
| return i; |
| } |
| |
| return 0; |
| } |
| |
| bool CPDF_Dest::GetXYZ(bool* pHasX, |
| bool* pHasY, |
| bool* pHasZoom, |
| float* pX, |
| float* pY, |
| float* pZoom) const { |
| *pHasX = false; |
| *pHasY = false; |
| *pHasZoom = false; |
| |
| if (!m_pArray) |
| return false; |
| |
| if (m_pArray->size() < 5) |
| return false; |
| |
| RetainPtr<const CPDF_Name> xyz = ToName(m_pArray->GetDirectObjectAt(1)); |
| if (!xyz || xyz->GetString() != "XYZ") |
| return false; |
| |
| RetainPtr<const CPDF_Number> numX = ToNumber(m_pArray->GetDirectObjectAt(2)); |
| RetainPtr<const CPDF_Number> numY = ToNumber(m_pArray->GetDirectObjectAt(3)); |
| RetainPtr<const CPDF_Number> numZoom = |
| ToNumber(m_pArray->GetDirectObjectAt(4)); |
| |
| // If the value is a CPDF_Null then ToNumber will return nullptr. |
| *pHasX = !!numX; |
| *pHasY = !!numY; |
| *pHasZoom = !!numZoom; |
| |
| if (numX) |
| *pX = numX->GetNumber(); |
| if (numY) |
| *pY = numY->GetNumber(); |
| |
| // A zoom value of 0 is equivalent to a null value, so treat it as a null. |
| if (numZoom) { |
| float num = numZoom->GetNumber(); |
| if (num == 0.0) |
| *pHasZoom = false; |
| else |
| *pZoom = num; |
| } |
| |
| return true; |
| } |
| |
| size_t CPDF_Dest::GetNumParams() const { |
| if (!m_pArray || m_pArray->size() < 2) |
| return 0; |
| |
| size_t maxParamsForFitType = kZoomModeMaxParamCount[GetZoomMode()]; |
| size_t numParamsInArray = m_pArray->size() - 2; |
| return std::min(maxParamsForFitType, numParamsInArray); |
| } |
| |
| float CPDF_Dest::GetParam(size_t index) const { |
| return m_pArray ? m_pArray->GetFloatAt(2 + index) : 0; |
| } |