Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(99)

Side by Side Diff: core/fpdfdoc/doc_annot.cpp

Issue 2163133002: Move CPDF_Annot and CPDF_AnnotList to their own header. (Closed) Base URL: https://pdfium.googlesource.com/pdfium.git@master
Patch Set: Move CPDF_Annot and CPDF_AnnotList to their own headers. Created 4 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « core/fpdfdoc/cpdf_annotlist.cpp ('k') | core/fpdfdoc/include/fpdf_doc.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // 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.
4
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
7 #include "core/fpdfapi/fpdf_page/include/cpdf_form.h"
8 #include "core/fpdfapi/fpdf_page/include/cpdf_page.h"
9 #include "core/fpdfapi/fpdf_parser/include/cpdf_array.h"
10 #include "core/fpdfapi/fpdf_parser/include/cpdf_document.h"
11 #include "core/fpdfapi/fpdf_parser/include/cpdf_reference.h"
12 #include "core/fpdfapi/fpdf_render/include/cpdf_rendercontext.h"
13 #include "core/fpdfapi/fpdf_render/include/cpdf_renderoptions.h"
14 #include "core/fpdfdoc/cpvt_generateap.h"
15 #include "core/fpdfdoc/include/fpdf_doc.h"
16 #include "core/fxge/include/fx_ge.h"
17
18 CPDF_AnnotList::CPDF_AnnotList(CPDF_Page* pPage)
19 : m_pDocument(pPage->m_pDocument) {
20 if (!pPage->m_pFormDict)
21 return;
22
23 CPDF_Array* pAnnots = pPage->m_pFormDict->GetArrayBy("Annots");
24 if (!pAnnots)
25 return;
26
27 CPDF_Dictionary* pRoot = m_pDocument->GetRoot();
28 CPDF_Dictionary* pAcroForm = pRoot->GetDictBy("AcroForm");
29 FX_BOOL bRegenerateAP =
30 pAcroForm && pAcroForm->GetBooleanBy("NeedAppearances");
31 for (size_t i = 0; i < pAnnots->GetCount(); ++i) {
32 CPDF_Dictionary* pDict = ToDictionary(pAnnots->GetDirectObjectAt(i));
33 if (!pDict)
34 continue;
35
36 uint32_t dwObjNum = pDict->GetObjNum();
37 if (dwObjNum == 0) {
38 dwObjNum = m_pDocument->AddIndirectObject(pDict);
39 CPDF_Reference* pAction = new CPDF_Reference(m_pDocument, dwObjNum);
40 pAnnots->InsertAt(i, pAction);
41 pAnnots->RemoveAt(i + 1);
42 pDict = pAnnots->GetDictAt(i);
43 }
44 m_AnnotList.push_back(
45 std::unique_ptr<CPDF_Annot>(new CPDF_Annot(pDict, m_pDocument)));
46 if (bRegenerateAP && pDict->GetStringBy("Subtype") == "Widget" &&
47 CPDF_InterForm::IsUpdateAPEnabled()) {
48 FPDF_GenerateAP(m_pDocument, pDict);
49 }
50 }
51 }
52
53 CPDF_AnnotList::~CPDF_AnnotList() {}
54
55 void CPDF_AnnotList::DisplayPass(CPDF_Page* pPage,
56 CFX_RenderDevice* pDevice,
57 CPDF_RenderContext* pContext,
58 FX_BOOL bPrinting,
59 CFX_Matrix* pMatrix,
60 FX_BOOL bWidgetPass,
61 CPDF_RenderOptions* pOptions,
62 FX_RECT* clip_rect) {
63 for (const auto& pAnnot : m_AnnotList) {
64 bool bWidget = pAnnot->GetSubType() == "Widget";
65 if ((bWidgetPass && !bWidget) || (!bWidgetPass && bWidget))
66 continue;
67
68 uint32_t annot_flags = pAnnot->GetFlags();
69 if (annot_flags & ANNOTFLAG_HIDDEN)
70 continue;
71
72 if (bPrinting && (annot_flags & ANNOTFLAG_PRINT) == 0)
73 continue;
74
75 if (!bPrinting && (annot_flags & ANNOTFLAG_NOVIEW))
76 continue;
77
78 if (pOptions) {
79 CPDF_OCContext* pOCContext = pOptions->m_pOCContext;
80 CPDF_Dictionary* pAnnotDict = pAnnot->GetAnnotDict();
81 if (pOCContext && pAnnotDict &&
82 !pOCContext->CheckOCGVisible(pAnnotDict->GetDictBy("OC"))) {
83 continue;
84 }
85 }
86 CFX_FloatRect annot_rect_f;
87 pAnnot->GetRect(annot_rect_f);
88 CFX_Matrix matrix = *pMatrix;
89 if (clip_rect) {
90 annot_rect_f.Transform(&matrix);
91 FX_RECT annot_rect = annot_rect_f.GetOutterRect();
92 annot_rect.Intersect(*clip_rect);
93 if (annot_rect.IsEmpty()) {
94 continue;
95 }
96 }
97 if (pContext) {
98 pAnnot->DrawInContext(pPage, pContext, &matrix, CPDF_Annot::Normal);
99 } else if (!pAnnot->DrawAppearance(pPage, pDevice, &matrix,
100 CPDF_Annot::Normal, pOptions)) {
101 pAnnot->DrawBorder(pDevice, &matrix, pOptions);
102 }
103 }
104 }
105
106 void CPDF_AnnotList::DisplayAnnots(CPDF_Page* pPage,
107 CFX_RenderDevice* pDevice,
108 CPDF_RenderContext* pContext,
109 FX_BOOL bPrinting,
110 CFX_Matrix* pUser2Device,
111 uint32_t dwAnnotFlags,
112 CPDF_RenderOptions* pOptions,
113 FX_RECT* pClipRect) {
114 if (dwAnnotFlags & 0x01) {
115 DisplayPass(pPage, pDevice, pContext, bPrinting, pUser2Device, FALSE,
116 pOptions, pClipRect);
117 }
118 if (dwAnnotFlags & 0x02) {
119 DisplayPass(pPage, pDevice, pContext, bPrinting, pUser2Device, TRUE,
120 pOptions, pClipRect);
121 }
122 }
123
124 CPDF_Annot::CPDF_Annot(CPDF_Dictionary* pDict, CPDF_Document* pDocument)
125 : m_pAnnotDict(pDict),
126 m_pDocument(pDocument),
127 m_sSubtype(m_pAnnotDict->GetStringBy("Subtype")) {}
128
129 CPDF_Annot::~CPDF_Annot() {
130 ClearCachedAP();
131 }
132
133 void CPDF_Annot::ClearCachedAP() {
134 for (const auto& pair : m_APMap) {
135 delete pair.second;
136 }
137 m_APMap.clear();
138 }
139 CFX_ByteString CPDF_Annot::GetSubType() const {
140 return m_sSubtype;
141 }
142
143 void CPDF_Annot::GetRect(CFX_FloatRect& rect) const {
144 if (!m_pAnnotDict) {
145 return;
146 }
147 rect = m_pAnnotDict->GetRectBy("Rect");
148 rect.Normalize();
149 }
150
151 uint32_t CPDF_Annot::GetFlags() const {
152 return m_pAnnotDict->GetIntegerBy("F");
153 }
154
155 CPDF_Stream* FPDFDOC_GetAnnotAP(CPDF_Dictionary* pAnnotDict,
156 CPDF_Annot::AppearanceMode mode) {
157 CPDF_Dictionary* pAP = pAnnotDict->GetDictBy("AP");
158 if (!pAP) {
159 return nullptr;
160 }
161 const FX_CHAR* ap_entry = "N";
162 if (mode == CPDF_Annot::Down)
163 ap_entry = "D";
164 else if (mode == CPDF_Annot::Rollover)
165 ap_entry = "R";
166 if (!pAP->KeyExist(ap_entry))
167 ap_entry = "N";
168
169 CPDF_Object* psub = pAP->GetDirectObjectBy(ap_entry);
170 if (!psub)
171 return nullptr;
172 if (CPDF_Stream* pStream = psub->AsStream())
173 return pStream;
174
175 if (CPDF_Dictionary* pDict = psub->AsDictionary()) {
176 CFX_ByteString as = pAnnotDict->GetStringBy("AS");
177 if (as.IsEmpty()) {
178 CFX_ByteString value = pAnnotDict->GetStringBy("V");
179 if (value.IsEmpty()) {
180 CPDF_Dictionary* pParentDict = pAnnotDict->GetDictBy("Parent");
181 value = pParentDict ? pParentDict->GetStringBy("V") : CFX_ByteString();
182 }
183 if (value.IsEmpty() || !pDict->KeyExist(value))
184 as = "Off";
185 else
186 as = value;
187 }
188 return pDict->GetStreamBy(as);
189 }
190 return nullptr;
191 }
192
193 CPDF_Form* CPDF_Annot::GetAPForm(const CPDF_Page* pPage, AppearanceMode mode) {
194 CPDF_Stream* pStream = FPDFDOC_GetAnnotAP(m_pAnnotDict, mode);
195 if (!pStream)
196 return nullptr;
197
198 auto it = m_APMap.find(pStream);
199 if (it != m_APMap.end())
200 return it->second;
201
202 CPDF_Form* pNewForm =
203 new CPDF_Form(m_pDocument, pPage->m_pResources, pStream);
204 pNewForm->ParseContent(nullptr, nullptr, nullptr);
205 m_APMap[pStream] = pNewForm;
206 return pNewForm;
207 }
208
209 static CPDF_Form* FPDFDOC_Annot_GetMatrix(const CPDF_Page* pPage,
210 CPDF_Annot* pAnnot,
211 CPDF_Annot::AppearanceMode mode,
212 const CFX_Matrix* pUser2Device,
213 CFX_Matrix& matrix) {
214 CPDF_Form* pForm = pAnnot->GetAPForm(pPage, mode);
215 if (!pForm) {
216 return nullptr;
217 }
218 CFX_FloatRect form_bbox = pForm->m_pFormDict->GetRectBy("BBox");
219 CFX_Matrix form_matrix = pForm->m_pFormDict->GetMatrixBy("Matrix");
220 form_matrix.TransformRect(form_bbox);
221 CFX_FloatRect arect;
222 pAnnot->GetRect(arect);
223 matrix.MatchRect(arect, form_bbox);
224 matrix.Concat(*pUser2Device);
225 return pForm;
226 }
227 FX_BOOL CPDF_Annot::DrawAppearance(CPDF_Page* pPage,
228 CFX_RenderDevice* pDevice,
229 const CFX_Matrix* pUser2Device,
230 AppearanceMode mode,
231 const CPDF_RenderOptions* pOptions) {
232 CFX_Matrix matrix;
233 CPDF_Form* pForm =
234 FPDFDOC_Annot_GetMatrix(pPage, this, mode, pUser2Device, matrix);
235 if (!pForm) {
236 return FALSE;
237 }
238 CPDF_RenderContext context(pPage);
239 context.AppendLayer(pForm, &matrix);
240 context.Render(pDevice, pOptions, nullptr);
241 return TRUE;
242 }
243 FX_BOOL CPDF_Annot::DrawInContext(const CPDF_Page* pPage,
244 CPDF_RenderContext* pContext,
245 const CFX_Matrix* pUser2Device,
246 AppearanceMode mode) {
247 CFX_Matrix matrix;
248 CPDF_Form* pForm =
249 FPDFDOC_Annot_GetMatrix(pPage, this, mode, pUser2Device, matrix);
250 if (!pForm) {
251 return FALSE;
252 }
253 pContext->AppendLayer(pForm, &matrix);
254 return TRUE;
255 }
256 void CPDF_Annot::DrawBorder(CFX_RenderDevice* pDevice,
257 const CFX_Matrix* pUser2Device,
258 const CPDF_RenderOptions* pOptions) {
259 if (GetSubType() == "Popup") {
260 return;
261 }
262 uint32_t annot_flags = GetFlags();
263 if (annot_flags & ANNOTFLAG_HIDDEN) {
264 return;
265 }
266 bool bPrinting = pDevice->GetDeviceClass() == FXDC_PRINTER ||
267 (pOptions && (pOptions->m_Flags & RENDER_PRINTPREVIEW));
268 if (bPrinting && (annot_flags & ANNOTFLAG_PRINT) == 0) {
269 return;
270 }
271 if (!bPrinting && (annot_flags & ANNOTFLAG_NOVIEW)) {
272 return;
273 }
274 CPDF_Dictionary* pBS = m_pAnnotDict->GetDictBy("BS");
275 char style_char;
276 FX_FLOAT width;
277 CPDF_Array* pDashArray = nullptr;
278 if (!pBS) {
279 CPDF_Array* pBorderArray = m_pAnnotDict->GetArrayBy("Border");
280 style_char = 'S';
281 if (pBorderArray) {
282 width = pBorderArray->GetNumberAt(2);
283 if (pBorderArray->GetCount() == 4) {
284 pDashArray = pBorderArray->GetArrayAt(3);
285 if (!pDashArray) {
286 return;
287 }
288 size_t nLen = pDashArray->GetCount();
289 size_t i = 0;
290 for (; i < nLen; ++i) {
291 CPDF_Object* pObj = pDashArray->GetDirectObjectAt(i);
292 if (pObj && pObj->GetInteger()) {
293 break;
294 }
295 }
296 if (i == nLen) {
297 return;
298 }
299 style_char = 'D';
300 }
301 } else {
302 width = 1;
303 }
304 } else {
305 CFX_ByteString style = pBS->GetStringBy("S");
306 pDashArray = pBS->GetArrayBy("D");
307 style_char = style[1];
308 width = pBS->GetNumberBy("W");
309 }
310 if (width <= 0) {
311 return;
312 }
313 CPDF_Array* pColor = m_pAnnotDict->GetArrayBy("C");
314 uint32_t argb = 0xff000000;
315 if (pColor) {
316 int R = (int32_t)(pColor->GetNumberAt(0) * 255);
317 int G = (int32_t)(pColor->GetNumberAt(1) * 255);
318 int B = (int32_t)(pColor->GetNumberAt(2) * 255);
319 argb = ArgbEncode(0xff, R, G, B);
320 }
321 CFX_GraphStateData graph_state;
322 graph_state.m_LineWidth = width;
323 if (style_char == 'D') {
324 if (pDashArray) {
325 size_t dash_count = pDashArray->GetCount();
326 if (dash_count % 2) {
327 dash_count++;
328 }
329 graph_state.m_DashArray = FX_Alloc(FX_FLOAT, dash_count);
330 graph_state.m_DashCount = dash_count;
331 size_t i;
332 for (i = 0; i < pDashArray->GetCount(); ++i) {
333 graph_state.m_DashArray[i] = pDashArray->GetNumberAt(i);
334 }
335 if (i < dash_count) {
336 graph_state.m_DashArray[i] = graph_state.m_DashArray[i - 1];
337 }
338 } else {
339 graph_state.m_DashArray = FX_Alloc(FX_FLOAT, 2);
340 graph_state.m_DashCount = 2;
341 graph_state.m_DashArray[0] = graph_state.m_DashArray[1] = 3 * 1.0f;
342 }
343 }
344 CFX_FloatRect rect;
345 GetRect(rect);
346 CFX_PathData path;
347 width /= 2;
348 path.AppendRect(rect.left + width, rect.bottom + width, rect.right - width,
349 rect.top - width);
350 int fill_type = 0;
351 if (pOptions && (pOptions->m_Flags & RENDER_NOPATHSMOOTH)) {
352 fill_type |= FXFILL_NOPATHSMOOTH;
353 }
354 pDevice->DrawPath(&path, pUser2Device, &graph_state, argb, argb, fill_type);
355 }
OLDNEW
« no previous file with comments | « core/fpdfdoc/cpdf_annotlist.cpp ('k') | core/fpdfdoc/include/fpdf_doc.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698