// 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_pagelabel.h" #include #include #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_document.h" #include "core/fpdfapi/parser/fpdf_parser_decode.h" #include "core/fpdfdoc/cpdf_numbertree.h" #include "core/fxcrt/stl_util.h" namespace { WideString MakeRoman(int num) { constexpr auto kArabic = fxcrt::ToArray( {1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1}); const auto kRoman = fxcrt::ToArray( {L"m", L"cm", L"d", L"cd", L"c", L"xc", L"l", L"xl", L"x", L"ix", L"v", L"iv", L"i"}); constexpr int kMaxNum = 1000000; num %= kMaxNum; int i = 0; WideString result; result.Reserve(10); // Should cover most use cases. while (num > 0) { while (num >= kArabic[i]) { num -= kArabic[i]; result += kRoman[i]; } ++i; } return result; } WideString MakeLetters(int num) { if (num == 0) { return WideString(); } constexpr int kMaxCount = 1000; constexpr int kLetterCount = 26; --num; const int count = (num / kLetterCount + 1) % kMaxCount; const wchar_t ch = L'a' + num % kLetterCount; WideString result; { auto result_span = result.GetBuffer(count); fxcrt::Fill(result_span, ch); result.ReleaseBuffer(count); } return result; } WideString GetLabelNumPortion(int num, const ByteString& style) { if (style.IsEmpty()) { return WideString(); } if (style == "D") { return WideString::FormatInteger(num); } if (style == "R") { WideString result = MakeRoman(num); result.MakeUpper(); return result; } if (style == "r") { return MakeRoman(num); } if (style == "A") { WideString result = MakeLetters(num); result.MakeUpper(); return result; } if (style == "a") { return MakeLetters(num); } return WideString(); } } // namespace CPDF_PageLabel::CPDF_PageLabel(CPDF_Document* doc) : doc_(doc) {} CPDF_PageLabel::~CPDF_PageLabel() = default; std::optional CPDF_PageLabel::GetLabel(int page_index) const { if (!doc_) { return std::nullopt; } if (page_index < 0 || page_index >= doc_->GetPageCount()) { return std::nullopt; } const CPDF_Dictionary* root_dict = doc_->GetRoot(); if (!root_dict) { return std::nullopt; } RetainPtr labels_dict = root_dict->GetDictFor("PageLabels"); if (!labels_dict) { return std::nullopt; } CPDF_NumberTree number_tree(std::move(labels_dict)); RetainPtr label_value; std::optional lower_bound = number_tree.GetLowerBound(page_index); if (lower_bound.has_value()) { label_value = lower_bound.value().value; } const CPDF_Dictionary* label_dict = label_value ? label_value->GetDirect()->AsDictionary() : nullptr; if (!label_dict) { return WideString::FormatInteger(page_index + 1); } WideString label; if (label_dict->KeyExist("P")) { label = label_dict->GetUnicodeTextFor("P"); } ByteString style = label_dict->GetByteStringFor("S", ByteString()); int label_number = page_index - lower_bound.value().key + label_dict->GetIntegerFor("St", 1); label += GetLabelNumPortion(label_number, style); return label; }