| Index: fpdfsdk/src/fpdfppo.cpp
|
| diff --git a/fpdfsdk/src/fpdfppo.cpp b/fpdfsdk/src/fpdfppo.cpp
|
| deleted file mode 100644
|
| index 4253cc5fd95186a235ef4ffca63e879b450825ee..0000000000000000000000000000000000000000
|
| --- a/fpdfsdk/src/fpdfppo.cpp
|
| +++ /dev/null
|
| @@ -1,415 +0,0 @@
|
| -// Copyright 2014 PDFium Authors. All rights reserved.
|
| -// 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 "public/fpdf_ppo.h"
|
| -
|
| -#include <map>
|
| -#include <memory>
|
| -#include <vector>
|
| -
|
| -#include "core/include/fpdfapi/cpdf_array.h"
|
| -#include "core/include/fpdfapi/cpdf_document.h"
|
| -#include "core/include/fpdfapi/cpdf_name.h"
|
| -#include "core/include/fpdfapi/cpdf_number.h"
|
| -#include "core/include/fpdfapi/cpdf_reference.h"
|
| -#include "core/include/fpdfapi/cpdf_string.h"
|
| -#include "fpdfsdk/include/fsdk_define.h"
|
| -#include "third_party/base/stl_util.h"
|
| -
|
| -class CPDF_PageOrganizer {
|
| - public:
|
| - using ObjectNumberMap = std::map<FX_DWORD, FX_DWORD>;
|
| - CPDF_PageOrganizer();
|
| - ~CPDF_PageOrganizer();
|
| -
|
| - FX_BOOL PDFDocInit(CPDF_Document* pDestPDFDoc, CPDF_Document* pSrcPDFDoc);
|
| - FX_BOOL ExportPage(CPDF_Document* pSrcPDFDoc,
|
| - std::vector<FX_WORD>* pPageNums,
|
| - CPDF_Document* pDestPDFDoc,
|
| - int nIndex);
|
| - CPDF_Object* PageDictGetInheritableTag(CPDF_Dictionary* pDict,
|
| - CFX_ByteString nSrctag);
|
| - FX_BOOL UpdateReference(CPDF_Object* pObj,
|
| - CPDF_Document* pDoc,
|
| - ObjectNumberMap* pObjNumberMap);
|
| - FX_DWORD GetNewObjId(CPDF_Document* pDoc,
|
| - ObjectNumberMap* pObjNumberMap,
|
| - CPDF_Reference* pRef);
|
| -};
|
| -
|
| -CPDF_PageOrganizer::CPDF_PageOrganizer() {}
|
| -
|
| -CPDF_PageOrganizer::~CPDF_PageOrganizer() {}
|
| -
|
| -FX_BOOL CPDF_PageOrganizer::PDFDocInit(CPDF_Document* pDestPDFDoc,
|
| - CPDF_Document* pSrcPDFDoc) {
|
| - if (!pDestPDFDoc || !pSrcPDFDoc)
|
| - return FALSE;
|
| -
|
| - CPDF_Dictionary* pNewRoot = pDestPDFDoc->GetRoot();
|
| - if (!pNewRoot)
|
| - return FALSE;
|
| -
|
| - // Set the document information
|
| - CPDF_Dictionary* DInfoDict = pDestPDFDoc->GetInfo();
|
| - if (!DInfoDict)
|
| - return FALSE;
|
| -
|
| - CFX_ByteString producerstr;
|
| - producerstr.Format("PDFium");
|
| - DInfoDict->SetAt("Producer", new CPDF_String(producerstr, FALSE));
|
| -
|
| - // Set type
|
| - CFX_ByteString cbRootType = pNewRoot->GetStringBy("Type", "");
|
| - if (cbRootType.Equal("")) {
|
| - pNewRoot->SetAt("Type", new CPDF_Name("Catalog"));
|
| - }
|
| -
|
| - CPDF_Object* pElement = pNewRoot->GetElement("Pages");
|
| - CPDF_Dictionary* pNewPages =
|
| - pElement ? ToDictionary(pElement->GetDirect()) : nullptr;
|
| - if (!pNewPages) {
|
| - pNewPages = new CPDF_Dictionary;
|
| - FX_DWORD NewPagesON = pDestPDFDoc->AddIndirectObject(pNewPages);
|
| - pNewRoot->SetAt("Pages", new CPDF_Reference(pDestPDFDoc, NewPagesON));
|
| - }
|
| -
|
| - CFX_ByteString cbPageType = pNewPages->GetStringBy("Type", "");
|
| - if (cbPageType.Equal("")) {
|
| - pNewPages->SetAt("Type", new CPDF_Name("Pages"));
|
| - }
|
| -
|
| - CPDF_Array* pKeysArray = pNewPages->GetArrayBy("Kids");
|
| - if (!pKeysArray) {
|
| - CPDF_Array* pNewKids = new CPDF_Array;
|
| - FX_DWORD Kidsobjnum = -1;
|
| - Kidsobjnum = pDestPDFDoc->AddIndirectObject(pNewKids);
|
| -
|
| - pNewPages->SetAt("Kids", new CPDF_Reference(pDestPDFDoc, Kidsobjnum));
|
| - pNewPages->SetAt("Count", new CPDF_Number(0));
|
| - }
|
| -
|
| - return TRUE;
|
| -}
|
| -
|
| -FX_BOOL CPDF_PageOrganizer::ExportPage(CPDF_Document* pSrcPDFDoc,
|
| - std::vector<FX_WORD>* pPageNums,
|
| - CPDF_Document* pDestPDFDoc,
|
| - int nIndex) {
|
| - int curpage = nIndex;
|
| - std::unique_ptr<ObjectNumberMap> pObjNumberMap(new ObjectNumberMap);
|
| - int nSize = pdfium::CollectionSize<int>(*pPageNums);
|
| - for (int i = 0; i < nSize; ++i) {
|
| - CPDF_Dictionary* pCurPageDict = pDestPDFDoc->CreateNewPage(curpage);
|
| - CPDF_Dictionary* pSrcPageDict = pSrcPDFDoc->GetPage(pPageNums->at(i) - 1);
|
| - if (!pSrcPageDict || !pCurPageDict)
|
| - return FALSE;
|
| -
|
| - // Clone the page dictionary
|
| - for (const auto& it : *pSrcPageDict) {
|
| - const CFX_ByteString& cbSrcKeyStr = it.first;
|
| - CPDF_Object* pObj = it.second;
|
| - if (cbSrcKeyStr.Compare(("Type")) && cbSrcKeyStr.Compare(("Parent"))) {
|
| - if (pCurPageDict->KeyExist(cbSrcKeyStr))
|
| - pCurPageDict->RemoveAt(cbSrcKeyStr);
|
| - pCurPageDict->SetAt(cbSrcKeyStr, pObj->Clone());
|
| - }
|
| - }
|
| -
|
| - // inheritable item
|
| - CPDF_Object* pInheritable = nullptr;
|
| - // 1 MediaBox //required
|
| - if (!pCurPageDict->KeyExist("MediaBox")) {
|
| - pInheritable = PageDictGetInheritableTag(pSrcPageDict, "MediaBox");
|
| - if (!pInheritable) {
|
| - // Search the "CropBox" from source page dictionary,
|
| - // if not exists,we take the letter size.
|
| - pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
|
| - if (pInheritable) {
|
| - pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
|
| - } else {
|
| - // Make the default size to be letter size (8.5'x11')
|
| - CPDF_Array* pArray = new CPDF_Array;
|
| - pArray->AddNumber(0);
|
| - pArray->AddNumber(0);
|
| - pArray->AddNumber(612);
|
| - pArray->AddNumber(792);
|
| - pCurPageDict->SetAt("MediaBox", pArray);
|
| - }
|
| - } else {
|
| - pCurPageDict->SetAt("MediaBox", pInheritable->Clone());
|
| - }
|
| - }
|
| - // 2 Resources //required
|
| - if (!pCurPageDict->KeyExist("Resources")) {
|
| - pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Resources");
|
| - if (!pInheritable)
|
| - return FALSE;
|
| - pCurPageDict->SetAt("Resources", pInheritable->Clone());
|
| - }
|
| - // 3 CropBox //Optional
|
| - if (!pCurPageDict->KeyExist("CropBox")) {
|
| - pInheritable = PageDictGetInheritableTag(pSrcPageDict, "CropBox");
|
| - if (pInheritable)
|
| - pCurPageDict->SetAt("CropBox", pInheritable->Clone());
|
| - }
|
| - // 4 Rotate //Optional
|
| - if (!pCurPageDict->KeyExist("Rotate")) {
|
| - pInheritable = PageDictGetInheritableTag(pSrcPageDict, "Rotate");
|
| - if (pInheritable)
|
| - pCurPageDict->SetAt("Rotate", pInheritable->Clone());
|
| - }
|
| -
|
| - // Update the reference
|
| - FX_DWORD dwOldPageObj = pSrcPageDict->GetObjNum();
|
| - FX_DWORD dwNewPageObj = pCurPageDict->GetObjNum();
|
| -
|
| - (*pObjNumberMap)[dwOldPageObj] = dwNewPageObj;
|
| -
|
| - UpdateReference(pCurPageDict, pDestPDFDoc, pObjNumberMap.get());
|
| - ++curpage;
|
| - }
|
| -
|
| - return TRUE;
|
| -}
|
| -
|
| -CPDF_Object* CPDF_PageOrganizer::PageDictGetInheritableTag(
|
| - CPDF_Dictionary* pDict,
|
| - CFX_ByteString nSrctag) {
|
| - if (!pDict || nSrctag.IsEmpty())
|
| - return nullptr;
|
| - if (!pDict->KeyExist("Parent") || !pDict->KeyExist("Type"))
|
| - return nullptr;
|
| -
|
| - CPDF_Object* pType = pDict->GetElement("Type")->GetDirect();
|
| - if (!ToName(pType))
|
| - return nullptr;
|
| - if (pType->GetString().Compare("Page"))
|
| - return nullptr;
|
| -
|
| - CPDF_Dictionary* pp = ToDictionary(pDict->GetElement("Parent")->GetDirect());
|
| - if (!pp)
|
| - return nullptr;
|
| -
|
| - if (pDict->KeyExist((const char*)nSrctag))
|
| - return pDict->GetElement((const char*)nSrctag);
|
| -
|
| - while (pp) {
|
| - if (pp->KeyExist((const char*)nSrctag))
|
| - return pp->GetElement((const char*)nSrctag);
|
| - if (!pp->KeyExist("Parent"))
|
| - break;
|
| - pp = ToDictionary(pp->GetElement("Parent")->GetDirect());
|
| - }
|
| - return nullptr;
|
| -}
|
| -
|
| -FX_BOOL CPDF_PageOrganizer::UpdateReference(CPDF_Object* pObj,
|
| - CPDF_Document* pDoc,
|
| - ObjectNumberMap* pObjNumberMap) {
|
| - switch (pObj->GetType()) {
|
| - case CPDF_Object::REFERENCE: {
|
| - CPDF_Reference* pReference = pObj->AsReference();
|
| - FX_DWORD newobjnum = GetNewObjId(pDoc, pObjNumberMap, pReference);
|
| - if (newobjnum == 0)
|
| - return FALSE;
|
| - pReference->SetRef(pDoc, newobjnum);
|
| - break;
|
| - }
|
| - case CPDF_Object::DICTIONARY: {
|
| - CPDF_Dictionary* pDict = pObj->AsDictionary();
|
| - auto it = pDict->begin();
|
| - while (it != pDict->end()) {
|
| - const CFX_ByteString& key = it->first;
|
| - CPDF_Object* pNextObj = it->second;
|
| - ++it;
|
| - if (!FXSYS_strcmp(key, "Parent") || !FXSYS_strcmp(key, "Prev") ||
|
| - !FXSYS_strcmp(key, "First")) {
|
| - continue;
|
| - }
|
| - if (pNextObj) {
|
| - if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
|
| - pDict->RemoveAt(key);
|
| - } else {
|
| - return FALSE;
|
| - }
|
| - }
|
| - break;
|
| - }
|
| - case CPDF_Object::ARRAY: {
|
| - CPDF_Array* pArray = pObj->AsArray();
|
| - FX_DWORD count = pArray->GetCount();
|
| - for (FX_DWORD i = 0; i < count; ++i) {
|
| - CPDF_Object* pNextObj = pArray->GetElement(i);
|
| - if (!pNextObj)
|
| - return FALSE;
|
| - if (!UpdateReference(pNextObj, pDoc, pObjNumberMap))
|
| - return FALSE;
|
| - }
|
| - break;
|
| - }
|
| - case CPDF_Object::STREAM: {
|
| - CPDF_Stream* pStream = pObj->AsStream();
|
| - CPDF_Dictionary* pDict = pStream->GetDict();
|
| - if (pDict) {
|
| - if (!UpdateReference(pDict, pDoc, pObjNumberMap))
|
| - return FALSE;
|
| - } else {
|
| - return FALSE;
|
| - }
|
| - break;
|
| - }
|
| - default:
|
| - break;
|
| - }
|
| -
|
| - return TRUE;
|
| -}
|
| -
|
| -FX_DWORD CPDF_PageOrganizer::GetNewObjId(CPDF_Document* pDoc,
|
| - ObjectNumberMap* pObjNumberMap,
|
| - CPDF_Reference* pRef) {
|
| - if (!pRef)
|
| - return 0;
|
| -
|
| - FX_DWORD dwObjnum = pRef->GetRefObjNum();
|
| - FX_DWORD dwNewObjNum = 0;
|
| - const auto it = pObjNumberMap->find(dwObjnum);
|
| - if (it != pObjNumberMap->end())
|
| - dwNewObjNum = it->second;
|
| - if (dwNewObjNum)
|
| - return dwNewObjNum;
|
| -
|
| - CPDF_Object* pDirect = pRef->GetDirect();
|
| - if (!pDirect)
|
| - return 0;
|
| -
|
| - CPDF_Object* pClone = pDirect->Clone();
|
| - if (!pClone)
|
| - return 0;
|
| -
|
| - if (CPDF_Dictionary* pDictClone = pClone->AsDictionary()) {
|
| - if (pDictClone->KeyExist("Type")) {
|
| - CFX_ByteString strType = pDictClone->GetStringBy("Type");
|
| - if (!FXSYS_stricmp(strType, "Pages")) {
|
| - pDictClone->Release();
|
| - return 4;
|
| - }
|
| - if (!FXSYS_stricmp(strType, "Page")) {
|
| - pDictClone->Release();
|
| - return 0;
|
| - }
|
| - }
|
| - }
|
| - dwNewObjNum = pDoc->AddIndirectObject(pClone);
|
| - (*pObjNumberMap)[dwObjnum] = dwNewObjNum;
|
| - if (!UpdateReference(pClone, pDoc, pObjNumberMap)) {
|
| - pClone->Release();
|
| - return 0;
|
| - }
|
| - return dwNewObjNum;
|
| -}
|
| -
|
| -FPDF_BOOL ParserPageRangeString(CFX_ByteString rangstring,
|
| - std::vector<FX_WORD>* pageArray,
|
| - int nCount) {
|
| - if (rangstring.GetLength() != 0) {
|
| - rangstring.Remove(' ');
|
| - int nLength = rangstring.GetLength();
|
| - CFX_ByteString cbCompareString("0123456789-,");
|
| - for (int i = 0; i < nLength; ++i) {
|
| - if (cbCompareString.Find(rangstring[i]) == -1)
|
| - return FALSE;
|
| - }
|
| - CFX_ByteString cbMidRange;
|
| - int nStringFrom = 0;
|
| - int nStringTo = 0;
|
| - while (nStringTo < nLength) {
|
| - nStringTo = rangstring.Find(',', nStringFrom);
|
| - if (nStringTo == -1)
|
| - nStringTo = nLength;
|
| - cbMidRange = rangstring.Mid(nStringFrom, nStringTo - nStringFrom);
|
| - int nMid = cbMidRange.Find('-');
|
| - if (nMid == -1) {
|
| - long lPageNum = atol(cbMidRange);
|
| - if (lPageNum <= 0 || lPageNum > nCount)
|
| - return FALSE;
|
| - pageArray->push_back((FX_WORD)lPageNum);
|
| - } else {
|
| - int nStartPageNum = atol(cbMidRange.Mid(0, nMid));
|
| - if (nStartPageNum == 0)
|
| - return FALSE;
|
| -
|
| - ++nMid;
|
| - int nEnd = cbMidRange.GetLength() - nMid;
|
| - if (nEnd == 0)
|
| - return FALSE;
|
| -
|
| - int nEndPageNum = atol(cbMidRange.Mid(nMid, nEnd));
|
| - if (nStartPageNum < 0 || nStartPageNum > nEndPageNum ||
|
| - nEndPageNum > nCount) {
|
| - return FALSE;
|
| - }
|
| - for (int i = nStartPageNum; i <= nEndPageNum; ++i) {
|
| - pageArray->push_back(i);
|
| - }
|
| - }
|
| - nStringFrom = nStringTo + 1;
|
| - }
|
| - }
|
| - return TRUE;
|
| -}
|
| -
|
| -DLLEXPORT FPDF_BOOL STDCALL FPDF_ImportPages(FPDF_DOCUMENT dest_doc,
|
| - FPDF_DOCUMENT src_doc,
|
| - FPDF_BYTESTRING pagerange,
|
| - int index) {
|
| - CPDF_Document* pDestDoc = CPDFDocumentFromFPDFDocument(dest_doc);
|
| - if (!dest_doc)
|
| - return FALSE;
|
| -
|
| - CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
|
| - if (!pSrcDoc)
|
| - return FALSE;
|
| -
|
| - std::vector<FX_WORD> pageArray;
|
| - int nCount = pSrcDoc->GetPageCount();
|
| - if (pagerange) {
|
| - if (!ParserPageRangeString(pagerange, &pageArray, nCount))
|
| - return FALSE;
|
| - } else {
|
| - for (int i = 1; i <= nCount; ++i) {
|
| - pageArray.push_back(i);
|
| - }
|
| - }
|
| -
|
| - CPDF_PageOrganizer pageOrg;
|
| - pageOrg.PDFDocInit(pDestDoc, pSrcDoc);
|
| - return pageOrg.ExportPage(pSrcDoc, &pageArray, pDestDoc, index);
|
| -}
|
| -
|
| -DLLEXPORT FPDF_BOOL STDCALL FPDF_CopyViewerPreferences(FPDF_DOCUMENT dest_doc,
|
| - FPDF_DOCUMENT src_doc) {
|
| - CPDF_Document* pDstDoc = CPDFDocumentFromFPDFDocument(dest_doc);
|
| - if (!pDstDoc)
|
| - return FALSE;
|
| -
|
| - CPDF_Document* pSrcDoc = CPDFDocumentFromFPDFDocument(src_doc);
|
| - if (!pSrcDoc)
|
| - return FALSE;
|
| -
|
| - CPDF_Dictionary* pSrcDict = pSrcDoc->GetRoot();
|
| - pSrcDict = pSrcDict->GetDictBy("ViewerPreferences");
|
| - if (!pSrcDict)
|
| - return FALSE;
|
| -
|
| - CPDF_Dictionary* pDstDict = pDstDoc->GetRoot();
|
| - if (!pDstDict)
|
| - return FALSE;
|
| -
|
| - pDstDict->SetAt("ViewerPreferences", pSrcDict->Clone(TRUE));
|
| - return TRUE;
|
| -}
|
|
|