blob: 68007d3ca53430e39a5199bfc44d71d9acee65fd [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_transformpage.h"
8
Dan Sinclair584b1e62016-03-21 09:15:45 -04009#include "core/fpdfapi/fpdf_page/include/cpdf_clippath.h"
Dan Sinclair455a4192016-03-16 09:48:56 -040010#include "core/fpdfapi/fpdf_page/include/cpdf_page.h"
Dan Sinclair584b1e62016-03-21 09:15:45 -040011#include "core/fpdfapi/fpdf_page/include/cpdf_pageobject.h"
12#include "core/fpdfapi/fpdf_page/include/cpdf_path.h"
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_number.h"
16#include "core/fpdfapi/fpdf_parser/include/cpdf_reference.h"
Dan Sinclair584b1e62016-03-21 09:15:45 -040017#include "core/fpdfapi/fpdf_parser/include/cpdf_stream.h"
npm660de3c2016-08-08 08:18:29 -070018#include "core/fxge/include/cfx_pathdata.h"
Lei Zhangbde53d22015-11-12 22:21:30 -080019#include "fpdfsdk/include/fsdk_define.h"
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070020
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070021namespace {
22
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070023void SetBoundingBox(CPDF_Page* page,
tsepez71a452f2016-05-13 17:51:27 -070024 const CFX_ByteString& key,
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070025 float left,
26 float bottom,
27 float right,
28 float top) {
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070029 CPDF_Array* pBoundingBoxArray = new CPDF_Array;
30 pBoundingBoxArray->Add(new CPDF_Number(left));
31 pBoundingBoxArray->Add(new CPDF_Number(bottom));
32 pBoundingBoxArray->Add(new CPDF_Number(right));
33 pBoundingBoxArray->Add(new CPDF_Number(top));
dsinclair38fd8442016-09-15 10:15:32 -070034 page->m_pFormDict->SetFor(key, pBoundingBoxArray);
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070035}
36
tsepez71a452f2016-05-13 17:51:27 -070037bool GetBoundingBox(CPDF_Page* page,
38 const CFX_ByteString& key,
39 float* left,
40 float* bottom,
41 float* right,
42 float* top) {
dsinclair38fd8442016-09-15 10:15:32 -070043 CPDF_Array* pArray = page->m_pFormDict->GetArrayFor(key);
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070044 if (!pArray)
tsepez71a452f2016-05-13 17:51:27 -070045 return false;
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070046
Wei Li9b761132016-01-29 15:44:20 -080047 *left = pArray->GetFloatAt(0);
48 *bottom = pArray->GetFloatAt(1);
49 *right = pArray->GetFloatAt(2);
50 *top = pArray->GetFloatAt(3);
tsepez71a452f2016-05-13 17:51:27 -070051 return true;
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070052}
53
54} // namespace
55
Nico Weber9d8ec5a2015-08-04 13:00:21 -070056DLLEXPORT void STDCALL FPDFPage_SetMediaBox(FPDF_PAGE page,
57 float left,
58 float bottom,
59 float right,
60 float top) {
Tom Sepezdb0be962015-10-16 14:00:21 -070061 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070062 if (!pPage)
63 return;
Lei Zhanga6d9f0e2015-06-13 00:48:38 -070064
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070065 SetBoundingBox(pPage, "MediaBox", left, bottom, right, top);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070066}
67
Nico Weber9d8ec5a2015-08-04 13:00:21 -070068DLLEXPORT void STDCALL FPDFPage_SetCropBox(FPDF_PAGE page,
69 float left,
70 float bottom,
71 float right,
72 float top) {
Tom Sepezdb0be962015-10-16 14:00:21 -070073 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Nico Weber9d8ec5a2015-08-04 13:00:21 -070074 if (!pPage)
75 return;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070076
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070077 SetBoundingBox(pPage, "CropBox", left, bottom, right, top);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070078}
79
Nico Weber9d8ec5a2015-08-04 13:00:21 -070080DLLEXPORT FPDF_BOOL STDCALL FPDFPage_GetMediaBox(FPDF_PAGE page,
81 float* left,
82 float* bottom,
83 float* right,
84 float* top) {
Tom Sepezdb0be962015-10-16 14:00:21 -070085 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070086 return pPage && GetBoundingBox(pPage, "MediaBox", left, bottom, right, top);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070087}
88
Nico Weber9d8ec5a2015-08-04 13:00:21 -070089DLLEXPORT FPDF_BOOL STDCALL FPDFPage_GetCropBox(FPDF_PAGE page,
90 float* left,
91 float* bottom,
92 float* right,
93 float* top) {
Tom Sepezdb0be962015-10-16 14:00:21 -070094 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Lei Zhang6ac0d2f2015-10-02 10:08:48 -070095 return pPage && GetBoundingBox(pPage, "CropBox", left, bottom, right, top);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -070096}
97
Nico Weber9d8ec5a2015-08-04 13:00:21 -070098DLLEXPORT FPDF_BOOL STDCALL FPDFPage_TransFormWithClip(FPDF_PAGE page,
99 FS_MATRIX* matrix,
100 FS_RECTF* clipRect) {
Tom Sepezdb0be962015-10-16 14:00:21 -0700101 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700102 if (!pPage)
103 return FALSE;
Bo Xufdc00a72014-10-28 23:03:33 -0700104
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700105 CFX_ByteTextBuf textBuf;
106 textBuf << "q ";
107 CFX_FloatRect rect(clipRect->left, clipRect->bottom, clipRect->right,
108 clipRect->top);
109 rect.Normalize();
110 CFX_ByteString bsClipping;
111 bsClipping.Format("%f %f %f %f re W* n ", rect.left, rect.bottom,
112 rect.Width(), rect.Height());
113 textBuf << bsClipping;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700114
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700115 CFX_ByteString bsMatix;
116 bsMatix.Format("%f %f %f %f %f %f cm ", matrix->a, matrix->b, matrix->c,
117 matrix->d, matrix->e, matrix->f);
118 textBuf << bsMatix;
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700119
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700120 CPDF_Dictionary* pPageDic = pPage->m_pFormDict;
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400121 CPDF_Object* pContentObj =
dsinclair38fd8442016-09-15 10:15:32 -0700122 pPageDic ? pPageDic->GetObjectFor("Contents") : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700123 if (!pContentObj)
dsinclair38fd8442016-09-15 10:15:32 -0700124 pContentObj = pPageDic ? pPageDic->GetArrayFor("Contents") : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700125 if (!pContentObj)
126 return FALSE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700127
Tom Sepezae51c812015-08-05 12:34:06 -0700128 CPDF_Dictionary* pDic = new CPDF_Dictionary;
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400129 CPDF_Stream* pStream = new CPDF_Stream(nullptr, 0, pDic);
tsepeze6db16e2016-09-19 10:45:09 -0700130 pStream->SetData(textBuf.GetBuffer(), textBuf.GetSize());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700131 CPDF_Document* pDoc = pPage->m_pDocument;
132 if (!pDoc)
133 return FALSE;
134 pDoc->AddIndirectObject(pStream);
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700135
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700136 pDic = new CPDF_Dictionary;
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400137 CPDF_Stream* pEndStream = new CPDF_Stream(nullptr, 0, pDic);
tsepeze6db16e2016-09-19 10:45:09 -0700138 pEndStream->SetData((const uint8_t*)" Q", 2);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700139 pDoc->AddIndirectObject(pEndStream);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700140
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400141 CPDF_Array* pContentArray = nullptr;
weilidb444d22016-06-02 15:48:15 -0700142 CPDF_Array* pArray = ToArray(pContentObj);
143 if (pArray) {
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400144 pContentArray = pArray;
Tom Sepezae51c812015-08-05 12:34:06 -0700145 CPDF_Reference* pRef = new CPDF_Reference(pDoc, pStream->GetObjNum());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700146 pContentArray->InsertAt(0, pRef);
tsepezbb577af2016-09-21 19:10:19 -0700147 pContentArray->AddReference(pDoc, pEndStream->GetObjNum());
Dan Sinclairbf81c142015-10-26 16:54:39 -0400148 } else if (CPDF_Reference* pReference = ToReference(pContentObj)) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700149 CPDF_Object* pDirectObj = pReference->GetDirect();
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400150 if (pDirectObj) {
weilidb444d22016-06-02 15:48:15 -0700151 CPDF_Array* pObjArray = pDirectObj->AsArray();
152 if (pObjArray) {
153 pContentArray = pObjArray;
Nico Weber077f1a32015-08-06 15:08:57 -0700154 CPDF_Reference* pRef = new CPDF_Reference(pDoc, pStream->GetObjNum());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700155 pContentArray->InsertAt(0, pRef);
tsepezbb577af2016-09-21 19:10:19 -0700156 pContentArray->AddReference(pDoc, pEndStream->GetObjNum());
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400157 } else if (pDirectObj->IsStream()) {
Tom Sepezae51c812015-08-05 12:34:06 -0700158 pContentArray = new CPDF_Array();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700159 pContentArray->AddReference(pDoc, pStream->GetObjNum());
160 pContentArray->AddReference(pDoc, pDirectObj->GetObjNum());
tsepezbb577af2016-09-21 19:10:19 -0700161 pContentArray->AddReference(pDoc, pEndStream->GetObjNum());
dsinclair38fd8442016-09-15 10:15:32 -0700162 pPageDic->SetReferenceFor("Contents", pDoc,
163 pDoc->AddIndirectObject(pContentArray));
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700164 }
165 }
166 }
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700167
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700168 // Need to transform the patterns as well.
dsinclair38fd8442016-09-15 10:15:32 -0700169 CPDF_Dictionary* pRes = pPageDic->GetDictFor("Resources");
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700170 if (pRes) {
dsinclair38fd8442016-09-15 10:15:32 -0700171 CPDF_Dictionary* pPattenDict = pRes->GetDictFor("Pattern");
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700172 if (pPattenDict) {
Oliver Chang3f1c71f2016-01-11 08:45:31 -0800173 for (const auto& it : *pPattenDict) {
174 CPDF_Object* pObj = it.second;
Dan Sinclairbf81c142015-10-26 16:54:39 -0400175 if (pObj->IsReference())
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700176 pObj = pObj->GetDirect();
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400177
Oliver Chang3f1c71f2016-01-11 08:45:31 -0800178 CPDF_Dictionary* pDict = nullptr;
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400179 if (pObj->IsDictionary())
Dan Sinclairf1251c12015-10-20 16:24:45 -0400180 pDict = pObj->AsDictionary();
weilidb444d22016-06-02 15:48:15 -0700181 else if (CPDF_Stream* pObjStream = pObj->AsStream())
182 pDict = pObjStream->GetDict();
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400183 else
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700184 continue;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700185
dsinclair38fd8442016-09-15 10:15:32 -0700186 CFX_Matrix m = pDict->GetMatrixFor("Matrix");
Tom Sepez60d909e2015-12-10 15:34:55 -0800187 CFX_Matrix t = *(CFX_Matrix*)matrix;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700188 m.Concat(t);
dsinclair38fd8442016-09-15 10:15:32 -0700189 pDict->SetMatrixFor("Matrix", m);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700190 }
191 }
192 }
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700193
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700194 return TRUE;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700195}
196
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700197DLLEXPORT void STDCALL
198FPDFPageObj_TransformClipPath(FPDF_PAGEOBJECT page_object,
199 double a,
200 double b,
201 double c,
202 double d,
203 double e,
204 double f) {
205 CPDF_PageObject* pPageObj = (CPDF_PageObject*)page_object;
Lei Zhang412e9082015-12-14 18:34:00 -0800206 if (!pPageObj)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700207 return;
Tom Sepez60d909e2015-12-10 15:34:55 -0800208 CFX_Matrix matrix((FX_FLOAT)a, (FX_FLOAT)b, (FX_FLOAT)c, (FX_FLOAT)d,
209 (FX_FLOAT)e, (FX_FLOAT)f);
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700210
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700211 // Special treatment to shading object, because the ClipPath for shading
212 // object is already transformed.
Wei Li7cf13c92016-02-19 11:53:03 -0800213 if (!pPageObj->IsShading())
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700214 pPageObj->TransformClipPath(matrix);
215 pPageObj->TransformGeneralState(matrix);
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700216}
217
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700218DLLEXPORT FPDF_CLIPPATH STDCALL FPDF_CreateClipPath(float left,
219 float bottom,
220 float right,
221 float top) {
tsepezfc1d16f2016-09-02 15:45:22 -0700222 CPDF_Path Path;
223 Path.AppendRect(left, bottom, right, top);
224
Tom Sepezae51c812015-08-05 12:34:06 -0700225 CPDF_ClipPath* pNewClipPath = new CPDF_ClipPath();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700226 pNewClipPath->AppendPath(Path, FXFILL_ALTERNATE, FALSE);
227 return pNewClipPath;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700228}
229
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700230DLLEXPORT void STDCALL FPDF_DestroyClipPath(FPDF_CLIPPATH clipPath) {
231 delete (CPDF_ClipPath*)clipPath;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700232}
233
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700234void OutputPath(CFX_ByteTextBuf& buf, CPDF_Path path) {
tsepez7d2a8d92016-06-08 11:51:23 -0700235 const CFX_PathData* pPathData = path.GetObject();
Lei Zhang412e9082015-12-14 18:34:00 -0800236 if (!pPathData)
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700237 return;
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700238
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700239 FX_PATHPOINT* pPoints = pPathData->GetPoints();
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700240
tsepez59601432016-08-29 14:26:57 -0700241 if (path.IsRect()) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700242 buf << (pPoints[0].m_PointX) << " " << (pPoints[0].m_PointY) << " "
243 << (pPoints[2].m_PointX - pPoints[0].m_PointX) << " "
244 << (pPoints[2].m_PointY - pPoints[0].m_PointY) << " re\n";
245 return;
246 }
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700247
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700248 CFX_ByteString temp;
249 for (int i = 0; i < pPathData->GetPointCount(); i++) {
250 buf << (pPoints[i].m_PointX) << " " << (pPoints[i].m_PointY);
251 int point_type = pPoints[i].m_Flag & FXPT_TYPE;
Dan Sinclair738b08c2016-03-01 14:45:20 -0500252 if (point_type == FXPT_MOVETO) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700253 buf << " m\n";
Dan Sinclair738b08c2016-03-01 14:45:20 -0500254 } else if (point_type == FXPT_BEZIERTO) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700255 buf << " " << (pPoints[i + 1].m_PointX) << " "
256 << (pPoints[i + 1].m_PointY) << " " << (pPoints[i + 2].m_PointX)
257 << " " << (pPoints[i + 2].m_PointY);
258 if (pPoints[i + 2].m_Flag & FXPT_CLOSEFIGURE)
259 buf << " c h\n";
260 else
261 buf << " c\n";
262 i += 2;
263 } else if (point_type == FXPT_LINETO) {
264 if (pPoints[i].m_Flag & FXPT_CLOSEFIGURE)
265 buf << " l h\n";
266 else
267 buf << " l\n";
268 }
269 }
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700270}
271
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700272DLLEXPORT void STDCALL FPDFPage_InsertClipPath(FPDF_PAGE page,
273 FPDF_CLIPPATH clipPath) {
Tom Sepezdb0be962015-10-16 14:00:21 -0700274 CPDF_Page* pPage = CPDFPageFromFPDFPage(page);
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700275 if (!pPage)
276 return;
Lei Zhang6ac0d2f2015-10-02 10:08:48 -0700277
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700278 CPDF_Dictionary* pPageDic = pPage->m_pFormDict;
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400279 CPDF_Object* pContentObj =
dsinclair38fd8442016-09-15 10:15:32 -0700280 pPageDic ? pPageDic->GetObjectFor("Contents") : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700281 if (!pContentObj)
dsinclair38fd8442016-09-15 10:15:32 -0700282 pContentObj = pPageDic ? pPageDic->GetArrayFor("Contents") : nullptr;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700283 if (!pContentObj)
284 return;
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700285
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700286 CFX_ByteTextBuf strClip;
287 CPDF_ClipPath* pClipPath = (CPDF_ClipPath*)clipPath;
tsepezc3255f52016-03-25 14:52:27 -0700288 uint32_t i;
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700289 for (i = 0; i < pClipPath->GetPathCount(); i++) {
290 CPDF_Path path = pClipPath->GetPath(i);
291 int iClipType = pClipPath->GetClipType(i);
tsepez59601432016-08-29 14:26:57 -0700292 if (path.GetPointCount() == 0) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700293 // Empty clipping (totally clipped out)
294 strClip << "0 0 m W n ";
295 } else {
296 OutputPath(strClip, path);
297 if (iClipType == FXFILL_WINDING)
298 strClip << "W n\n";
299 else
300 strClip << "W* n\n";
301 }
302 }
Tom Sepezae51c812015-08-05 12:34:06 -0700303 CPDF_Dictionary* pDic = new CPDF_Dictionary;
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400304 CPDF_Stream* pStream = new CPDF_Stream(nullptr, 0, pDic);
tsepeze6db16e2016-09-19 10:45:09 -0700305 pStream->SetData(strClip.GetBuffer(), strClip.GetSize());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700306 CPDF_Document* pDoc = pPage->m_pDocument;
307 if (!pDoc)
308 return;
tsepezbb577af2016-09-21 19:10:19 -0700309
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700310 pDoc->AddIndirectObject(pStream);
Lei Zhanga6d9f0e2015-06-13 00:48:38 -0700311
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400312 CPDF_Array* pContentArray = nullptr;
weilidb444d22016-06-02 15:48:15 -0700313 CPDF_Array* pArray = ToArray(pContentObj);
314 if (pArray) {
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400315 pContentArray = pArray;
Tom Sepezae51c812015-08-05 12:34:06 -0700316 CPDF_Reference* pRef = new CPDF_Reference(pDoc, pStream->GetObjNum());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700317 pContentArray->InsertAt(0, pRef);
Dan Sinclairbf81c142015-10-26 16:54:39 -0400318 } else if (CPDF_Reference* pReference = ToReference(pContentObj)) {
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700319 CPDF_Object* pDirectObj = pReference->GetDirect();
Dan Sinclair2b11dc12015-10-22 15:02:06 -0400320 if (pDirectObj) {
weilidb444d22016-06-02 15:48:15 -0700321 CPDF_Array* pObjArray = pDirectObj->AsArray();
322 if (pObjArray) {
323 pContentArray = pObjArray;
Nico Weber077f1a32015-08-06 15:08:57 -0700324 CPDF_Reference* pRef = new CPDF_Reference(pDoc, pStream->GetObjNum());
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700325 pContentArray->InsertAt(0, pRef);
Dan Sinclairaa435ba2015-10-22 16:45:48 -0400326 } else if (pDirectObj->IsStream()) {
Tom Sepezae51c812015-08-05 12:34:06 -0700327 pContentArray = new CPDF_Array();
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700328 pContentArray->AddReference(pDoc, pStream->GetObjNum());
329 pContentArray->AddReference(pDoc, pDirectObj->GetObjNum());
dsinclair38fd8442016-09-15 10:15:32 -0700330 pPageDic->SetReferenceFor("Contents", pDoc,
331 pDoc->AddIndirectObject(pContentArray));
Nico Weber9d8ec5a2015-08-04 13:00:21 -0700332 }
333 }
334 }
John Abd-El-Malek3f3b45c2014-05-23 17:28:10 -0700335}