blob: 507b97b184ff5c61f4292c8f8b97fabe05dfff32 [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
46 // Set the document information////////////////////////////////////////////
47
48 CPDF_Dictionary* DInfoDict = pDestPDFDoc->GetInfo();
Nico Weber9d8ec5a2015-08-04 13:00:21 -070049 if (!DInfoDict)
50 return FALSE;
51
52 CFX_ByteString producerstr;
53 producerstr.Format("PDFium");
54 DInfoDict->SetAt("Producer", new CPDF_String(producerstr));
55
56 // Set type////////////////////////////////////////////////////////////////
57 CFX_ByteString cbRootType = pNewRoot->GetString("Type", "");
58 if (cbRootType.Equal("")) {
59 pNewRoot->SetAt("Type", new CPDF_Name("Catalog"));
60 }
61
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080062 CPDF_Object* pElement = pNewRoot->GetElement("Pages");
63 CPDF_Dictionary* pNewPages =
64 pElement ? ToDictionary(pElement->GetDirect()) : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070065 if (!pNewPages) {
66 pNewPages = new CPDF_Dictionary;
67 FX_DWORD NewPagesON = pDestPDFDoc->AddIndirectObject(pNewPages);
68 pNewRoot->SetAt("Pages", new CPDF_Reference(pDestPDFDoc, NewPagesON));
69 }
70
71 CFX_ByteString cbPageType = pNewPages->GetString("Type", "");
72 if (cbPageType.Equal("")) {
73 pNewPages->SetAt("Type", new CPDF_Name("Pages"));
74 }
75
76 CPDF_Array* pKeysArray = pNewPages->GetArray("Kids");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080077 if (!pKeysArray) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070078 CPDF_Array* pNewKids = new CPDF_Array;
79 FX_DWORD Kidsobjnum = -1;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080080 Kidsobjnum = pDestPDFDoc->AddIndirectObject(pNewKids);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070081
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080082 pNewPages->SetAt("Kids", new CPDF_Reference(pDestPDFDoc, Kidsobjnum));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070083 pNewPages->SetAt("Count", new CPDF_Number(0));
84 }
85
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080086 return TRUE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070087}
88
Nico Weber9d8ec5a2015-08-04 13:00:21 -070089FX_BOOL CPDF_PageOrganizer::ExportPage(CPDF_Document* pSrcPDFDoc,
90 CFX_WordArray* nPageNum,
91 CPDF_Document* pDestPDFDoc,
92 int nIndex) {
93 int curpage = nIndex;
94
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080095 nonstd::unique_ptr<ObjectNumberMap> pObjNumberMap(new ObjectNumberMap);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070096
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080097 for (int i = 0; i < nPageNum->GetSize(); ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070098 CPDF_Dictionary* pCurPageDict = pDestPDFDoc->CreateNewPage(curpage);
99 CPDF_Dictionary* pSrcPageDict = pSrcPDFDoc->GetPage(nPageNum->GetAt(i) - 1);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800100 if (!pSrcPageDict || !pCurPageDict)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700101 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700102
103 // Clone the page dictionary///////////
104 FX_POSITION SrcPos = pSrcPageDict->GetStartPos();
105 while (SrcPos) {
106 CFX_ByteString cbSrcKeyStr;
107 CPDF_Object* pObj = pSrcPageDict->GetNextElement(SrcPos, cbSrcKeyStr);
108 if (cbSrcKeyStr.Compare(("Type")) && cbSrcKeyStr.Compare(("Parent"))) {
109 if (pCurPageDict->KeyExist(cbSrcKeyStr))
110 pCurPageDict->RemoveAt(cbSrcKeyStr);
111 pCurPageDict->SetAt(cbSrcKeyStr, pObj->Clone());
112 }
113 }
114
115 // inheritable item///////////////////////
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800116 CPDF_Object* pInheritable = nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700117 // 1 MediaBox //required
118 if (!pCurPageDict->KeyExist("MediaBox")) {
119 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "MediaBox");
120 if (!pInheritable) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800121 // Search the "CropBox" from source page dictionary,
122 // if not exists,we take the letter size.
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700123 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800124 if (pInheritable) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700125 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800126 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700127 // Make the default size to be letter size (8.5'x11')
128 CPDF_Array* pArray = new CPDF_Array;
129 pArray->AddNumber(0);
130 pArray->AddNumber(0);
131 pArray->AddNumber(612);
132 pArray->AddNumber(792);
133 pCurPageDict->SetAt("MediaBox", pArray);
134 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800135 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700136 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800137 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700138 }
139 // 2 Resources //required
140 if (!pCurPageDict->KeyExist("Resources")) {
141 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Resources");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800142 if (!pInheritable)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700143 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700144 pCurPageDict->SetAt("Resources", pInheritable->Clone());
145 }
146 // 3 CropBox //Optional
147 if (!pCurPageDict->KeyExist("CropBox")) {
148 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
149 if (pInheritable)
150 pCurPageDict->SetAt("CropBox", pInheritable->Clone());
151 }
152 // 4 Rotate //Optional
153 if (!pCurPageDict->KeyExist("Rotate")) {
154 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Rotate");
155 if (pInheritable)
156 pCurPageDict->SetAt("Rotate", pInheritable->Clone());
157 }
158
159 /////////////////////////////////////////////
160 // Update the reference
161 FX_DWORD dwOldPageObj = pSrcPageDict->GetObjNum();
162 FX_DWORD dwNewPageObj = pCurPageDict->GetObjNum();
163
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800164 (*pObjNumberMap)[dwOldPageObj] = dwNewPageObj;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700165
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800166 UpdateReference(pCurPageDict, pDestPDFDoc, pObjNumberMap.get());
167 ++curpage;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700168 }
169
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700170 return TRUE;
171}
172
173CPDF_Object* CPDF_PageOrganizer::PageDictGetInheritableTag(
174 CPDF_Dictionary* pDict,
175 CFX_ByteString nSrctag) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800176 if (!pDict || nSrctag.IsEmpty())
177 return nullptr;
178 if (!pDict->KeyExist("Parent") || !pDict->KeyExist("Type"))
179 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700180
181 CPDF_Object* pType = pDict->GetElement("Type")->GetDirect();
Dan Sinclair710c9092015-10-21 15:46:10 -0400182 if (!ToName(pType))
183 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700184 if (pType->GetString().Compare("Page"))
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800185 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700186
Dan Sinclairf1251c12015-10-20 16:24:45 -0400187 CPDF_Dictionary* pp = ToDictionary(pDict->GetElement("Parent")->GetDirect());
188 if (!pp)
189 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700190
191 if (pDict->KeyExist((const char*)nSrctag))
192 return pDict->GetElement((const char*)nSrctag);
Dan Sinclairf1251c12015-10-20 16:24:45 -0400193
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700194 while (pp) {
195 if (pp->KeyExist((const char*)nSrctag))
196 return pp->GetElement((const char*)nSrctag);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800197 if (!pp->KeyExist("Parent"))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700198 break;
Dan Sinclairf1251c12015-10-20 16:24:45 -0400199 pp = ToDictionary(pp->GetElement("Parent")->GetDirect());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700200 }
Dan Sinclairf1251c12015-10-20 16:24:45 -0400201 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700202}
203
204FX_BOOL CPDF_PageOrganizer::UpdateReference(CPDF_Object* pObj,
205 CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800206 ObjectNumberMap* pObjNumberMap) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700207 switch (pObj->GetType()) {
208 case PDFOBJ_REFERENCE: {
Dan Sinclairbf81c142015-10-26 16:54:39 -0400209 CPDF_Reference* pReference = pObj->AsReference();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800210 FX_DWORD newobjnum = GetNewObjId(pDoc, pObjNumberMap, pReference);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700211 if (newobjnum == 0)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700212 return FALSE;
Dan Sinclairbf81c142015-10-26 16:54:39 -0400213 pReference->SetRef(pDoc, newobjnum);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700214 break;
215 }
216 case PDFOBJ_DICTIONARY: {
Dan Sinclairf1251c12015-10-20 16:24:45 -0400217 CPDF_Dictionary* pDict = pObj->AsDictionary();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700218
219 FX_POSITION pos = pDict->GetStartPos();
220 while (pos) {
221 CFX_ByteString key("");
222 CPDF_Object* pNextObj = pDict->GetNextElement(pos, key);
223 if (!FXSYS_strcmp(key, "Parent") || !FXSYS_strcmp(key, "Prev") ||
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800224 !FXSYS_strcmp(key, "First")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700225 continue;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800226 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700227 if (pNextObj) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800228 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700229 pDict->RemoveAt(key);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800230 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700231 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800232 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700233 }
234 break;
235 }
236 case PDFOBJ_ARRAY: {
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400237 CPDF_Array* pArray = pObj->AsArray();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700238 FX_DWORD count = pArray->GetCount();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800239 for (FX_DWORD i = 0; i < count; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700240 CPDF_Object* pNextObj = pArray->GetElement(i);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800241 if (!pNextObj)
242 return FALSE;
243 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700244 return FALSE;
245 }
246 break;
247 }
248 case PDFOBJ_STREAM: {
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400249 CPDF_Stream* pStream = pObj->AsStream();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700250 CPDF_Dictionary* pDict = pStream->GetDict();
251 if (pDict) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800252 if (!UpdateReference(pDict, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700253 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800254 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700255 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800256 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700257 break;
258 }
259 default:
260 break;
261 }
262
263 return TRUE;
264}
265
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800266FX_DWORD CPDF_PageOrganizer::GetNewObjId(CPDF_Document* pDoc,
267 ObjectNumberMap* pObjNumberMap,
268 CPDF_Reference* pRef) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700269 if (!pRef)
270 return 0;
271
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800272 FX_DWORD dwObjnum = pRef->GetRefObjNum();
273 FX_DWORD dwNewObjNum = 0;
274 const auto it = pObjNumberMap->find(dwObjnum);
275 if (it != pObjNumberMap->end())
276 dwNewObjNum = it->second;
277 if (dwNewObjNum)
278 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700279
280 CPDF_Object* pDirect = pRef->GetDirect();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800281 if (!pDirect)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700282 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700283
284 CPDF_Object* pClone = pDirect->Clone();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800285 if (!pClone)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700286 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700287
Dan Sinclairf1251c12015-10-20 16:24:45 -0400288 if (CPDF_Dictionary* pDictClone = pClone->AsDictionary()) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700289 if (pDictClone->KeyExist("Type")) {
290 CFX_ByteString strType = pDictClone->GetString("Type");
291 if (!FXSYS_stricmp(strType, "Pages")) {
292 pDictClone->Release();
293 return 4;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800294 }
295 if (!FXSYS_stricmp(strType, "Page")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700296 pDictClone->Release();
297 return 0;
298 }
299 }
300 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700301 dwNewObjNum = pDoc->AddIndirectObject(pClone);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800302 (*pObjNumberMap)[dwObjnum] = dwNewObjNum;
303 if (!UpdateReference(pClone, pDoc, pObjNumberMap)) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700304 pClone->Release();
305 return 0;
306 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800307 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700308}
309
310FPDF_BOOL ParserPageRangeString(CFX_ByteString rangstring,
311 CFX_WordArray* pageArray,
312 int nCount) {
313 if (rangstring.GetLength() != 0) {
314 rangstring.Remove(' ');
315 int nLength = rangstring.GetLength();
316 CFX_ByteString cbCompareString("0123456789-,");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800317 for (int i = 0; i < nLength; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700318 if (cbCompareString.Find(rangstring[i]) == -1)
319 return FALSE;
320 }
321 CFX_ByteString cbMidRange;
322 int nStringFrom = 0;
323 int nStringTo = 0;
324 while (nStringTo < nLength) {
325 nStringTo = rangstring.Find(',', nStringFrom);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800326 if (nStringTo == -1)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700327 nStringTo = nLength;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700328 cbMidRange = rangstring.Mid(nStringFrom, nStringTo - nStringFrom);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700329 int nMid = cbMidRange.Find('-');
330 if (nMid == -1) {
331 long lPageNum = atol(cbMidRange);
332 if (lPageNum <= 0 || lPageNum > nCount)
333 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700334 pageArray->Add((FX_WORD)lPageNum);
335 } else {
336 int nStartPageNum = atol(cbMidRange.Mid(0, nMid));
337 if (nStartPageNum == 0)
338 return FALSE;
339
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800340 ++nMid;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700341 int nEnd = cbMidRange.GetLength() - nMid;
342 if (nEnd == 0)
343 return FALSE;
344
345 int nEndPageNum = atol(cbMidRange.Mid(nMid, nEnd));
346 if (nStartPageNum < 0 || nStartPageNum > nEndPageNum ||
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800347 nEndPageNum > nCount) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700348 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800349 }
350 for (int i = nStartPageNum; i <= nEndPageNum; ++i) {
351 pageArray->Add(i);
352 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700353 }
354 nStringFrom = nStringTo + 1;
355 }
356 }
357 return TRUE;
358}
359
360DLLEXPORT FPDF_BOOL STDCALL FPDF_ImportPages(FPDF_DOCUMENT dest_doc,
361 FPDF_DOCUMENT src_doc,
362 FPDF_BYTESTRING pagerange,
363 int index) {
Tom Sepez471a1032015-10-15 16:17:18 -0700364 CPDF_Document* pDestDoc = CPDFDocumentFromFPDFDocument(dest_doc);
365 if (!dest_doc)
366 return FALSE;
367
368 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
369 if (!pSrcDoc)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700370 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800371
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700372 CFX_WordArray pageArray;
Tom Sepez471a1032015-10-15 16:17:18 -0700373 int nCount = pSrcDoc->GetPageCount();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700374 if (pagerange) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800375 if (!ParserPageRangeString(pagerange, &pageArray, nCount))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700376 return FALSE;
377 } else {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800378 for (int i = 1; i <= nCount; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700379 pageArray.Add(i);
380 }
381 }
382
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700383 CPDF_PageOrganizer pageOrg;
Tom Sepez471a1032015-10-15 16:17:18 -0700384 pageOrg.PDFDocInit(pDestDoc, pSrcDoc);
385 return pageOrg.ExportPage(pSrcDoc, &pageArray, pDestDoc, index);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700386}
387
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700388DLLEXPORT FPDF_BOOL STDCALL FPDF_CopyViewerPreferences(FPDF_DOCUMENT dest_doc,
389 FPDF_DOCUMENT src_doc) {
Tom Sepez471a1032015-10-15 16:17:18 -0700390 CPDF_Document* pDstDoc = CPDFDocumentFromFPDFDocument(dest_doc);
391 if (!pDstDoc)
392 return FALSE;
393
394 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
395 if (!pSrcDoc)
396 return FALSE;
397
398 CPDF_Dictionary* pSrcDict = pSrcDoc->GetRoot();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700399 pSrcDict = pSrcDict->GetDict(FX_BSTRC("ViewerPreferences"));
400 if (!pSrcDict)
401 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700402
403 CPDF_Dictionary* pDstDict = pDstDoc->GetRoot();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700404 if (!pDstDict)
405 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700406
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700407 pDstDict->SetAt(FX_BSTRC("ViewerPreferences"), pSrcDict->Clone(TRUE));
408 return TRUE;
409}