blob: 300e7ba36e01fb6b652259c57e1294989f134465 [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
Dan Sinclair3ebd1212016-03-09 09:59:23 -05009#include <map>
Lei Zhangaa8bf7e2015-12-24 19:13:32 -080010#include <memory>
Dan Sinclair3ebd1212016-03-09 09:59:23 -050011#include <vector>
Lei Zhangaa8bf7e2015-12-24 19:13:32 -080012
Dan Sinclairaa403d32016-03-15 14:57:22 -040013#include "core/fpdfapi/fpdf_parser/include/cpdf_array.h"
14#include "core/fpdfapi/fpdf_parser/include/cpdf_document.h"
15#include "core/fpdfapi/fpdf_parser/include/cpdf_name.h"
16#include "core/fpdfapi/fpdf_parser/include/cpdf_number.h"
17#include "core/fpdfapi/fpdf_parser/include/cpdf_reference.h"
Dan Sinclair584b1e62016-03-21 09:15:45 -040018#include "core/fpdfapi/fpdf_parser/include/cpdf_stream.h"
Dan Sinclairaa403d32016-03-15 14:57:22 -040019#include "core/fpdfapi/fpdf_parser/include/cpdf_string.h"
Lei Zhangbde53d22015-11-12 22:21:30 -080020#include "fpdfsdk/include/fsdk_define.h"
Tom Sepez11d93552016-02-09 09:55:54 -080021#include "third_party/base/stl_util.h"
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070022
Nico Weber9d8ec5a2015-08-04 13:00:21 -070023class CPDF_PageOrganizer {
24 public:
tsepezc3255f52016-03-25 14:52:27 -070025 using ObjectNumberMap = std::map<uint32_t, uint32_t>;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070026 CPDF_PageOrganizer();
27 ~CPDF_PageOrganizer();
Lei Zhanga6d9f0e2015-06-13 00:48:38 -070028
Nico Weber9d8ec5a2015-08-04 13:00:21 -070029 FX_BOOL PDFDocInit(CPDF_Document* pDestPDFDoc, CPDF_Document* pSrcPDFDoc);
30 FX_BOOL ExportPage(CPDF_Document* pSrcPDFDoc,
Tom Sepez62a70f92016-03-21 15:00:20 -070031 std::vector<uint16_t>* pPageNums,
Nico Weber9d8ec5a2015-08-04 13:00:21 -070032 CPDF_Document* pDestPDFDoc,
33 int nIndex);
34 CPDF_Object* PageDictGetInheritableTag(CPDF_Dictionary* pDict,
35 CFX_ByteString nSrctag);
36 FX_BOOL UpdateReference(CPDF_Object* pObj,
37 CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080038 ObjectNumberMap* pObjNumberMap);
tsepezc3255f52016-03-25 14:52:27 -070039 uint32_t GetNewObjId(CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080040 ObjectNumberMap* pObjNumberMap,
41 CPDF_Reference* pRef);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070042};
43
Nico Weber9d8ec5a2015-08-04 13:00:21 -070044CPDF_PageOrganizer::CPDF_PageOrganizer() {}
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070045
Nico Weber9d8ec5a2015-08-04 13:00:21 -070046CPDF_PageOrganizer::~CPDF_PageOrganizer() {}
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070047
Nico Weber9d8ec5a2015-08-04 13:00:21 -070048FX_BOOL CPDF_PageOrganizer::PDFDocInit(CPDF_Document* pDestPDFDoc,
49 CPDF_Document* pSrcPDFDoc) {
50 if (!pDestPDFDoc || !pSrcPDFDoc)
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080051 return FALSE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070052
Nico Weber9d8ec5a2015-08-04 13:00:21 -070053 CPDF_Dictionary* pNewRoot = pDestPDFDoc->GetRoot();
54 if (!pNewRoot)
Tom Sepez2f2ffec2015-07-23 14:42:09 -070055 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070056
Nico Weber9d8ec5a2015-08-04 13:00:21 -070057 CPDF_Dictionary* DInfoDict = pDestPDFDoc->GetInfo();
Nico Weber9d8ec5a2015-08-04 13:00:21 -070058 if (!DInfoDict)
59 return FALSE;
60
61 CFX_ByteString producerstr;
62 producerstr.Format("PDFium");
Lei Zhang4880d1a2015-12-18 17:05:11 -080063 DInfoDict->SetAt("Producer", new CPDF_String(producerstr, FALSE));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070064
Wei Li9b761132016-01-29 15:44:20 -080065 CFX_ByteString cbRootType = pNewRoot->GetStringBy("Type", "");
tsepez9f2970c2016-04-01 10:23:04 -070066 if (cbRootType.IsEmpty())
Nico Weber9d8ec5a2015-08-04 13:00:21 -070067 pNewRoot->SetAt("Type", new CPDF_Name("Catalog"));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070068
tsepezbd567552016-03-29 14:51:50 -070069 CPDF_Object* pElement = pNewRoot->GetObjectBy("Pages");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080070 CPDF_Dictionary* pNewPages =
71 pElement ? ToDictionary(pElement->GetDirect()) : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -070072 if (!pNewPages) {
73 pNewPages = new CPDF_Dictionary;
tsepezc3255f52016-03-25 14:52:27 -070074 uint32_t NewPagesON = pDestPDFDoc->AddIndirectObject(pNewPages);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070075 pNewRoot->SetAt("Pages", new CPDF_Reference(pDestPDFDoc, NewPagesON));
76 }
77
Wei Li9b761132016-01-29 15:44:20 -080078 CFX_ByteString cbPageType = pNewPages->GetStringBy("Type", "");
tsepez9f2970c2016-04-01 10:23:04 -070079 if (cbPageType == "") {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070080 pNewPages->SetAt("Type", new CPDF_Name("Pages"));
81 }
82
Wei Li9b761132016-01-29 15:44:20 -080083 CPDF_Array* pKeysArray = pNewPages->GetArrayBy("Kids");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080084 if (!pKeysArray) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -070085 CPDF_Array* pNewKids = new CPDF_Array;
tsepezc3255f52016-03-25 14:52:27 -070086 uint32_t Kidsobjnum = pDestPDFDoc->AddIndirectObject(pNewKids);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070087
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080088 pNewPages->SetAt("Kids", new CPDF_Reference(pDestPDFDoc, Kidsobjnum));
Nico Weber9d8ec5a2015-08-04 13:00:21 -070089 pNewPages->SetAt("Count", new CPDF_Number(0));
90 }
91
Tom Sepezc7e4c4f2015-11-20 09:45:24 -080092 return TRUE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070093}
94
Nico Weber9d8ec5a2015-08-04 13:00:21 -070095FX_BOOL CPDF_PageOrganizer::ExportPage(CPDF_Document* pSrcPDFDoc,
Tom Sepez62a70f92016-03-21 15:00:20 -070096 std::vector<uint16_t>* pPageNums,
Nico Weber9d8ec5a2015-08-04 13:00:21 -070097 CPDF_Document* pDestPDFDoc,
98 int nIndex) {
99 int curpage = nIndex;
Lei Zhangaa8bf7e2015-12-24 19:13:32 -0800100 std::unique_ptr<ObjectNumberMap> pObjNumberMap(new ObjectNumberMap);
Tom Sepez11d93552016-02-09 09:55:54 -0800101 int nSize = pdfium::CollectionSize<int>(*pPageNums);
102 for (int i = 0; i < nSize; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700103 CPDF_Dictionary* pCurPageDict = pDestPDFDoc->CreateNewPage(curpage);
Tom Sepez11d93552016-02-09 09:55:54 -0800104 CPDF_Dictionary* pSrcPageDict = pSrcPDFDoc->GetPage(pPageNums->at(i) - 1);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800105 if (!pSrcPageDict || !pCurPageDict)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700106 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700107
Lei Zhang4880d1a2015-12-18 17:05:11 -0800108 // Clone the page dictionary
Oliver Chang3f1c71f2016-01-11 08:45:31 -0800109 for (const auto& it : *pSrcPageDict) {
110 const CFX_ByteString& cbSrcKeyStr = it.first;
111 CPDF_Object* pObj = it.second;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700112 if (cbSrcKeyStr.Compare(("Type")) && cbSrcKeyStr.Compare(("Parent"))) {
tsepez4c3debb2016-04-08 12:20:38 -0700113 if (pCurPageDict->KeyExist(cbSrcKeyStr.AsStringC()))
114 pCurPageDict->RemoveAt(cbSrcKeyStr.AsStringC());
115 pCurPageDict->SetAt(cbSrcKeyStr.AsStringC(), pObj->Clone());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700116 }
117 }
118
Lei Zhang4880d1a2015-12-18 17:05:11 -0800119 // inheritable item
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800120 CPDF_Object* pInheritable = nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700121 // 1 MediaBox //required
122 if (!pCurPageDict->KeyExist("MediaBox")) {
123 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "MediaBox");
124 if (!pInheritable) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800125 // Search the "CropBox" from source page dictionary,
126 // if not exists,we take the letter size.
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700127 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800128 if (pInheritable) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700129 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800130 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700131 // Make the default size to be letter size (8.5'x11')
132 CPDF_Array* pArray = new CPDF_Array;
133 pArray->AddNumber(0);
134 pArray->AddNumber(0);
135 pArray->AddNumber(612);
136 pArray->AddNumber(792);
137 pCurPageDict->SetAt("MediaBox", pArray);
138 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800139 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700140 pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800141 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700142 }
143 // 2 Resources //required
144 if (!pCurPageDict->KeyExist("Resources")) {
145 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Resources");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800146 if (!pInheritable)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700147 return FALSE;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700148 pCurPageDict->SetAt("Resources", pInheritable->Clone());
149 }
150 // 3 CropBox //Optional
151 if (!pCurPageDict->KeyExist("CropBox")) {
152 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
153 if (pInheritable)
154 pCurPageDict->SetAt("CropBox", pInheritable->Clone());
155 }
156 // 4 Rotate //Optional
157 if (!pCurPageDict->KeyExist("Rotate")) {
158 pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Rotate");
159 if (pInheritable)
160 pCurPageDict->SetAt("Rotate", pInheritable->Clone());
161 }
162
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700163 // Update the reference
tsepezc3255f52016-03-25 14:52:27 -0700164 uint32_t dwOldPageObj = pSrcPageDict->GetObjNum();
165 uint32_t dwNewPageObj = pCurPageDict->GetObjNum();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700166
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800167 (*pObjNumberMap)[dwOldPageObj] = dwNewPageObj;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700168
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800169 UpdateReference(pCurPageDict, pDestPDFDoc, pObjNumberMap.get());
170 ++curpage;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700171 }
172
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700173 return TRUE;
174}
175
176CPDF_Object* CPDF_PageOrganizer::PageDictGetInheritableTag(
177 CPDF_Dictionary* pDict,
178 CFX_ByteString nSrctag) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800179 if (!pDict || nSrctag.IsEmpty())
180 return nullptr;
181 if (!pDict->KeyExist("Parent") || !pDict->KeyExist("Type"))
182 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700183
tsepezbd567552016-03-29 14:51:50 -0700184 CPDF_Object* pType = pDict->GetObjectBy("Type")->GetDirect();
Dan Sinclair710c9092015-10-21 15:46:10 -0400185 if (!ToName(pType))
186 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700187 if (pType->GetString().Compare("Page"))
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800188 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700189
tsepezbd567552016-03-29 14:51:50 -0700190 CPDF_Dictionary* pp = ToDictionary(pDict->GetObjectBy("Parent")->GetDirect());
Dan Sinclairf1251c12015-10-20 16:24:45 -0400191 if (!pp)
192 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700193
tsepezb4c9f3f2016-04-13 15:41:21 -0700194 CFX_ByteStringC sSrcTag = nSrctag.AsStringC();
195 if (pDict->KeyExist(sSrcTag))
196 return pDict->GetObjectBy(sSrcTag);
Dan Sinclairf1251c12015-10-20 16:24:45 -0400197
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700198 while (pp) {
tsepezb4c9f3f2016-04-13 15:41:21 -0700199 if (pp->KeyExist(sSrcTag))
200 return pp->GetObjectBy(sSrcTag);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800201 if (!pp->KeyExist("Parent"))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700202 break;
tsepezbd567552016-03-29 14:51:50 -0700203 pp = ToDictionary(pp->GetObjectBy("Parent")->GetDirect());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700204 }
Dan Sinclairf1251c12015-10-20 16:24:45 -0400205 return nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700206}
207
208FX_BOOL CPDF_PageOrganizer::UpdateReference(CPDF_Object* pObj,
209 CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800210 ObjectNumberMap* pObjNumberMap) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700211 switch (pObj->GetType()) {
Tom Sepez8e5cd192016-01-26 13:20:26 -0800212 case CPDF_Object::REFERENCE: {
Dan Sinclairbf81c142015-10-26 16:54:39 -0400213 CPDF_Reference* pReference = pObj->AsReference();
tsepezc3255f52016-03-25 14:52:27 -0700214 uint32_t newobjnum = GetNewObjId(pDoc, pObjNumberMap, pReference);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700215 if (newobjnum == 0)
Tom Sepez2f2ffec2015-07-23 14:42:09 -0700216 return FALSE;
Dan Sinclairbf81c142015-10-26 16:54:39 -0400217 pReference->SetRef(pDoc, newobjnum);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700218 break;
219 }
Tom Sepez8e5cd192016-01-26 13:20:26 -0800220 case CPDF_Object::DICTIONARY: {
Dan Sinclairf1251c12015-10-20 16:24:45 -0400221 CPDF_Dictionary* pDict = pObj->AsDictionary();
Oliver Changbd292ae2016-01-13 18:46:09 -0800222 auto it = pDict->begin();
223 while (it != pDict->end()) {
224 const CFX_ByteString& key = it->first;
225 CPDF_Object* pNextObj = it->second;
226 ++it;
tsepezb4c9f3f2016-04-13 15:41:21 -0700227 if (!FXSYS_strcmp(key.c_str(), "Parent") ||
228 !FXSYS_strcmp(key.c_str(), "Prev") ||
229 !FXSYS_strcmp(key.c_str(), "First")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700230 continue;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800231 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700232 if (pNextObj) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800233 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
tsepez4c3debb2016-04-08 12:20:38 -0700234 pDict->RemoveAt(key.AsStringC());
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800235 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700236 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800237 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700238 }
239 break;
240 }
Tom Sepez8e5cd192016-01-26 13:20:26 -0800241 case CPDF_Object::ARRAY: {
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400242 CPDF_Array* pArray = pObj->AsArray();
Wei Lie1aebd42016-04-11 10:02:09 -0700243 for (size_t i = 0; i < pArray->GetCount(); ++i) {
tsepezbd567552016-03-29 14:51:50 -0700244 CPDF_Object* pNextObj = pArray->GetObjectAt(i);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800245 if (!pNextObj)
246 return FALSE;
247 if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700248 return FALSE;
249 }
250 break;
251 }
Tom Sepez8e5cd192016-01-26 13:20:26 -0800252 case CPDF_Object::STREAM: {
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400253 CPDF_Stream* pStream = pObj->AsStream();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700254 CPDF_Dictionary* pDict = pStream->GetDict();
255 if (pDict) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800256 if (!UpdateReference(pDict, pDoc, pObjNumberMap))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700257 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800258 } else {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700259 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800260 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700261 break;
262 }
263 default:
264 break;
265 }
266
267 return TRUE;
268}
269
tsepezc3255f52016-03-25 14:52:27 -0700270uint32_t CPDF_PageOrganizer::GetNewObjId(CPDF_Document* pDoc,
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800271 ObjectNumberMap* pObjNumberMap,
272 CPDF_Reference* pRef) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700273 if (!pRef)
274 return 0;
275
tsepezc3255f52016-03-25 14:52:27 -0700276 uint32_t dwObjnum = pRef->GetRefObjNum();
277 uint32_t dwNewObjNum = 0;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800278 const auto it = pObjNumberMap->find(dwObjnum);
279 if (it != pObjNumberMap->end())
280 dwNewObjNum = it->second;
281 if (dwNewObjNum)
282 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700283
284 CPDF_Object* pDirect = pRef->GetDirect();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800285 if (!pDirect)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700286 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700287
288 CPDF_Object* pClone = pDirect->Clone();
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800289 if (!pClone)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700290 return 0;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700291
Dan Sinclairf1251c12015-10-20 16:24:45 -0400292 if (CPDF_Dictionary* pDictClone = pClone->AsDictionary()) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700293 if (pDictClone->KeyExist("Type")) {
Wei Li9b761132016-01-29 15:44:20 -0800294 CFX_ByteString strType = pDictClone->GetStringBy("Type");
tsepezb4c9f3f2016-04-13 15:41:21 -0700295 if (!FXSYS_stricmp(strType.c_str(), "Pages")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700296 pDictClone->Release();
297 return 4;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800298 }
tsepezb4c9f3f2016-04-13 15:41:21 -0700299 if (!FXSYS_stricmp(strType.c_str(), "Page")) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700300 pDictClone->Release();
301 return 0;
302 }
303 }
304 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700305 dwNewObjNum = pDoc->AddIndirectObject(pClone);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800306 (*pObjNumberMap)[dwObjnum] = dwNewObjNum;
307 if (!UpdateReference(pClone, pDoc, pObjNumberMap)) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700308 pClone->Release();
309 return 0;
310 }
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800311 return dwNewObjNum;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700312}
313
314FPDF_BOOL ParserPageRangeString(CFX_ByteString rangstring,
Tom Sepez62a70f92016-03-21 15:00:20 -0700315 std::vector<uint16_t>* pageArray,
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700316 int nCount) {
317 if (rangstring.GetLength() != 0) {
318 rangstring.Remove(' ');
319 int nLength = rangstring.GetLength();
320 CFX_ByteString cbCompareString("0123456789-,");
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800321 for (int i = 0; i < nLength; ++i) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700322 if (cbCompareString.Find(rangstring[i]) == -1)
323 return FALSE;
324 }
325 CFX_ByteString cbMidRange;
326 int nStringFrom = 0;
327 int nStringTo = 0;
328 while (nStringTo < nLength) {
329 nStringTo = rangstring.Find(',', nStringFrom);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800330 if (nStringTo == -1)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700331 nStringTo = nLength;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700332 cbMidRange = rangstring.Mid(nStringFrom, nStringTo - nStringFrom);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700333 int nMid = cbMidRange.Find('-');
334 if (nMid == -1) {
tsepezb4c9f3f2016-04-13 15:41:21 -0700335 long lPageNum = atol(cbMidRange.c_str());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700336 if (lPageNum <= 0 || lPageNum > nCount)
337 return FALSE;
Tom Sepez62a70f92016-03-21 15:00:20 -0700338 pageArray->push_back((uint16_t)lPageNum);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700339 } else {
tsepezb4c9f3f2016-04-13 15:41:21 -0700340 int nStartPageNum = atol(cbMidRange.Mid(0, nMid).c_str());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700341 if (nStartPageNum == 0)
342 return FALSE;
343
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800344 ++nMid;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700345 int nEnd = cbMidRange.GetLength() - nMid;
346 if (nEnd == 0)
347 return FALSE;
348
tsepezb4c9f3f2016-04-13 15:41:21 -0700349 int nEndPageNum = atol(cbMidRange.Mid(nMid, nEnd).c_str());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700350 if (nStartPageNum < 0 || nStartPageNum > nEndPageNum ||
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800351 nEndPageNum > nCount) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700352 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800353 }
354 for (int i = nStartPageNum; i <= nEndPageNum; ++i) {
Tom Sepez11d93552016-02-09 09:55:54 -0800355 pageArray->push_back(i);
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800356 }
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700357 }
358 nStringFrom = nStringTo + 1;
359 }
360 }
361 return TRUE;
362}
363
364DLLEXPORT FPDF_BOOL STDCALL FPDF_ImportPages(FPDF_DOCUMENT dest_doc,
365 FPDF_DOCUMENT src_doc,
366 FPDF_BYTESTRING pagerange,
367 int index) {
Tom Sepez471a1032015-10-15 16:17:18 -0700368 CPDF_Document* pDestDoc = CPDFDocumentFromFPDFDocument(dest_doc);
369 if (!dest_doc)
370 return FALSE;
371
372 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
373 if (!pSrcDoc)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700374 return FALSE;
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800375
Tom Sepez62a70f92016-03-21 15:00:20 -0700376 std::vector<uint16_t> pageArray;
Tom Sepez471a1032015-10-15 16:17:18 -0700377 int nCount = pSrcDoc->GetPageCount();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700378 if (pagerange) {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800379 if (!ParserPageRangeString(pagerange, &pageArray, nCount))
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700380 return FALSE;
381 } else {
Tom Sepezc7e4c4f2015-11-20 09:45:24 -0800382 for (int i = 1; i <= nCount; ++i) {
Tom Sepez11d93552016-02-09 09:55:54 -0800383 pageArray.push_back(i);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700384 }
385 }
386
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700387 CPDF_PageOrganizer pageOrg;
Tom Sepez471a1032015-10-15 16:17:18 -0700388 pageOrg.PDFDocInit(pDestDoc, pSrcDoc);
389 return pageOrg.ExportPage(pSrcDoc, &pageArray, pDestDoc, index);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700390}
391
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700392DLLEXPORT FPDF_BOOL STDCALL FPDF_CopyViewerPreferences(FPDF_DOCUMENT dest_doc,
393 FPDF_DOCUMENT src_doc) {
Tom Sepez471a1032015-10-15 16:17:18 -0700394 CPDF_Document* pDstDoc = CPDFDocumentFromFPDFDocument(dest_doc);
395 if (!pDstDoc)
396 return FALSE;
397
398 CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
399 if (!pSrcDoc)
400 return FALSE;
401
402 CPDF_Dictionary* pSrcDict = pSrcDoc->GetRoot();
Wei Li9b761132016-01-29 15:44:20 -0800403 pSrcDict = pSrcDict->GetDictBy("ViewerPreferences");
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700404 if (!pSrcDict)
405 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700406
407 CPDF_Dictionary* pDstDict = pDstDoc->GetRoot();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700408 if (!pDstDict)
409 return FALSE;
Tom Sepez471a1032015-10-15 16:17:18 -0700410
Lei Zhangd983b092015-12-14 16:58:33 -0800411 pDstDict->SetAt("ViewerPreferences", pSrcDict->Clone(TRUE));
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700412 return TRUE;
413}