1 // Copyright 2016 The PDFium Authors
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 "fpdfsdk/cpdfsdk_pageview.h"
8
9 #include <memory>
10 #include <utility>
11 #include <vector>
12
13 #include "core/fpdfapi/parser/cpdf_dictionary.h"
14 #include "core/fpdfapi/parser/cpdf_document.h"
15 #include "core/fpdfapi/render/cpdf_renderoptions.h"
16 #include "core/fpdfdoc/cpdf_annotlist.h"
17 #include "core/fpdfdoc/cpdf_interactiveform.h"
18 #include "core/fxcrt/autorestorer.h"
19 #include "core/fxcrt/stl_util.h"
20 #include "fpdfsdk/cpdfsdk_annot.h"
21 #include "fpdfsdk/cpdfsdk_annotiteration.h"
22 #include "fpdfsdk/cpdfsdk_annotiterator.h"
23 #include "fpdfsdk/cpdfsdk_formfillenvironment.h"
24 #include "fpdfsdk/cpdfsdk_helpers.h"
25 #include "fpdfsdk/cpdfsdk_interactiveform.h"
26 #include "third_party/base/check.h"
27 #include "third_party/base/containers/contains.h"
28
29 #ifdef PDF_ENABLE_XFA
30 #include "fpdfsdk/fpdfxfa/cpdfxfa_page.h"
31 #include "fpdfsdk/fpdfxfa/cpdfxfa_widget.h"
32 #include "xfa/fxfa/cxfa_ffpageview.h"
33 #endif // PDF_ENABLE_XFA
34
CPDFSDK_PageView(CPDFSDK_FormFillEnvironment * pFormFillEnv,IPDF_Page * page)35 CPDFSDK_PageView::CPDFSDK_PageView(CPDFSDK_FormFillEnvironment* pFormFillEnv,
36 IPDF_Page* page)
37 : m_page(page), m_pFormFillEnv(pFormFillEnv) {
38 DCHECK(m_page);
39 CPDF_Page* pPDFPage = ToPDFPage(page);
40 if (pPDFPage) {
41 CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm();
42 CPDF_InteractiveForm* pPDFForm = pForm->GetInteractiveForm();
43 pPDFForm->FixPageFields(pPDFPage);
44 if (!page->AsXFAPage())
45 pPDFPage->SetView(this);
46 }
47 }
48
~CPDFSDK_PageView()49 CPDFSDK_PageView::~CPDFSDK_PageView() {
50 if (!m_page->AsXFAPage()) {
51 // Deleting from `m_SDKAnnotArray` below can cause the page pointed to by
52 // `m_page` to be freed, which will cause issues if we try to cleanup the
53 // pageview pointer in `m_page`. So, reset the pageview pointer before doing
54 // anything else.
55 m_page->AsPDFPage()->SetView(nullptr);
56 }
57
58 // Manually reset elements to ensure they are deleted in order.
59 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray)
60 pAnnot.reset();
61
62 m_SDKAnnotArray.clear();
63 m_pAnnotList.reset();
64 }
65
ClearPage(CPDF_Page * pPage)66 void CPDFSDK_PageView::ClearPage(CPDF_Page* pPage) {
67 if (!IsBeingDestroyed())
68 GetFormFillEnv()->RemovePageView(pPage);
69 }
70
PageView_OnDraw(CFX_RenderDevice * pDevice,const CFX_Matrix & mtUser2Device,CPDF_RenderOptions * pOptions,const FX_RECT & pClip)71 void CPDFSDK_PageView::PageView_OnDraw(CFX_RenderDevice* pDevice,
72 const CFX_Matrix& mtUser2Device,
73 CPDF_RenderOptions* pOptions,
74 const FX_RECT& pClip) {
75 m_curMatrix = mtUser2Device;
76
77 #ifdef PDF_ENABLE_XFA
78 IPDF_Page* pPage = GetXFAPage();
79 CPDF_Document::Extension* pContext =
80 pPage ? pPage->GetDocument()->GetExtension() : nullptr;
81 if (pContext && pContext->ContainsExtensionFullForm()) {
82 static_cast<CPDFXFA_Page*>(pPage)->DrawFocusAnnot(pDevice, GetFocusAnnot(),
83 mtUser2Device, pClip);
84 return;
85 }
86 #endif // PDF_ENABLE_XFA
87
88 // for pdf/static xfa.
89 auto annot_iteration = CPDFSDK_AnnotIteration::CreateForDrawing(this);
90 for (const auto& pSDKAnnot : annot_iteration) {
91 pSDKAnnot->OnDraw(pDevice, mtUser2Device, pOptions->GetDrawAnnots());
92 }
93 }
94
NewAnnot(CPDF_Annot * annot)95 std::unique_ptr<CPDFSDK_Annot> CPDFSDK_PageView::NewAnnot(CPDF_Annot* annot) {
96 const CPDF_Annot::Subtype sub_type = annot->GetSubtype();
97 if (sub_type == CPDF_Annot::Subtype::WIDGET) {
98 CPDFSDK_InteractiveForm* form = m_pFormFillEnv->GetInteractiveForm();
99 CPDF_InteractiveForm* pdf_form = form->GetInteractiveForm();
100 CPDF_FormControl* form_control =
101 pdf_form->GetControlByDict(annot->GetAnnotDict());
102 if (!form_control)
103 return nullptr;
104
105 auto ret = std::make_unique<CPDFSDK_Widget>(annot, this, form);
106 form->AddMap(form_control, ret.get());
107 if (pdf_form->NeedConstructAP())
108 ret->ResetAppearance(absl::nullopt, CPDFSDK_Widget::kValueUnchanged);
109 return ret;
110 }
111
112 #ifdef PDF_ENABLE_XFA
113 if (sub_type == CPDF_Annot::Subtype::XFAWIDGET)
114 return nullptr;
115 #endif // PDF_ENABLE_XFA
116
117 return std::make_unique<CPDFSDK_BAAnnot>(annot, this);
118 }
119
GetFXAnnotAtPoint(const CFX_PointF & point)120 CPDFSDK_Annot* CPDFSDK_PageView::GetFXAnnotAtPoint(const CFX_PointF& point) {
121 CPDFSDK_AnnotIteration annot_iteration(this);
122 for (const auto& pSDKAnnot : annot_iteration) {
123 CFX_FloatRect rc = pSDKAnnot->GetViewBBox();
124 if (pSDKAnnot->GetAnnotSubtype() == CPDF_Annot::Subtype::POPUP)
125 continue;
126 if (rc.Contains(point))
127 return pSDKAnnot.Get();
128 }
129 return nullptr;
130 }
131
GetFXWidgetAtPoint(const CFX_PointF & point)132 CPDFSDK_Annot* CPDFSDK_PageView::GetFXWidgetAtPoint(const CFX_PointF& point) {
133 CPDFSDK_AnnotIteration annot_iteration(this);
134 for (const auto& pSDKAnnot : annot_iteration) {
135 const CPDF_Annot::Subtype sub_type = pSDKAnnot->GetAnnotSubtype();
136 bool do_hit_test = sub_type == CPDF_Annot::Subtype::WIDGET;
137 #ifdef PDF_ENABLE_XFA
138 do_hit_test = do_hit_test || sub_type == CPDF_Annot::Subtype::XFAWIDGET;
139 #endif // PDF_ENABLE_XFA
140 if (do_hit_test && pSDKAnnot->DoHitTest(point))
141 return pSDKAnnot.Get();
142 }
143 return nullptr;
144 }
145
146 #ifdef PDF_ENABLE_XFA
AddAnnotForFFWidget(CXFA_FFWidget * pWidget)147 CPDFSDK_Annot* CPDFSDK_PageView::AddAnnotForFFWidget(CXFA_FFWidget* pWidget) {
148 CPDFSDK_Annot* pSDKAnnot = GetAnnotForFFWidget(pWidget);
149 if (pSDKAnnot)
150 return pSDKAnnot;
151
152 m_SDKAnnotArray.push_back(std::make_unique<CPDFXFA_Widget>(pWidget, this));
153 return m_SDKAnnotArray.back().get();
154 }
155
DeleteAnnotForFFWidget(CXFA_FFWidget * pWidget)156 void CPDFSDK_PageView::DeleteAnnotForFFWidget(CXFA_FFWidget* pWidget) {
157 CPDFSDK_Annot* pAnnot = GetAnnotForFFWidget(pWidget);
158 if (!pAnnot)
159 return;
160
161 IPDF_Page* pPage = pAnnot->GetXFAPage();
162 if (!pPage)
163 return;
164
165 CPDF_Document::Extension* pContext = pPage->GetDocument()->GetExtension();
166 if (pContext && !pContext->ContainsExtensionForm())
167 return;
168
169 ObservedPtr<CPDFSDK_Annot> pObserved(pAnnot);
170 if (GetFocusAnnot() == pAnnot)
171 m_pFormFillEnv->KillFocusAnnot({}); // May invoke JS, invalidating pAnnot.
172
173 if (pObserved) {
174 auto it = std::find(m_SDKAnnotArray.begin(), m_SDKAnnotArray.end(),
175 fxcrt::MakeFakeUniquePtr(pAnnot));
176 if (it != m_SDKAnnotArray.end())
177 m_SDKAnnotArray.erase(it);
178 }
179
180 if (m_pCaptureWidget.Get() == pAnnot)
181 m_pCaptureWidget.Reset();
182 }
183
XFAPageIfNotBackedByPDFPage()184 CPDFXFA_Page* CPDFSDK_PageView::XFAPageIfNotBackedByPDFPage() {
185 auto* pPage = static_cast<CPDFXFA_Page*>(GetXFAPage());
186 return pPage && !pPage->AsPDFPage() ? pPage : nullptr;
187 }
188 #endif // PDF_ENABLE_XFA
189
GetPDFDocument()190 CPDF_Document* CPDFSDK_PageView::GetPDFDocument() {
191 return m_page->GetDocument();
192 }
193
GetPDFPage() const194 CPDF_Page* CPDFSDK_PageView::GetPDFPage() const {
195 return ToPDFPage(m_page);
196 }
197
GetInteractiveForm() const198 CPDFSDK_InteractiveForm* CPDFSDK_PageView::GetInteractiveForm() const {
199 return m_pFormFillEnv->GetInteractiveForm();
200 }
201
GetAnnotList() const202 std::vector<CPDFSDK_Annot*> CPDFSDK_PageView::GetAnnotList() const {
203 std::vector<CPDFSDK_Annot*> list;
204 list.reserve(m_SDKAnnotArray.size());
205 for (const std::unique_ptr<CPDFSDK_Annot>& elem : m_SDKAnnotArray)
206 list.push_back(elem.get());
207 return list;
208 }
209
GetAnnotByDict(const CPDF_Dictionary * pDict)210 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotByDict(const CPDF_Dictionary* pDict) {
211 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray) {
212 CPDF_Annot* pPDFAnnot = pAnnot->GetPDFAnnot();
213 if (pPDFAnnot && pPDFAnnot->GetAnnotDict() == pDict)
214 return pAnnot.get();
215 }
216 return nullptr;
217 }
218
219 #ifdef PDF_ENABLE_XFA
GetAnnotForFFWidget(CXFA_FFWidget * pWidget)220 CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotForFFWidget(CXFA_FFWidget* pWidget) {
221 if (!pWidget)
222 return nullptr;
223
224 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray) {
225 CPDFXFA_Widget* pCurrentWidget = pAnnot->AsXFAWidget();
226 if (pCurrentWidget && pCurrentWidget->GetXFAFFWidget() == pWidget)
227 return pAnnot.get();
228 }
229 return nullptr;
230 }
231
GetXFAPage()232 IPDF_Page* CPDFSDK_PageView::GetXFAPage() {
233 return ToXFAPage(m_page);
234 }
235 #endif // PDF_ENABLE_XFA
236
GetFocusedFormText()237 WideString CPDFSDK_PageView::GetFocusedFormText() {
238 CPDFSDK_Annot* annot = GetFocusAnnot();
239 return annot ? annot->GetText() : WideString();
240 }
241
GetNextAnnot(CPDFSDK_Annot * pAnnot)242 CPDFSDK_Annot* CPDFSDK_PageView::GetNextAnnot(CPDFSDK_Annot* pAnnot) {
243 #ifdef PDF_ENABLE_XFA
244 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
245 if (pXFAPage)
246 return pXFAPage->GetNextXFAAnnot(pAnnot);
247 #endif // PDF_ENABLE_XFA
248 CPDFSDK_AnnotIterator ai(this, GetFormFillEnv()->GetFocusableAnnotSubtypes());
249 return ai.GetNextAnnot(pAnnot);
250 }
251
GetPrevAnnot(CPDFSDK_Annot * pAnnot)252 CPDFSDK_Annot* CPDFSDK_PageView::GetPrevAnnot(CPDFSDK_Annot* pAnnot) {
253 #ifdef PDF_ENABLE_XFA
254 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
255 if (pXFAPage)
256 return pXFAPage->GetPrevXFAAnnot(pAnnot);
257 #endif // PDF_ENABLE_XFA
258 CPDFSDK_AnnotIterator ai(this, GetFormFillEnv()->GetFocusableAnnotSubtypes());
259 return ai.GetPrevAnnot(pAnnot);
260 }
261
GetFirstFocusableAnnot()262 CPDFSDK_Annot* CPDFSDK_PageView::GetFirstFocusableAnnot() {
263 #ifdef PDF_ENABLE_XFA
264 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
265 if (pXFAPage)
266 return pXFAPage->GetFirstXFAAnnot(this);
267 #endif // PDF_ENABLE_XFA
268 CPDFSDK_AnnotIterator ai(this, GetFormFillEnv()->GetFocusableAnnotSubtypes());
269 return ai.GetFirstAnnot();
270 }
271
GetLastFocusableAnnot()272 CPDFSDK_Annot* CPDFSDK_PageView::GetLastFocusableAnnot() {
273 #ifdef PDF_ENABLE_XFA
274 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
275 if (pXFAPage)
276 return pXFAPage->GetLastXFAAnnot(this);
277 #endif // PDF_ENABLE_XFA
278 CPDFSDK_AnnotIterator ai(this, GetFormFillEnv()->GetFocusableAnnotSubtypes());
279 return ai.GetLastAnnot();
280 }
281
GetSelectedText()282 WideString CPDFSDK_PageView::GetSelectedText() {
283 CPDFSDK_Annot* annot = GetFocusAnnot();
284 if (!annot)
285 return WideString();
286 return annot->GetSelectedText();
287 }
288
ReplaceAndKeepSelection(const WideString & text)289 void CPDFSDK_PageView::ReplaceAndKeepSelection(const WideString& text) {
290 CPDFSDK_Annot* annot = GetFocusAnnot();
291 if (annot)
292 annot->ReplaceAndKeepSelection(text);
293 }
294
ReplaceSelection(const WideString & text)295 void CPDFSDK_PageView::ReplaceSelection(const WideString& text) {
296 CPDFSDK_Annot* annot = GetFocusAnnot();
297 if (annot)
298 annot->ReplaceSelection(text);
299 }
300
SelectAllText()301 bool CPDFSDK_PageView::SelectAllText() {
302 CPDFSDK_Annot* annot = GetFocusAnnot();
303 return annot && annot->SelectAllText();
304 }
305
CanUndo()306 bool CPDFSDK_PageView::CanUndo() {
307 CPDFSDK_Annot* annot = GetFocusAnnot();
308 return annot && annot->CanUndo();
309 }
310
CanRedo()311 bool CPDFSDK_PageView::CanRedo() {
312 CPDFSDK_Annot* annot = GetFocusAnnot();
313 return annot && annot->CanRedo();
314 }
315
Undo()316 bool CPDFSDK_PageView::Undo() {
317 CPDFSDK_Annot* annot = GetFocusAnnot();
318 return annot && annot->Undo();
319 }
320
Redo()321 bool CPDFSDK_PageView::Redo() {
322 CPDFSDK_Annot* annot = GetFocusAnnot();
323 return annot && annot->Redo();
324 }
325
OnFocus(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)326 bool CPDFSDK_PageView::OnFocus(Mask<FWL_EVENTFLAG> nFlags,
327 const CFX_PointF& point) {
328 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
329 if (!pAnnot) {
330 m_pFormFillEnv->KillFocusAnnot(nFlags);
331 return false;
332 }
333
334 m_pFormFillEnv->SetFocusAnnot(pAnnot);
335 return true;
336 }
337
OnLButtonDown(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)338 bool CPDFSDK_PageView::OnLButtonDown(Mask<FWL_EVENTFLAG> nFlags,
339 const CFX_PointF& point) {
340 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
341 if (!pAnnot) {
342 m_pFormFillEnv->KillFocusAnnot(nFlags);
343 return false;
344 }
345
346 if (!CPDFSDK_Annot::OnLButtonDown(pAnnot, nFlags, point))
347 return false;
348
349 if (!pAnnot)
350 return false;
351
352 m_pFormFillEnv->SetFocusAnnot(pAnnot);
353 return true;
354 }
355
OnLButtonUp(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)356 bool CPDFSDK_PageView::OnLButtonUp(Mask<FWL_EVENTFLAG> nFlags,
357 const CFX_PointF& point) {
358 ObservedPtr<CPDFSDK_Annot> pFXAnnot(GetFXWidgetAtPoint(point));
359 ObservedPtr<CPDFSDK_Annot> pFocusAnnot(GetFocusAnnot());
360 if (pFocusAnnot && pFocusAnnot != pFXAnnot) {
361 // Last focus Annot gets a chance to handle the event.
362 if (CPDFSDK_Annot::OnLButtonUp(pFocusAnnot, nFlags, point))
363 return true;
364 }
365 return pFXAnnot && CPDFSDK_Annot::OnLButtonUp(pFXAnnot, nFlags, point);
366 }
367
OnLButtonDblClk(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)368 bool CPDFSDK_PageView::OnLButtonDblClk(Mask<FWL_EVENTFLAG> nFlags,
369 const CFX_PointF& point) {
370 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
371 if (!pAnnot) {
372 m_pFormFillEnv->KillFocusAnnot(nFlags);
373 return false;
374 }
375
376 if (!CPDFSDK_Annot::OnLButtonDblClk(pAnnot, nFlags, point))
377 return false;
378
379 if (!pAnnot)
380 return false;
381
382 m_pFormFillEnv->SetFocusAnnot(pAnnot);
383 return true;
384 }
385
OnRButtonDown(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)386 bool CPDFSDK_PageView::OnRButtonDown(Mask<FWL_EVENTFLAG> nFlags,
387 const CFX_PointF& point) {
388 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
389 if (!pAnnot)
390 return false;
391
392 bool ok = CPDFSDK_Annot::OnRButtonDown(pAnnot, nFlags, point);
393 if (!pAnnot)
394 return false;
395
396 if (ok)
397 m_pFormFillEnv->SetFocusAnnot(pAnnot);
398
399 return true;
400 }
401
OnRButtonUp(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)402 bool CPDFSDK_PageView::OnRButtonUp(Mask<FWL_EVENTFLAG> nFlags,
403 const CFX_PointF& point) {
404 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
405 if (!pAnnot)
406 return false;
407
408 bool ok = CPDFSDK_Annot::OnRButtonUp(pAnnot, nFlags, point);
409 if (!pAnnot)
410 return false;
411
412 if (ok)
413 m_pFormFillEnv->SetFocusAnnot(pAnnot);
414
415 return true;
416 }
417
OnMouseMove(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point)418 bool CPDFSDK_PageView::OnMouseMove(Mask<FWL_EVENTFLAG> nFlags,
419 const CFX_PointF& point) {
420 ObservedPtr<CPDFSDK_Annot> pFXAnnot(GetFXAnnotAtPoint(point));
421 ObservedPtr<CPDFSDK_PageView> pThis(this);
422
423 if (m_bOnWidget && m_pCaptureWidget != pFXAnnot)
424 ExitWidget(true, nFlags);
425
426 // ExitWidget() may have invalidated objects.
427 if (!pThis || !pFXAnnot)
428 return false;
429
430 if (!m_bOnWidget) {
431 EnterWidget(pFXAnnot, nFlags);
432
433 // EnterWidget() may have invalidated objects.
434 if (!pThis)
435 return false;
436
437 if (!pFXAnnot) {
438 ExitWidget(false, nFlags);
439 return true;
440 }
441 }
442 CPDFSDK_Annot::OnMouseMove(pFXAnnot, nFlags, point);
443 return true;
444 }
445
EnterWidget(ObservedPtr<CPDFSDK_Annot> & pAnnot,Mask<FWL_EVENTFLAG> nFlags)446 void CPDFSDK_PageView::EnterWidget(ObservedPtr<CPDFSDK_Annot>& pAnnot,
447 Mask<FWL_EVENTFLAG> nFlags) {
448 m_bOnWidget = true;
449 m_pCaptureWidget.Reset(pAnnot.Get());
450 CPDFSDK_Annot::OnMouseEnter(m_pCaptureWidget, nFlags);
451 }
452
ExitWidget(bool callExitCallback,Mask<FWL_EVENTFLAG> nFlags)453 void CPDFSDK_PageView::ExitWidget(bool callExitCallback,
454 Mask<FWL_EVENTFLAG> nFlags) {
455 m_bOnWidget = false;
456 if (!m_pCaptureWidget)
457 return;
458
459 if (callExitCallback) {
460 ObservedPtr<CPDFSDK_PageView> pThis(this);
461 CPDFSDK_Annot::OnMouseExit(m_pCaptureWidget, nFlags);
462
463 // OnMouseExit() may have invalidated |this|.
464 if (!pThis)
465 return;
466 }
467
468 m_pCaptureWidget.Reset();
469 }
470
OnMouseWheel(Mask<FWL_EVENTFLAG> nFlags,const CFX_PointF & point,const CFX_Vector & delta)471 bool CPDFSDK_PageView::OnMouseWheel(Mask<FWL_EVENTFLAG> nFlags,
472 const CFX_PointF& point,
473 const CFX_Vector& delta) {
474 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
475 if (!pAnnot)
476 return false;
477
478 return CPDFSDK_Annot::OnMouseWheel(pAnnot, nFlags, point, delta);
479 }
480
SetIndexSelected(int index,bool selected)481 bool CPDFSDK_PageView::SetIndexSelected(int index, bool selected) {
482 CPDFSDK_Annot* annot = GetFocusAnnot();
483 return annot && annot->SetIndexSelected(index, selected);
484 }
485
IsIndexSelected(int index)486 bool CPDFSDK_PageView::IsIndexSelected(int index) {
487 CPDFSDK_Annot* annot = GetFocusAnnot();
488 return annot && annot->IsIndexSelected(index);
489 }
490
OnChar(uint32_t nChar,Mask<FWL_EVENTFLAG> nFlags)491 bool CPDFSDK_PageView::OnChar(uint32_t nChar, Mask<FWL_EVENTFLAG> nFlags) {
492 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFocusAnnot());
493 return pAnnot && CPDFSDK_Annot::OnChar(pAnnot, nChar, nFlags);
494 }
495
OnKeyDown(FWL_VKEYCODE nKeyCode,Mask<FWL_EVENTFLAG> nFlags)496 bool CPDFSDK_PageView::OnKeyDown(FWL_VKEYCODE nKeyCode,
497 Mask<FWL_EVENTFLAG> nFlags) {
498 ObservedPtr<CPDFSDK_Annot> pAnnot(GetFocusAnnot());
499 if (!pAnnot) {
500 // If pressed key is not tab then no action is needed.
501 if (nKeyCode != FWL_VKEY_Tab)
502 return false;
503
504 // If ctrl key or alt key is pressed, then no action is needed.
505 if (CPWL_Wnd::IsCTRLKeyDown(nFlags) || CPWL_Wnd::IsALTKeyDown(nFlags))
506 return false;
507
508 ObservedPtr<CPDFSDK_Annot> end_annot(CPWL_Wnd::IsSHIFTKeyDown(nFlags)
509 ? GetLastFocusableAnnot()
510 : GetFirstFocusableAnnot());
511 return end_annot && m_pFormFillEnv->SetFocusAnnot(end_annot);
512 }
513
514 if (CPWL_Wnd::IsCTRLKeyDown(nFlags) || CPWL_Wnd::IsALTKeyDown(nFlags))
515 return CPDFSDK_Annot::OnKeyDown(pAnnot, nKeyCode, nFlags);
516
517 CPDFSDK_Annot* pFocusAnnot = GetFocusAnnot();
518 if (pFocusAnnot && (nKeyCode == FWL_VKEY_Tab)) {
519 ObservedPtr<CPDFSDK_Annot> pNext(CPWL_Wnd::IsSHIFTKeyDown(nFlags)
520 ? GetPrevAnnot(pFocusAnnot)
521 : GetNextAnnot(pFocusAnnot));
522 if (!pNext)
523 return false;
524 if (pNext.Get() != pFocusAnnot) {
525 GetFormFillEnv()->SetFocusAnnot(pNext);
526 return true;
527 }
528 }
529
530 // Check |pAnnot| again because JS may have destroyed it in GetNextAnnot().
531 if (!pAnnot)
532 return false;
533
534 return CPDFSDK_Annot::OnKeyDown(pAnnot, nKeyCode, nFlags);
535 }
536
LoadFXAnnots()537 void CPDFSDK_PageView::LoadFXAnnots() {
538 AutoRestorer<bool> lock(&m_bLocked);
539 m_bLocked = true;
540
541 #ifdef PDF_ENABLE_XFA
542 RetainPtr<CPDFXFA_Page> protector(ToXFAPage(m_page));
543 CPDF_Document::Extension* pContext = m_pFormFillEnv->GetDocExtension();
544 if (pContext && pContext->ContainsExtensionFullForm()) {
545 CXFA_FFPageView* pageView = protector->GetXFAPageView();
546 CXFA_FFPageWidgetIterator pWidgetHandler(
547 pageView, Mask<XFA_WidgetStatus>{XFA_WidgetStatus::kVisible,
548 XFA_WidgetStatus::kViewable});
549
550 while (CXFA_FFWidget* pXFAAnnot = pWidgetHandler.MoveToNext()) {
551 m_SDKAnnotArray.push_back(
552 std::make_unique<CPDFXFA_Widget>(pXFAAnnot, this));
553 m_SDKAnnotArray.back()->OnLoad();
554 }
555 return;
556 }
557 #endif // PDF_ENABLE_XFA
558
559 CPDF_Page* pPage = GetPDFPage();
560 DCHECK(pPage);
561 bool bUpdateAP = CPDF_InteractiveForm::IsUpdateAPEnabled();
562 // Disable the default AP construction.
563 CPDF_InteractiveForm::SetUpdateAP(false);
564 m_pAnnotList = std::make_unique<CPDF_AnnotList>(pPage);
565 CPDF_InteractiveForm::SetUpdateAP(bUpdateAP);
566
567 const size_t nCount = m_pAnnotList->Count();
568 for (size_t i = 0; i < nCount; ++i) {
569 CPDF_Annot* pPDFAnnot = m_pAnnotList->GetAt(i);
570 CheckForUnsupportedAnnot(pPDFAnnot);
571 std::unique_ptr<CPDFSDK_Annot> pAnnot = NewAnnot(pPDFAnnot);
572 if (!pAnnot)
573 continue;
574 m_SDKAnnotArray.push_back(std::move(pAnnot));
575 m_SDKAnnotArray.back()->OnLoad();
576 }
577 }
578
UpdateRects(const std::vector<CFX_FloatRect> & rects)579 void CPDFSDK_PageView::UpdateRects(const std::vector<CFX_FloatRect>& rects) {
580 for (const auto& rc : rects)
581 m_pFormFillEnv->Invalidate(m_page, rc.GetOuterRect());
582 }
583
UpdateView(CPDFSDK_Annot * pAnnot)584 void CPDFSDK_PageView::UpdateView(CPDFSDK_Annot* pAnnot) {
585 CFX_FloatRect rcWindow = pAnnot->GetRect();
586 m_pFormFillEnv->Invalidate(m_page, rcWindow.GetOuterRect());
587 }
588
GetPageIndex() const589 int CPDFSDK_PageView::GetPageIndex() const {
590 #ifdef PDF_ENABLE_XFA
591 CPDF_Document::Extension* pContext = m_page->GetDocument()->GetExtension();
592 if (pContext && pContext->ContainsExtensionFullForm()) {
593 CXFA_FFPageView* pPageView = m_page->AsXFAPage()->GetXFAPageView();
594 return pPageView ? pPageView->GetLayoutItem()->GetPageIndex() : -1;
595 }
596 #endif // PDF_ENABLE_XFA
597 return GetPageIndexForStaticPDF();
598 }
599
IsValidAnnot(const CPDF_Annot * p) const600 bool CPDFSDK_PageView::IsValidAnnot(const CPDF_Annot* p) const {
601 return p && m_pAnnotList->Contains(p);
602 }
603
IsValidSDKAnnot(const CPDFSDK_Annot * p) const604 bool CPDFSDK_PageView::IsValidSDKAnnot(const CPDFSDK_Annot* p) const {
605 return p && pdfium::Contains(m_SDKAnnotArray, fxcrt::MakeFakeUniquePtr(p));
606 }
607
GetFocusAnnot()608 CPDFSDK_Annot* CPDFSDK_PageView::GetFocusAnnot() {
609 CPDFSDK_Annot* pFocusAnnot = m_pFormFillEnv->GetFocusAnnot();
610 return IsValidSDKAnnot(pFocusAnnot) ? pFocusAnnot : nullptr;
611 }
612
GetPageIndexForStaticPDF() const613 int CPDFSDK_PageView::GetPageIndexForStaticPDF() const {
614 CPDF_Document* pDoc = m_pFormFillEnv->GetPDFDocument();
615 return pDoc->GetPageIndex(GetPDFPage()->GetDict()->GetObjNum());
616 }
617