7#include "fpdfsdk/cpdfsdk_pageview.h"
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"
30#include "fpdfsdk/fpdfxfa/cpdfxfa_page.h"
31#include "fpdfsdk/fpdfxfa/cpdfxfa_widget.h"
32#include "xfa/fxfa/cxfa_ffpageview.h"
50 if (!m_page->AsXFAPage()) {
55 m_page->AsPDFPage()->SetView(
nullptr);
59 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray)
62 m_SDKAnnotArray.clear();
75 m_curMatrix
= mtUser2Device;
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);
90 for (
const auto& pSDKAnnot : annot_iteration) {
91 pSDKAnnot->OnDraw(pDevice, mtUser2Device, pOptions->GetDrawAnnots());
95std::unique_ptr<CPDFSDK_Annot> CPDFSDK_PageView::NewAnnot(
CPDF_Annot* annot) {
98 CPDFSDK_InteractiveForm* form = m_pFormFillEnv->GetInteractiveForm();
105 auto ret =
std::make_unique<CPDFSDK_Widget>(annot,
this, form);
107 if (pdf_form->NeedConstructAP())
108 ret->ResetAppearance(absl::nullopt, CPDFSDK_Widget::kValueUnchanged);
113 if (sub_type == CPDF_Annot::Subtype::XFAWIDGET)
120CPDFSDK_Annot* CPDFSDK_PageView::GetFXAnnotAtPoint(
const CFX_PointF& point) {
122 for (
const auto& pSDKAnnot : annot_iteration) {
123 CFX_FloatRect rc = pSDKAnnot->GetViewBBox();
124 if (pSDKAnnot->GetAnnotSubtype() == CPDF_Annot::Subtype::POPUP)
126 if (rc.Contains(point))
127 return pSDKAnnot.Get();
132CPDFSDK_Annot* CPDFSDK_PageView::GetFXWidgetAtPoint(
const CFX_PointF& point) {
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;
138 do_hit_test = do_hit_test || sub_type == CPDF_Annot::Subtype::XFAWIDGET;
140 if (do_hit_test && pSDKAnnot->DoHitTest(point))
141 return pSDKAnnot.Get();
147CPDFSDK_Annot* CPDFSDK_PageView::AddAnnotForFFWidget(CXFA_FFWidget* pWidget) {
148 CPDFSDK_Annot* pSDKAnnot = GetAnnotForFFWidget(pWidget);
152 m_SDKAnnotArray.push_back(std::make_unique<CPDFXFA_Widget>(pWidget,
this));
153 return m_SDKAnnotArray.back().get();
156void CPDFSDK_PageView::DeleteAnnotForFFWidget(CXFA_FFWidget* pWidget) {
157 CPDFSDK_Annot* pAnnot = GetAnnotForFFWidget(pWidget);
161 IPDF_Page* pPage = pAnnot->GetXFAPage();
165 CPDF_Document::Extension* pContext = pPage->GetDocument()->GetExtension();
166 if (pContext && !pContext->ContainsExtensionForm())
169 ObservedPtr<CPDFSDK_Annot> pObserved(pAnnot);
170 if (GetFocusAnnot() == pAnnot)
171 m_pFormFillEnv->KillFocusAnnot({});
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);
180 if (m_pCaptureWidget.Get() == pAnnot)
181 m_pCaptureWidget.Reset();
184CPDFXFA_Page* CPDFSDK_PageView::XFAPageIfNotBackedByPDFPage() {
185 auto* pPage =
static_cast<CPDFXFA_Page*>(GetXFAPage());
186 return pPage && !pPage->AsPDFPage() ? pPage :
nullptr;
191 return m_page->GetDocument();
195 return ToPDFPage(m_page);
198CPDFSDK_InteractiveForm* CPDFSDK_PageView::GetInteractiveForm()
const {
199 return m_pFormFillEnv->GetInteractiveForm();
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());
211 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray) {
212 CPDF_Annot* pPDFAnnot = pAnnot->GetPDFAnnot();
213 if (pPDFAnnot && pPDFAnnot->GetAnnotDict() == pDict)
220CPDFSDK_Annot* CPDFSDK_PageView::GetAnnotForFFWidget(CXFA_FFWidget* pWidget) {
224 for (std::unique_ptr<CPDFSDK_Annot>& pAnnot : m_SDKAnnotArray) {
225 CPDFXFA_Widget* pCurrentWidget = pAnnot->AsXFAWidget();
226 if (pCurrentWidget && pCurrentWidget->GetXFAFFWidget() == pWidget)
232IPDF_Page* CPDFSDK_PageView::GetXFAPage() {
233 return ToXFAPage(m_page);
244 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
246 return pXFAPage->GetNextXFAAnnot(pAnnot);
254 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
256 return pXFAPage->GetPrevXFAAnnot(pAnnot);
264 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
266 return pXFAPage->GetFirstXFAAnnot(
this);
274 CPDFXFA_Page* pXFAPage = XFAPageIfNotBackedByPDFPage();
276 return pXFAPage->GetLastXFAAnnot(
this);
327 const CFX_PointF& point) {
328 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
330 m_pFormFillEnv->KillFocusAnnot(nFlags);
334 m_pFormFillEnv->SetFocusAnnot(pAnnot);
339 const CFX_PointF& point) {
340 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
342 m_pFormFillEnv->KillFocusAnnot(nFlags);
352 m_pFormFillEnv->SetFocusAnnot(pAnnot);
357 const CFX_PointF& point) {
358 ObservedPtr<
CPDFSDK_Annot> pFXAnnot(GetFXWidgetAtPoint(point));
360 if (pFocusAnnot && pFocusAnnot != pFXAnnot) {
369 const CFX_PointF& point) {
370 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
372 m_pFormFillEnv->KillFocusAnnot(nFlags);
382 m_pFormFillEnv->SetFocusAnnot(pAnnot);
387 const CFX_PointF& point) {
388 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
397 m_pFormFillEnv->SetFocusAnnot(pAnnot);
403 const CFX_PointF& point) {
404 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
413 m_pFormFillEnv->SetFocusAnnot(pAnnot);
419 const CFX_PointF& point) {
420 ObservedPtr<
CPDFSDK_Annot> pFXAnnot(GetFXAnnotAtPoint(point));
421 ObservedPtr<CPDFSDK_PageView> pThis(
this);
423 if (m_bOnWidget && m_pCaptureWidget != pFXAnnot)
424 ExitWidget(
true, nFlags);
427 if (!pThis || !pFXAnnot)
431 EnterWidget(pFXAnnot, nFlags);
438 ExitWidget(
false, nFlags);
446void CPDFSDK_PageView::EnterWidget(ObservedPtr<
CPDFSDK_Annot>& pAnnot,
447 Mask<FWL_EVENTFLAG> nFlags) {
449 m_pCaptureWidget.Reset(pAnnot.Get());
450 CPDFSDK_Annot::OnMouseEnter(m_pCaptureWidget, nFlags);
453void CPDFSDK_PageView::ExitWidget(
bool callExitCallback,
454 Mask<FWL_EVENTFLAG> nFlags) {
456 if (!m_pCaptureWidget)
459 if (callExitCallback) {
460 ObservedPtr<CPDFSDK_PageView> pThis(
this);
461 CPDFSDK_Annot::OnMouseExit(m_pCaptureWidget, nFlags);
468 m_pCaptureWidget.Reset();
472 const CFX_PointF& point,
473 const CFX_Vector& delta) {
474 ObservedPtr<
CPDFSDK_Annot> pAnnot(GetFXWidgetAtPoint(point));
491bool CPDFSDK_PageView::
OnChar(uint32_t nChar,
Mask<FWL_EVENTFLAG> nFlags) {
497 Mask<FWL_EVENTFLAG> nFlags) {
508 ObservedPtr<
CPDFSDK_Annot> end_annot(CPWL_Wnd::IsSHIFTKeyDown(nFlags)
509 ? GetLastFocusableAnnot()
510 : GetFirstFocusableAnnot());
511 return end_annot && m_pFormFillEnv->SetFocusAnnot(end_annot);
519 ObservedPtr<
CPDFSDK_Annot> pNext(CPWL_Wnd::IsSHIFTKeyDown(nFlags)
520 ? GetPrevAnnot(pFocusAnnot)
521 : GetNextAnnot(pFocusAnnot));
524 if (pNext.Get() != pFocusAnnot) {
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});
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();
564 m_pAnnotList = std::make_unique<CPDF_AnnotList>(pPage);
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);
571 std::unique_ptr<CPDFSDK_Annot> pAnnot = NewAnnot(pPDFAnnot);
574 m_SDKAnnotArray.push_back(std::move(pAnnot));
575 m_SDKAnnotArray.back()->OnLoad();
579void CPDFSDK_PageView::
UpdateRects(
const std::vector<CFX_FloatRect>& rects) {
580 for (
const auto& rc : rects)
581 m_pFormFillEnv->Invalidate(m_page, rc.GetOuterRect());
586 m_pFormFillEnv->Invalidate(m_page, rcWindow.GetOuterRect());
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;
597 return GetPageIndexForStaticPDF();
601 return p && m_pAnnotList->Contains(p);
605 return p && pdfium::Contains(m_SDKAnnotArray, fxcrt::MakeFakeUniquePtr(p));
609 CPDFSDK_Annot* pFocusAnnot = m_pFormFillEnv->GetFocusAnnot();
613int CPDFSDK_PageView::GetPageIndexForStaticPDF()
const {
CFX_Matrix & operator=(const CFX_Matrix &other)=default
CPDFSDK_AnnotIteration(CPDFSDK_PageView *page_view)
static CPDFSDK_AnnotIteration CreateForDrawing(CPDFSDK_PageView *page_view)
CPDFSDK_Annot * GetFirstAnnot()
CPDFSDK_Annot * GetLastAnnot()
CPDFSDK_Annot * GetNextAnnot(CPDFSDK_Annot *pAnnot)
CPDFSDK_Annot * GetPrevAnnot(CPDFSDK_Annot *pAnnot)
virtual bool SetIndexSelected(int index, bool selected)=0
virtual bool IsIndexSelected(int index)=0
static bool OnRButtonUp(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
static bool OnKeyDown(ObservedPtr< CPDFSDK_Annot > &pAnnot, FWL_VKEYCODE nKeyCode, Mask< FWL_EVENTFLAG > nFlags)
virtual void ReplaceSelection(const WideString &text)=0
static bool OnRButtonDown(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
virtual WideString GetText()=0
static bool OnMouseMove(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
virtual CFX_FloatRect GetRect() const =0
static bool OnLButtonUp(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
static bool OnMouseWheel(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point, const CFX_Vector &delta)
virtual WideString GetSelectedText()=0
virtual bool SelectAllText()=0
static bool OnChar(ObservedPtr< CPDFSDK_Annot > &pAnnot, uint32_t nChar, Mask< FWL_EVENTFLAG > nFlags)
virtual void ReplaceAndKeepSelection(const WideString &text)=0
static bool OnLButtonDblClk(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
static bool OnLButtonDown(ObservedPtr< CPDFSDK_Annot > &pAnnot, Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
CPDFSDK_Annot * GetFocusAnnot()
bool OnKeyDown(FWL_VKEYCODE nKeyCode, Mask< FWL_EVENTFLAG > nFlags)
CPDF_Document * GetPDFDocument()
std::vector< CPDFSDK_Annot * > GetAnnotList() const
bool OnMouseMove(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
CPDFSDK_Annot * GetNextAnnot(CPDFSDK_Annot *pAnnot)
CPDFSDK_Annot * GetPrevAnnot(CPDFSDK_Annot *pAnnot)
void ReplaceSelection(const WideString &text)
bool OnFocus(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
CPDFSDK_Annot * GetLastFocusableAnnot()
bool IsValidSDKAnnot(const CPDFSDK_Annot *p) const
bool IsBeingDestroyed() const
CPDFSDK_PageView(CPDFSDK_FormFillEnvironment *pFormFillEnv, IPDF_Page *page)
void PageView_OnDraw(CFX_RenderDevice *pDevice, const CFX_Matrix &mtUser2Device, CPDF_RenderOptions *pOptions, const FX_RECT &pClip)
bool OnRButtonUp(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
void ClearPage(CPDF_Page *pPage) override
bool OnLButtonDblClk(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
CPDFSDK_FormFillEnvironment * GetFormFillEnv() const
void UpdateRects(const std::vector< CFX_FloatRect > &rects)
CPDF_Page * GetPDFPage() const
void UpdateView(CPDFSDK_Annot *pAnnot)
bool OnRButtonDown(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
bool OnLButtonUp(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
WideString GetFocusedFormText()
CPDFSDK_Annot * GetAnnotByDict(const CPDF_Dictionary *pDict)
WideString GetSelectedText()
void ReplaceAndKeepSelection(const WideString &text)
bool OnLButtonDown(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point)
bool IsIndexSelected(int index)
bool OnMouseWheel(Mask< FWL_EVENTFLAG > nFlags, const CFX_PointF &point, const CFX_Vector &delta)
bool IsValidAnnot(const CPDF_Annot *p) const
bool SetIndexSelected(int index, bool selected)
CPDFSDK_Annot * GetFirstFocusableAnnot()
bool OnChar(uint32_t nChar, Mask< FWL_EVENTFLAG > nFlags)
Subtype GetSubtype() const
const CPDF_Dictionary * GetAnnotDict() const
int GetPageIndex(uint32_t objnum)
void SetView(View *pView)
static bool IsCTRLKeyDown(Mask< FWL_EVENTFLAG > nFlag)
static bool IsALTKeyDown(Mask< FWL_EVENTFLAG > nFlag)
virtual CPDFXFA_Page * AsXFAPage()=0
void CheckForUnsupportedAnnot(const CPDF_Annot *pAnnot)
CPDF_Page * ToPDFPage(IPDF_Page *pBase)