1 // Copyright 2016 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6 
7 #include "core/fpdfdoc/cpdf_pagelabel.h"
8 
9 #include "core/fpdfapi/parser/cpdf_dictionary.h"
10 #include "core/fpdfapi/parser/cpdf_document.h"
11 #include "core/fpdfapi/parser/fpdf_parser_decode.h"
12 #include "core/fpdfdoc/cpdf_numbertree.h"
13 
14 namespace {
15 
MakeRoman(int num)16 WideString MakeRoman(int num) {
17   const int kArabic[] = {1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1};
18   const WideString kRoman[] = {L"m",  L"cm", L"d",  L"cd", L"c",  L"xc", L"l",
19                                L"xl", L"x",  L"ix", L"v",  L"iv", L"i"};
20   const int kMaxNum = 1000000;
21 
22   num %= kMaxNum;
23   int i = 0;
24   WideString wsRomanNumber;
25   while (num > 0) {
26     while (num >= kArabic[i]) {
27       num = num - kArabic[i];
28       wsRomanNumber += kRoman[i];
29     }
30     i = i + 1;
31   }
32   return wsRomanNumber;
33 }
34 
MakeLetters(int num)35 WideString MakeLetters(int num) {
36   if (num == 0)
37     return WideString();
38 
39   WideString wsLetters;
40   const int nMaxCount = 1000;
41   const int nLetterCount = 26;
42   --num;
43 
44   int count = num / nLetterCount + 1;
45   count %= nMaxCount;
46   wchar_t ch = L'a' + num % nLetterCount;
47   for (int i = 0; i < count; i++)
48     wsLetters += ch;
49   return wsLetters;
50 }
51 
GetLabelNumPortion(int num,const ByteString & bsStyle)52 WideString GetLabelNumPortion(int num, const ByteString& bsStyle) {
53   if (bsStyle.IsEmpty())
54     return L"";
55   if (bsStyle == "D")
56     return WideString::Format(L"%d", num);
57   if (bsStyle == "R") {
58     WideString wsNumPortion = MakeRoman(num);
59     wsNumPortion.MakeUpper();
60     return wsNumPortion;
61   }
62   if (bsStyle == "r")
63     return MakeRoman(num);
64   if (bsStyle == "A") {
65     WideString wsNumPortion = MakeLetters(num);
66     wsNumPortion.MakeUpper();
67     return wsNumPortion;
68   }
69   if (bsStyle == "a")
70     return MakeLetters(num);
71   return L"";
72 }
73 
74 }  // namespace
75 
CPDF_PageLabel(CPDF_Document * pDocument)76 CPDF_PageLabel::CPDF_PageLabel(CPDF_Document* pDocument)
77     : m_pDocument(pDocument) {}
78 
~CPDF_PageLabel()79 CPDF_PageLabel::~CPDF_PageLabel() {}
80 
GetLabel(int nPage) const81 Optional<WideString> CPDF_PageLabel::GetLabel(int nPage) const {
82   if (!m_pDocument)
83     return {};
84 
85   if (nPage < 0 || nPage >= m_pDocument->GetPageCount())
86     return {};
87 
88   const CPDF_Dictionary* pPDFRoot = m_pDocument->GetRoot();
89   if (!pPDFRoot)
90     return {};
91 
92   CPDF_Dictionary* pLabels = pPDFRoot->GetDictFor("PageLabels");
93   if (!pLabels)
94     return {};
95 
96   CPDF_NumberTree numberTree(pLabels);
97   CPDF_Object* pValue = nullptr;
98   int n = nPage;
99   while (n >= 0) {
100     pValue = numberTree.LookupValue(n);
101     if (pValue)
102       break;
103     n--;
104   }
105 
106   WideString label;
107   if (pValue) {
108     pValue = pValue->GetDirect();
109     if (CPDF_Dictionary* pLabel = pValue->AsDictionary()) {
110       if (pLabel->KeyExist("P"))
111         label += pLabel->GetUnicodeTextFor("P");
112 
113       ByteString bsNumberingStyle = pLabel->GetStringFor("S", "");
114       int nLabelNum = nPage - n + pLabel->GetIntegerFor("St", 1);
115       WideString wsNumPortion = GetLabelNumPortion(nLabelNum, bsNumberingStyle);
116       label += wsNumPortion;
117       return {label};
118     }
119   }
120   label = WideString::Format(L"%d", nPage + 1);
121   return {label};
122 }
123 
GetPageByLabel(const ByteStringView & bsLabel) const124 int32_t CPDF_PageLabel::GetPageByLabel(const ByteStringView& bsLabel) const {
125   if (!m_pDocument)
126     return -1;
127 
128   const CPDF_Dictionary* pPDFRoot = m_pDocument->GetRoot();
129   if (!pPDFRoot)
130     return -1;
131 
132   int nPages = m_pDocument->GetPageCount();
133   for (int i = 0; i < nPages; i++) {
134     Optional<WideString> str = GetLabel(i);
135     if (!str.has_value())
136       continue;
137     if (PDF_EncodeText(str.value()).Compare(bsLabel))
138       return i;
139   }
140 
141   int nPage = FXSYS_atoi(ByteString(bsLabel).c_str());  // NUL terminate.
142   return nPage > 0 && nPage <= nPages ? nPage : -1;
143 }
144 
GetPageByLabel(const WideStringView & wsLabel) const145 int32_t CPDF_PageLabel::GetPageByLabel(const WideStringView& wsLabel) const {
146   // TODO(tsepez): check usage of c_str() below.
147   return GetPageByLabel(
148       PDF_EncodeText(wsLabel.unterminated_c_str()).AsStringView());
149 }
150