| // Copyright 2016 PDFium Authors. All rights reserved. |
| // 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_numbertree.h" |
| |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| |
| namespace { |
| |
| const CPDF_Object* SearchNumberNode(const CPDF_Dictionary* pNode, int num) { |
| const CPDF_Array* pLimits = pNode->GetArrayFor("Limits"); |
| if (pLimits && |
| (num < pLimits->GetIntegerAt(0) || num > pLimits->GetIntegerAt(1))) { |
| return nullptr; |
| } |
| const CPDF_Array* pNumbers = pNode->GetArrayFor("Nums"); |
| if (pNumbers) { |
| for (size_t i = 0; i < pNumbers->size() / 2; i++) { |
| int index = pNumbers->GetIntegerAt(i * 2); |
| if (num == index) |
| return pNumbers->GetDirectObjectAt(i * 2 + 1); |
| if (index > num) |
| break; |
| } |
| return nullptr; |
| } |
| |
| const CPDF_Array* pKids = pNode->GetArrayFor("Kids"); |
| if (!pKids) |
| return nullptr; |
| |
| for (size_t i = 0; i < pKids->size(); i++) { |
| const CPDF_Dictionary* pKid = pKids->GetDictAt(i); |
| if (!pKid) |
| continue; |
| |
| const CPDF_Object* pFound = SearchNumberNode(pKid, num); |
| if (pFound) |
| return pFound; |
| } |
| return nullptr; |
| } |
| |
| } // namespace |
| |
| CPDF_NumberTree::CPDF_NumberTree(const CPDF_Dictionary* pRoot) |
| : m_pRoot(pRoot) {} |
| |
| CPDF_NumberTree::~CPDF_NumberTree() = default; |
| |
| const CPDF_Object* CPDF_NumberTree::LookupValue(int num) const { |
| return SearchNumberNode(m_pRoot.Get(), num); |
| } |