blob: 2fce7adfc43787649efa9a786e2d5177754f073e [file] [log] [blame]
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -07001// Copyright 2014 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.
Lei Zhanga6d9f0e2015-06-13 00:48:38 -07004
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -07005// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
Lei Zhangb4e7f302015-11-06 15:52:32 -08007#include "public/fpdf_ppo.h"
8
Lei Zhangbde53d22015-11-12 22:21:30 -08009#include "fpdfsdk/include/fsdk_define.h"
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080010#include "third_party/base/nonstd_unique_ptr.h"
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070011
Nico Weber9d8ec5a2015-08-04 13:00:21 -070012class CPDF_PageOrganizer {
13 public:
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080014 using ObjectNumberMap = std::map<FX_DWORD, FX_DWORD>;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070015 CPDF_PageOrganizer();
16 ~CPDF_PageOrganizer();
Lei Zhanga6d9f0e2015-06-13 00:48:38 -070017
Nico Weber9d8ec5a2015-08-04 13:00:21 -070018 FX_BOOL PDFDocInit(CPDF_Document* pDestPDFDoc, CPDF_Document* pSrcPDFDoc);
19 FX_BOOL ExportPage(CPDF_Document* pSrcPDFDoc,
20 CFX_WordArray* nPageNum,
21 CPDF_Document* pDestPDFDoc,
22 int nIndex);
23 CPDF_Object* PageDictGetInheritableTag(CPDF_Dictionary* pDict,
24 CFX_ByteString nSrctag);
25 FX_BOOL UpdateReference(CPDF_Object* pObj,
26 CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080027 ObjectNumberMap* pObjNumberMap);
28 FX_DWORD GetNewObjId(CPDF_Document* pDoc,
29 ObjectNumberMap* pObjNumberMap,
30 CPDF_Reference* pRef);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070031};
32
Nico Weber9d8ec5a2015-08-04 13:00:21 -070033CPDF_PageOrganizer::CPDF_PageOrganizer() {}
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070034
Nico Weber9d8ec5a2015-08-04 13:00:21 -070035CPDF_PageOrganizer::~CPDF_PageOrganizer() {}
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070036
Nico Weber9d8ec5a2015-08-04 13:00:21 -070037FX_BOOL CPDF_PageOrganizer::PDFDocInit(CPDF_Document* pDestPDFDoc,
38 CPDF_Document* pSrcPDFDoc) {
39 if (!pDestPDFDoc || !pSrcPDFDoc)
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080040 return FALSE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070041
Nico Weber9d8ec5a2015-08-04 13:00:21 -070042 CPDF_Dictionary* pNewRoot = pDestPDFDoc->GetRoot();
43 if (!pNewRoot)
Tom Sepez2f2ffec2015-07-23 14:42:09 -070044 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070045
Lei Zhang4880d1a2015-12-18 17:05:11 -080046 // Set the document information
Nico Weber9d8ec5a2015-08-04 13:00:21 -070047 CPDF_Dictionary* DInfoDict = pDestPDFDoc->GetInfo();
Nico Weber9d8ec5a2015-08-04 13:00:21 -070048 if (!DInfoDict)
49 return FALSE;
50
51 CFX_ByteString producerstr;
52 producerstr.Format("PDFium");
Lei Zhang4880d1a2015-12-18 17:05:11 -080053 DInfoDict->SetAt("Producer", new CPDF_String(producerstr, FALSE));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070054
Lei Zhang4880d1a2015-12-18 17:05:11 -080055 // Set type
Nico Weber9d8ec5a2015-08-04 13:00:21 -070056 CFX_ByteString cbRootType = pNewRoot->GetString("Type", "");
57 if (cbRootType.Equal("")) {
58 pNewRoot->SetAt("Type", new CPDF_Name("Catalog"));
59 }
60
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080061 CPDF_Object* pElement = pNewRoot->GetElement("Pages");
62 CPDF_Dictionary* pNewPages =
63 pElement ? ToDictionary(pElement->GetDirect()) : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070064 if (!pNewPages) {
65 pNewPages = new CPDF_Dictionary;
66 FX_DWORD NewPagesON = pDestPDFDoc->AddIndirectObject(pNewPages);
67 pNewRoot->SetAt("Pages", new CPDF_Reference(pDestPDFDoc, NewPagesON));
68 }
69
70 CFX_ByteString cbPageType = pNewPages->GetString("Type", "");
71 if (cbPageType.Equal("")) {
72 pNewPages->SetAt("Type", new CPDF_Name("Pages"));
73 }
74
75 CPDF_Array* pKeysArray = pNewPages->GetArray("Kids");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080076 if (!pKeysArray) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070077 CPDF_Array* pNewKids = new CPDF_Array;
78 FX_DWORD Kidsobjnum = -1;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080079 Kidsobjnum = pDestPDFDoc->AddIndirectObject(pNewKids);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070080
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080081 pNewPages->SetAt("Kids", new CPDF_Reference(pDestPDFDoc, Kidsobjnum));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070082 pNewPages->SetAt("Count", new CPDF_Number(0));
83 }
84
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080085 return TRUE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070086}
87
Nico Weber9d8ec5a2015-08-04 13:00:21 -070088FX_BOOL CPDF_PageOrganizer::ExportPage(CPDF_Document* pSrcPDFDoc,
89 CFX_WordArray* nPageNum,
90 CPDF_Document* pDestPDFDoc,
91 int nIndex) {
92 int curpage = nIndex;
93
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080094 nonstd::unique_ptr<ObjectNumberMap> pObjNumberMap(new ObjectNumberMap);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070095
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080096 for (int i = 0; i < nPageNum->GetSize(); ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070097 CPDF_Dictionary* pCurPageDict = pDestPDFDoc->CreateNewPage(curpage);
98 CPDF_Dictionary* pSrcPageDict = pSrcPDFDoc->GetPage(nPageNum->GetAt(i) - 1);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080099 if (!pSrcPageDict || !pCurPageDict)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700100 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700101
Lei Zhang4880d1a2015-12-18 17:05:11 -0800102 // Clone the page dictionary
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700103 FX_POSITION SrcPos = pSrcPageDict->GetStartPos();
104 while (SrcPos) {
105 CFX_ByteString cbSrcKeyStr;
106 CPDF_Object* pObj = pSrcPageDict->GetNextElement(SrcPos, cbSrcKeyStr);
107 if (cbSrcKeyStr.Compare(("Type")) && cbSrcKeyStr.Compare(("Parent"))) {
108 if (pCurPageDict->KeyExist(cbSrcKeyStr))
109 pCurPageDict->RemoveAt(cbSrcKeyStr);
110 pCurPageDict->SetAt(cbSrcKeyStr, pObj->Clone());
111 }
112 }
113
Lei Zhang4880d1a2015-12-18 17:05:11 -0800114 // inheritable item
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800115 CPDF_Object* pInheritable = nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700116 // 1 MediaBox //required
117 if (!pCurPageDict->KeyExist("MediaBox")) {
118 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "MediaBox");
119 if (!pInheritable) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800120 // Search the "CropBox" from source page dictionary,
121 // if not exists,we take the letter size.
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700122 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800123 if (pInheritable) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700124 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800125 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700126 // Make the default size to be letter size (8.5'x11')
127 CPDF_Array* pArray = new CPDF_Array;
128 pArray->AddNumber(0);
129 pArray->AddNumber(0);
130 pArray->AddNumber(612);
131 pArray->AddNumber(792);
132 pCurPageDict->SetAt("MediaBox", pArray);
133 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800134 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700135 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800136 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700137 }
138 // 2 Resources //required
139 if (!pCurPageDict->KeyExist("Resources")) {
140 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Resources");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800141 if (!pInheritable)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700142 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700143 pCurPageDict->SetAt("Resources", pInheritable->Clone());
144 }
145 // 3 CropBox //Optional
146 if (!pCurPageDict->KeyExist("CropBox")) {
147 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
148 if (pInheritable)
149 pCurPageDict->SetAt("CropBox", pInheritable->Clone());
150 }
151 // 4 Rotate //Optional
152 if (!pCurPageDict->KeyExist("Rotate")) {
153 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Rotate");
154 if (pInheritable)
155 pCurPageDict->SetAt("Rotate", pInheritable->Clone());
156 }
157
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700158 // Update the reference
159 FX_DWORD dwOldPageObj = pSrcPageDict->GetObjNum();
160 FX_DWORD dwNewPageObj = pCurPageDict->GetObjNum();
161
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800162 (*pObjNumberMap)[dwOldPageObj] = dwNewPageObj;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700163
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800164 UpdateReference(pCurPageDict, pDestPDFDoc, pObjNumberMap.get());
165 ++curpage;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700166 }
167
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700168 return TRUE;
169}
170
171CPDF_Object* CPDF_PageOrganizer::PageDictGetInheritableTag(
172 CPDF_Dictionary* pDict,
173 CFX_ByteString nSrctag) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800174 if (!pDict || nSrctag.IsEmpty())
175 return nullptr;
176 if (!pDict->KeyExist("Parent") || !pDict->KeyExist("Type"))
177 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700178
179 CPDF_Object* pType = pDict->GetElement("Type")->GetDirect();
Dan Sinclair710c9092015-10-21 15:46:10 -0400180 if (!ToName(pType))
181 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700182 if (pType->GetString().Compare("Page"))
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800183 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700184
Dan Sinclairf1251c12015-10-20 16:24:45 -0400185 CPDF_Dictionary* pp = ToDictionary(pDict->GetElement("Parent")->GetDirect());
186 if (!pp)
187 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700188
189 if (pDict->KeyExist((const char*)nSrctag))
190 return pDict->GetElement((const char*)nSrctag);
Dan Sinclairf1251c12015-10-20 16:24:45 -0400191
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700192 while (pp) {
193 if (pp->KeyExist((const char*)nSrctag))
194 return pp->GetElement((const char*)nSrctag);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800195 if (!pp->KeyExist("Parent"))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700196 break;
Dan Sinclairf1251c12015-10-20 16:24:45 -0400197 pp = ToDictionary(pp->GetElement("Parent")->GetDirect());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700198 }
Dan Sinclairf1251c12015-10-20 16:24:45 -0400199 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700200}
201
202FX_BOOL CPDF_PageOrganizer::UpdateReference(CPDF_Object* pObj,
203 CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800204 ObjectNumberMap* pObjNumberMap) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700205 switch (pObj->GetType()) {
206 case PDFOBJ_REFERENCE: {
Dan Sinclairbf81c142015-10-26 16:54:39 -0400207 CPDF_Reference* pReference = pObj->AsReference();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800208 FX_DWORD newobjnum = GetNewObjId(pDoc, pObjNumberMap, pReference);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700209 if (newobjnum == 0)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700210 return FALSE;
Dan Sinclairbf81c142015-10-26 16:54:39 -0400211 pReference->SetRef(pDoc, newobjnum);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700212 break;
213 }
214 case PDFOBJ_DICTIONARY: {
Dan Sinclairf1251c12015-10-20 16:24:45 -0400215 CPDF_Dictionary* pDict = pObj->AsDictionary();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700216
217 FX_POSITION pos = pDict->GetStartPos();
218 while (pos) {
219 CFX_ByteString key("");
220 CPDF_Object* pNextObj = pDict->GetNextElement(pos, key);
221 if (!FXSYS_strcmp(key, "Parent") || !FXSYS_strcmp(key, "Prev") ||
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800222 !FXSYS_strcmp(key, "First")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700223 continue;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800224 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700225 if (pNextObj) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800226 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700227 pDict->RemoveAt(key);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800228 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700229 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800230 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700231 }
232 break;
233 }
234 case PDFOBJ_ARRAY: {
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400235 CPDF_Array* pArray = pObj->AsArray();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700236 FX_DWORD count = pArray->GetCount();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800237 for (FX_DWORD i = 0; i < count; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700238 CPDF_Object* pNextObj = pArray->GetElement(i);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800239 if (!pNextObj)
240 return FALSE;
241 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700242 return FALSE;
243 }
244 break;
245 }
246 case PDFOBJ_STREAM: {
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400247 CPDF_Stream* pStream = pObj->AsStream();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700248 CPDF_Dictionary* pDict = pStream->GetDict();
249 if (pDict) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800250 if (!UpdateReference(pDict, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700251 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800252 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700253 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800254 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700255 break;
256 }
257 default:
258 break;
259 }
260
261 return TRUE;
262}
263
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800264FX_DWORD CPDF_PageOrganizer::GetNewObjId(CPDF_Document* pDoc,
265 ObjectNumberMap* pObjNumberMap,
266 CPDF_Reference* pRef) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700267 if (!pRef)
268 return 0;
269
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800270 FX_DWORD dwObjnum = pRef->GetRefObjNum();
271 FX_DWORD dwNewObjNum = 0;
272 const auto it = pObjNumberMap->find(dwObjnum);
273 if (it != pObjNumberMap->end())
274 dwNewObjNum = it->second;
275 if (dwNewObjNum)
276 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700277
278 CPDF_Object* pDirect = pRef->GetDirect();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800279 if (!pDirect)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700280 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700281
282 CPDF_Object* pClone = pDirect->Clone();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800283 if (!pClone)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700284 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700285
Dan Sinclairf1251c12015-10-20 16:24:45 -0400286 if (CPDF_Dictionary* pDictClone = pClone->AsDictionary()) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700287 if (pDictClone->KeyExist("Type")) {
288 CFX_ByteString strType = pDictClone->GetString("Type");
289 if (!FXSYS_stricmp(strType, "Pages")) {
290 pDictClone->Release();
291 return 4;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800292 }
293 if (!FXSYS_stricmp(strType, "Page")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700294 pDictClone->Release();
295 return 0;
296 }
297 }
298 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700299 dwNewObjNum = pDoc->AddIndirectObject(pClone);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800300 (*pObjNumberMap)[dwObjnum] = dwNewObjNum;
301 if (!UpdateReference(pClone, pDoc, pObjNumberMap)) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700302 pClone->Release();
303 return 0;
304 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800305 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700306}
307
308FPDF_BOOL ParserPageRangeString(CFX_ByteString rangstring,
309 CFX_WordArray* pageArray,
310 int nCount) {
311 if (rangstring.GetLength() != 0) {
312 rangstring.Remove(' ');
313 int nLength = rangstring.GetLength();
314 CFX_ByteString cbCompareString("0123456789-,");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800315 for (int i = 0; i < nLength; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700316 if (cbCompareString.Find(rangstring[i]) == -1)
317 return FALSE;
318 }
319 CFX_ByteString cbMidRange;
320 int nStringFrom = 0;
321 int nStringTo = 0;
322 while (nStringTo < nLength) {
323 nStringTo = rangstring.Find(',', nStringFrom);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800324 if (nStringTo == -1)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700325 nStringTo = nLength;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700326 cbMidRange = rangstring.Mid(nStringFrom, nStringTo - nStringFrom);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700327 int nMid = cbMidRange.Find('-');
328 if (nMid == -1) {
329 long lPageNum = atol(cbMidRange);
330 if (lPageNum <= 0 || lPageNum > nCount)
331 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700332 pageArray->Add((FX_WORD)lPageNum);
333 } else {
334 int nStartPageNum = atol(cbMidRange.Mid(0, nMid));
335 if (nStartPageNum == 0)
336 return FALSE;
337
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800338 ++nMid;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700339 int nEnd = cbMidRange.GetLength() - nMid;
340 if (nEnd == 0)
341 return FALSE;
342
343 int nEndPageNum = atol(cbMidRange.Mid(nMid, nEnd));
344 if (nStartPageNum < 0 || nStartPageNum > nEndPageNum ||
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800345 nEndPageNum > nCount) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700346 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800347 }
348 for (int i = nStartPageNum; i <= nEndPageNum; ++i) {
349 pageArray->Add(i);
350 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700351 }
352 nStringFrom = nStringTo + 1;
353 }
354 }
355 return TRUE;
356}
357
358DLLEXPORT FPDF_BOOL STDCALL FPDF_ImportPages(FPDF_DOCUMENT dest_doc,
359 FPDF_DOCUMENT src_doc,
360 FPDF_BYTESTRING pagerange,
361 int index) {
Tom Sepez471a1032015-10-15 16:17:18 -0700362 CPDF_Document* pDestDoc = CPDFDocumentFromFPDFDocument(dest_doc);
363 if (!dest_doc)
364 return FALSE;
365
366 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
367 if (!pSrcDoc)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700368 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800369
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700370 CFX_WordArray pageArray;
Tom Sepez471a1032015-10-15 16:17:18 -0700371 int nCount = pSrcDoc->GetPageCount();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700372 if (pagerange) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800373 if (!ParserPageRangeString(pagerange, &pageArray, nCount))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700374 return FALSE;
375 } else {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800376 for (int i = 1; i <= nCount; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700377 pageArray.Add(i);
378 }
379 }
380
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700381 CPDF_PageOrganizer pageOrg;
Tom Sepez471a1032015-10-15 16:17:18 -0700382 pageOrg.PDFDocInit(pDestDoc, pSrcDoc);
383 return pageOrg.ExportPage(pSrcDoc, &pageArray, pDestDoc, index);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700384}
385
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700386DLLEXPORT FPDF_BOOL STDCALL FPDF_CopyViewerPreferences(FPDF_DOCUMENT dest_doc,
387 FPDF_DOCUMENT src_doc) {
Tom Sepez471a1032015-10-15 16:17:18 -0700388 CPDF_Document* pDstDoc = CPDFDocumentFromFPDFDocument(dest_doc);
389 if (!pDstDoc)
390 return FALSE;
391
392 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
393 if (!pSrcDoc)
394 return FALSE;
395
396 CPDF_Dictionary* pSrcDict = pSrcDoc->GetRoot();
Lei Zhangd983b092015-12-14 16:58:33 -0800397 pSrcDict = pSrcDict->GetDict("ViewerPreferences");
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700398 if (!pSrcDict)
399 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700400
401 CPDF_Dictionary* pDstDict = pDstDoc->GetRoot();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700402 if (!pDstDict)
403 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700404
Lei Zhangd983b092015-12-14 16:58:33 -0800405 pDstDict->SetAt("ViewerPreferences", pSrcDict->Clone(TRUE));
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700406 return TRUE;
407}