| // 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 "fpdfsdk/formfiller/cffl_interactiveformfiller.h" | 
 |  | 
 | #include "constants/access_permissions.h" | 
 | #include "constants/ascii.h" | 
 | #include "constants/form_flags.h" | 
 | #include "core/fpdfapi/page/cpdf_page.h" | 
 | #include "core/fxcrt/autorestorer.h" | 
 | #include "core/fxge/cfx_drawutils.h" | 
 | #include "fpdfsdk/cpdfsdk_pageview.h" | 
 | #include "fpdfsdk/cpdfsdk_widget.h" | 
 | #include "fpdfsdk/formfiller/cffl_checkbox.h" | 
 | #include "fpdfsdk/formfiller/cffl_combobox.h" | 
 | #include "fpdfsdk/formfiller/cffl_formfield.h" | 
 | #include "fpdfsdk/formfiller/cffl_listbox.h" | 
 | #include "fpdfsdk/formfiller/cffl_perwindowdata.h" | 
 | #include "fpdfsdk/formfiller/cffl_pushbutton.h" | 
 | #include "fpdfsdk/formfiller/cffl_radiobutton.h" | 
 | #include "fpdfsdk/formfiller/cffl_textfield.h" | 
 | #include "public/fpdf_fwlevent.h" | 
 | #include "third_party/base/check.h" | 
 | #include "third_party/base/cxx17_backports.h" | 
 |  | 
 | CFFL_InteractiveFormFiller::CFFL_InteractiveFormFiller( | 
 |     CallbackIface* pCallbackIface) | 
 |     : m_pCallbackIface(pCallbackIface) {} | 
 |  | 
 | CFFL_InteractiveFormFiller::~CFFL_InteractiveFormFiller() = default; | 
 |  | 
 | bool CFFL_InteractiveFormFiller::Annot_HitTest(const CPDFSDK_Widget* pWidget, | 
 |                                                const CFX_PointF& point) { | 
 |   return pWidget->GetRect().Contains(point); | 
 | } | 
 |  | 
 | FX_RECT CFFL_InteractiveFormFiller::GetViewBBox( | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     CPDFSDK_Widget* pWidget) { | 
 |   if (CFFL_FormField* pFormField = GetFormField(pWidget)) | 
 |     return pFormField->GetViewBBox(pPageView); | 
 |  | 
 |   DCHECK(pPageView); | 
 |  | 
 |   CPDF_Annot* pPDFAnnot = pWidget->GetPDFAnnot(); | 
 |   CFX_FloatRect rcWin = pPDFAnnot->GetRect(); | 
 |   if (!rcWin.IsEmpty()) { | 
 |     rcWin.Inflate(1, 1); | 
 |     rcWin.Normalize(); | 
 |   } | 
 |   return rcWin.GetOuterRect(); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnDraw(CPDFSDK_PageView* pPageView, | 
 |                                         CPDFSDK_Widget* pWidget, | 
 |                                         CFX_RenderDevice* pDevice, | 
 |                                         const CFX_Matrix& mtUser2Device) { | 
 |   DCHECK(pPageView); | 
 |   if (!IsVisible(pWidget)) | 
 |     return; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   if (pFormField && pFormField->IsValid()) { | 
 |     pFormField->OnDraw(pPageView, pWidget, pDevice, mtUser2Device); | 
 |     if (m_pCallbackIface->GetFocusAnnot() != pWidget) | 
 |       return; | 
 |  | 
 |     CFX_FloatRect rcFocus = pFormField->GetFocusBox(pPageView); | 
 |     if (rcFocus.IsEmpty()) | 
 |       return; | 
 |  | 
 |     CFX_DrawUtils::DrawFocusRect(pDevice, mtUser2Device, rcFocus); | 
 |  | 
 |     return; | 
 |   } | 
 |  | 
 |   if (pFormField) { | 
 |     pFormField->OnDrawDeactive(pPageView, pWidget, pDevice, mtUser2Device); | 
 |   } else { | 
 |     pWidget->DrawAppearance(pDevice, mtUser2Device, | 
 |                             CPDF_Annot::AppearanceMode::kNormal); | 
 |   } | 
 |  | 
 |   if (!IsReadOnly(pWidget) && IsFillingAllowed(pWidget)) | 
 |     pWidget->DrawShadow(pDevice, pPageView); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnDelete(CPDFSDK_Widget* pWidget) { | 
 |   UnregisterFormField(pWidget); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnMouseEnter( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!m_bNotifying) { | 
 |     if (pWidget->GetAAction(CPDF_AAction::kCursorEnter).GetDict()) { | 
 |       uint32_t nValueAge = pWidget->GetValueAge(); | 
 |       pWidget->ClearAppModified(); | 
 |       DCHECK(pPageView); | 
 |       { | 
 |         AutoRestorer<bool> restorer(&m_bNotifying); | 
 |         m_bNotifying = true; | 
 |  | 
 |         CFFL_FieldAction fa; | 
 |         fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |         fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |         pWidget->OnAAction(CPDF_AAction::kCursorEnter, &fa, pPageView); | 
 |       } | 
 |       if (!pWidget) | 
 |         return; | 
 |  | 
 |       if (pWidget->IsAppModified()) { | 
 |         CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |         if (pFormField) | 
 |           pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), | 
 |                                                 nValueAge); | 
 |       } | 
 |     } | 
 |   } | 
 |   if (CFFL_FormField* pFormField = GetOrCreateFormField(pWidget.Get())) | 
 |     pFormField->OnMouseEnter(pPageView); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnMouseExit( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!m_bNotifying) { | 
 |     if (pWidget->GetAAction(CPDF_AAction::kCursorExit).GetDict()) { | 
 |       uint32_t nValueAge = pWidget->GetValueAge(); | 
 |       pWidget->ClearAppModified(); | 
 |       DCHECK(pPageView); | 
 |       { | 
 |         AutoRestorer<bool> restorer(&m_bNotifying); | 
 |         m_bNotifying = true; | 
 |  | 
 |         CFFL_FieldAction fa; | 
 |         fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |         fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |         pWidget->OnAAction(CPDF_AAction::kCursorExit, &fa, pPageView); | 
 |       } | 
 |       if (!pWidget) | 
 |         return; | 
 |  | 
 |       if (pWidget->IsAppModified()) { | 
 |         CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |         if (pFormField) { | 
 |           pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), | 
 |                                                 nValueAge); | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |   if (CFFL_FormField* pFormField = GetFormField(pWidget.Get())) | 
 |     pFormField->OnMouseExit(pPageView); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnLButtonDown( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   if (!m_bNotifying) { | 
 |     if (Annot_HitTest(pWidget.Get(), point) && | 
 |         pWidget->GetAAction(CPDF_AAction::kButtonDown).GetDict()) { | 
 |       uint32_t nValueAge = pWidget->GetValueAge(); | 
 |       pWidget->ClearAppModified(); | 
 |       DCHECK(pPageView); | 
 |       { | 
 |         AutoRestorer<bool> restorer(&m_bNotifying); | 
 |         m_bNotifying = true; | 
 |  | 
 |         CFFL_FieldAction fa; | 
 |         fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlags); | 
 |         fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlags); | 
 |         pWidget->OnAAction(CPDF_AAction::kButtonDown, &fa, pPageView); | 
 |       } | 
 |       if (!pWidget) | 
 |         return true; | 
 |  | 
 |       if (!IsValidAnnot(pPageView, pWidget.Get())) | 
 |         return true; | 
 |  | 
 |       if (pWidget->IsAppModified()) { | 
 |         CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |         if (pFormField) { | 
 |           pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), | 
 |                                                 nValueAge); | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && | 
 |          pFormField->OnLButtonDown(pPageView, pWidget.Get(), nFlags, point); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnLButtonUp( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   bool bSetFocus; | 
 |   switch (pWidget->GetFieldType()) { | 
 |     case FormFieldType::kPushButton: | 
 |     case FormFieldType::kCheckBox: | 
 |     case FormFieldType::kRadioButton: { | 
 |       FX_RECT bbox = GetViewBBox(pPageView, pWidget.Get()); | 
 |       bSetFocus = | 
 |           bbox.Contains(static_cast<int>(point.x), static_cast<int>(point.y)); | 
 |       break; | 
 |     } | 
 |     default: | 
 |       bSetFocus = true; | 
 |       break; | 
 |   } | 
 |   if (bSetFocus) { | 
 |     ObservedPtr<CPDFSDK_Annot> pObserved(pWidget.Get()); | 
 |     m_pCallbackIface->SetFocusAnnot(pObserved); | 
 |   } | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   bool bRet = pFormField && | 
 |               pFormField->OnLButtonUp(pPageView, pWidget.Get(), nFlags, point); | 
 |   if (m_pCallbackIface->GetFocusAnnot() != pWidget.Get()) | 
 |     return bRet; | 
 |   if (OnButtonUp(pWidget, pPageView, nFlags) || !pWidget) | 
 |     return true; | 
 | #ifdef PDF_ENABLE_XFA | 
 |   if (OnClick(pWidget, pPageView, nFlags) || !pWidget) | 
 |     return true; | 
 | #endif  // PDF_ENABLE_XFA | 
 |   return bRet; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnButtonUp( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return false; | 
 |  | 
 |   if (!pWidget->GetAAction(CPDF_AAction::kButtonUp).GetDict()) | 
 |     return false; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |   DCHECK(pPageView); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |     pWidget->OnAAction(CPDF_AAction::kButtonUp, &fa, pPageView); | 
 |   } | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) | 
 |     return true; | 
 |   if (nAge == pWidget->GetAppearanceAge()) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (pFormField) | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::SetIndexSelected( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     int index, | 
 |     bool selected) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->SetIndexSelected(index, selected); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::IsIndexSelected( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     int index) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->IsIndexSelected(index); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnLButtonDblClk( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->OnLButtonDblClk(pPageView, nFlags, point); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnMouseMove( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   CFFL_FormField* pFormField = GetOrCreateFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->OnMouseMove(pPageView, nFlags, point); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnMouseWheel( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point, | 
 |     const CFX_Vector& delta) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && | 
 |          pFormField->OnMouseWheel(pPageView, nFlags, point, delta); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnRButtonDown( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->OnRButtonDown(pPageView, nFlags, point); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnRButtonUp( | 
 |     CPDFSDK_PageView* pPageView, | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlags, | 
 |     const CFX_PointF& point) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   return pFormField && pFormField->OnRButtonUp(pPageView, nFlags, point); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnKeyDown(CPDFSDK_Widget* pWidget, | 
 |                                            FWL_VKEYCODE nKeyCode, | 
 |                                            Mask<FWL_EVENTFLAG> nFlags) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->OnKeyDown(nKeyCode, nFlags); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnChar(CPDFSDK_Widget* pWidget, | 
 |                                         uint32_t nChar, | 
 |                                         Mask<FWL_EVENTFLAG> nFlags) { | 
 |   if (nChar == pdfium::ascii::kTab) | 
 |     return true; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->OnChar(pWidget, nChar, nFlags); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnSetFocus( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!pWidget) | 
 |     return false; | 
 |  | 
 |   if (!m_bNotifying) { | 
 |     if (pWidget->GetAAction(CPDF_AAction::kGetFocus).GetDict()) { | 
 |       uint32_t nValueAge = pWidget->GetValueAge(); | 
 |       pWidget->ClearAppModified(); | 
 |  | 
 |       CFFL_FormField* pFormField = GetOrCreateFormField(pWidget.Get()); | 
 |       if (!pFormField) | 
 |         return false; | 
 |  | 
 |       CPDFSDK_PageView* pPageView = pWidget->GetPageView(); | 
 |       DCHECK(pPageView); | 
 |       { | 
 |         AutoRestorer<bool> restorer(&m_bNotifying); | 
 |         m_bNotifying = true; | 
 |  | 
 |         CFFL_FieldAction fa; | 
 |         fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |         fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |         pFormField->GetActionData(pPageView, CPDF_AAction::kGetFocus, fa); | 
 |         pWidget->OnAAction(CPDF_AAction::kGetFocus, &fa, pPageView); | 
 |       } | 
 |       if (!pWidget) | 
 |         return false; | 
 |  | 
 |       if (pWidget->IsAppModified()) { | 
 |         CFFL_FormField* pFiller = GetFormField(pWidget.Get()); | 
 |         if (pFiller) { | 
 |           pFiller->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), | 
 |                                              nValueAge); | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   if (CFFL_FormField* pFormField = GetOrCreateFormField(pWidget.Get())) | 
 |     pFormField->SetFocusForAnnot(pWidget.Get(), nFlag); | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnKillFocus( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!pWidget) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (!pFormField) | 
 |     return true; | 
 |  | 
 |   pFormField->KillFocusForAnnot(nFlag); | 
 |   if (!pWidget) | 
 |     return false; | 
 |  | 
 |   if (m_bNotifying) | 
 |     return true; | 
 |  | 
 |   if (!pWidget->GetAAction(CPDF_AAction::kLoseFocus).GetDict()) | 
 |     return true; | 
 |  | 
 |   pWidget->ClearAppModified(); | 
 |  | 
 |   CPDFSDK_PageView* pPageView = pWidget->GetPageView(); | 
 |   DCHECK(pPageView); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |     pFormField->GetActionData(pPageView, CPDF_AAction::kLoseFocus, fa); | 
 |     pWidget->OnAAction(CPDF_AAction::kLoseFocus, &fa, pPageView); | 
 |   } | 
 |   return !!pWidget; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::IsVisible(CPDFSDK_Widget* pWidget) { | 
 |   return pWidget->IsVisible(); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::IsReadOnly(CPDFSDK_Widget* pWidget) { | 
 |   int nFieldFlags = pWidget->GetFieldFlags(); | 
 |   return !!(nFieldFlags & pdfium::form_flags::kReadOnly); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::IsFillingAllowed( | 
 |     CPDFSDK_Widget* pWidget) const { | 
 |   if (pWidget->GetFieldType() == FormFieldType::kPushButton) | 
 |     return false; | 
 |  | 
 |   return m_pCallbackIface->HasPermissions( | 
 |       pdfium::access_permissions::kFillForm | | 
 |       pdfium::access_permissions::kModifyAnnotation | | 
 |       pdfium::access_permissions::kModifyContent); | 
 | } | 
 |  | 
 | CFFL_FormField* CFFL_InteractiveFormFiller::GetFormField( | 
 |     CPDFSDK_Widget* pWidget) { | 
 |   auto it = m_Map.find(pWidget); | 
 |   return it != m_Map.end() ? it->second.get() : nullptr; | 
 | } | 
 |  | 
 | CFFL_FormField* CFFL_InteractiveFormFiller::GetOrCreateFormField( | 
 |     CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* result = GetFormField(pWidget); | 
 |   if (result) | 
 |     return result; | 
 |  | 
 |   std::unique_ptr<CFFL_FormField> pFormField; | 
 |   switch (pWidget->GetFieldType()) { | 
 |     case FormFieldType::kPushButton: | 
 |       pFormField = std::make_unique<CFFL_PushButton>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kCheckBox: | 
 |       pFormField = std::make_unique<CFFL_CheckBox>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kRadioButton: | 
 |       pFormField = std::make_unique<CFFL_RadioButton>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kTextField: | 
 |       pFormField = std::make_unique<CFFL_TextField>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kListBox: | 
 |       pFormField = std::make_unique<CFFL_ListBox>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kComboBox: | 
 |       pFormField = std::make_unique<CFFL_ComboBox>(this, pWidget); | 
 |       break; | 
 |     case FormFieldType::kUnknown: | 
 |     default: | 
 |       return nullptr; | 
 |   } | 
 |  | 
 |   result = pFormField.get(); | 
 |   m_Map[pWidget] = std::move(pFormField); | 
 |   return result; | 
 | } | 
 |  | 
 | WideString CFFL_InteractiveFormFiller::GetText(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField ? pFormField->GetText() : WideString(); | 
 | } | 
 |  | 
 | WideString CFFL_InteractiveFormFiller::GetSelectedText( | 
 |     CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField ? pFormField->GetSelectedText() : WideString(); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::ReplaceSelection(CPDFSDK_Widget* pWidget, | 
 |                                                   const WideString& text) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   if (!pFormField) | 
 |     return; | 
 |  | 
 |   pFormField->ReplaceSelection(text); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::SelectAllText(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->SelectAllText(); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::CanUndo(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->CanUndo(); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::CanRedo(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->CanRedo(); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::Undo(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->Undo(); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::Redo(CPDFSDK_Widget* pWidget) { | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget); | 
 |   return pFormField && pFormField->Redo(); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::UnregisterFormField(CPDFSDK_Widget* pWidget) { | 
 |   auto it = m_Map.find(pWidget); | 
 |   if (it == m_Map.end()) | 
 |     return; | 
 |  | 
 |   m_Map.erase(it); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::InvalidateRect(PerWindowData* pWidgetData, | 
 |                                                 const CFX_FloatRect& rect) { | 
 |   auto* pPrivateData = static_cast<CFFL_PerWindowData*>(pWidgetData); | 
 |   CPDFSDK_Widget* pWidget = pPrivateData->GetWidget(); | 
 |   if (!pWidget) | 
 |     return; | 
 |  | 
 |   GetCallbackIface()->InvalidateRect(pWidget, rect); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OutputSelectedRect(PerWindowData* pWidgetData, | 
 |                                                     const CFX_FloatRect& rect) { | 
 |   auto* pPrivateData = static_cast<CFFL_PerWindowData*>(pWidgetData); | 
 |   if (!pPrivateData) | 
 |     return; | 
 |  | 
 |   CFFL_FormField* pFormField = pPrivateData->GetFormField(); | 
 |   if (!pFormField) | 
 |     return; | 
 |  | 
 |   GetCallbackIface()->OutputSelectedRect(pFormField, rect); | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::IsSelectionImplemented() const { | 
 |   return GetCallbackIface()->IsSelectionImplemented(); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::SetCursor(CursorStyle nCursorStyle) { | 
 |   GetCallbackIface()->SetCursor(nCursorStyle); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::QueryWherePopup( | 
 |     const IPWL_FillerNotify::PerWindowData* pAttached, | 
 |     float fPopupMin, | 
 |     float fPopupMax, | 
 |     bool* bBottom, | 
 |     float* fPopupRet) { | 
 |   auto* pData = static_cast<const CFFL_PerWindowData*>(pAttached); | 
 |   CPDFSDK_Widget* pWidget = pData->GetWidget(); | 
 |   CPDF_Page* pPage = pWidget->GetPDFPage(); | 
 |  | 
 |   CFX_FloatRect rcPageView(0, pPage->GetPageHeight(), pPage->GetPageWidth(), 0); | 
 |   rcPageView.Normalize(); | 
 |  | 
 |   CFX_FloatRect rcAnnot = pWidget->GetRect(); | 
 |   float fTop = 0.0f; | 
 |   float fBottom = 0.0f; | 
 |   switch (pWidget->GetRotate() / 90) { | 
 |     default: | 
 |     case 0: | 
 |       fTop = rcPageView.top - rcAnnot.top; | 
 |       fBottom = rcAnnot.bottom - rcPageView.bottom; | 
 |       break; | 
 |     case 1: | 
 |       fTop = rcAnnot.left - rcPageView.left; | 
 |       fBottom = rcPageView.right - rcAnnot.right; | 
 |       break; | 
 |     case 2: | 
 |       fTop = rcAnnot.bottom - rcPageView.bottom; | 
 |       fBottom = rcPageView.top - rcAnnot.top; | 
 |       break; | 
 |     case 3: | 
 |       fTop = rcPageView.right - rcAnnot.right; | 
 |       fBottom = rcAnnot.left - rcPageView.left; | 
 |       break; | 
 |   } | 
 |  | 
 |   constexpr float kMaxListBoxHeight = 140; | 
 |   const float fMaxListBoxHeight = | 
 |       pdfium::clamp(kMaxListBoxHeight, fPopupMin, fPopupMax); | 
 |  | 
 |   if (fBottom > fMaxListBoxHeight) { | 
 |     *fPopupRet = fMaxListBoxHeight; | 
 |     *bBottom = true; | 
 |     return; | 
 |   } | 
 |  | 
 |   if (fTop > fMaxListBoxHeight) { | 
 |     *fPopupRet = fMaxListBoxHeight; | 
 |     *bBottom = false; | 
 |     return; | 
 |   } | 
 |  | 
 |   if (fTop > fBottom) { | 
 |     *fPopupRet = fTop; | 
 |     *bBottom = false; | 
 |   } else { | 
 |     *fPopupRet = fBottom; | 
 |     *bBottom = true; | 
 |   } | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnKeyStrokeCommit( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return true; | 
 |  | 
 |   if (!pWidget->GetAAction(CPDF_AAction::kKeyStroke).GetDict()) | 
 |     return true; | 
 |  | 
 |   DCHECK(pPageView); | 
 |   pWidget->ClearAppModified(); | 
 |  | 
 |   AutoRestorer<bool> restorer(&m_bNotifying); | 
 |   m_bNotifying = true; | 
 |  | 
 |   CFFL_FieldAction fa; | 
 |   fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |   fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |   fa.bWillCommit = true; | 
 |   fa.bKeyDown = true; | 
 |   fa.bRC = true; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   pFormField->GetActionData(pPageView, CPDF_AAction::kKeyStroke, fa); | 
 |   pFormField->SavePWLWindowState(pPageView); | 
 |   pWidget->OnAAction(CPDF_AAction::kKeyStroke, &fa, pPageView); | 
 |  | 
 |   if (!pWidget) | 
 |     return true; | 
 |  | 
 |   return fa.bRC; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnValidate( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return true; | 
 |  | 
 |   if (!pWidget->GetAAction(CPDF_AAction::kValidate).GetDict()) | 
 |     return true; | 
 |  | 
 |   DCHECK(pPageView); | 
 |   pWidget->ClearAppModified(); | 
 |  | 
 |   AutoRestorer<bool> restorer(&m_bNotifying); | 
 |   m_bNotifying = true; | 
 |  | 
 |   CFFL_FieldAction fa; | 
 |   fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |   fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |   fa.bKeyDown = true; | 
 |   fa.bRC = true; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   pFormField->GetActionData(pPageView, CPDF_AAction::kValidate, fa); | 
 |   pFormField->SavePWLWindowState(pPageView); | 
 |   pWidget->OnAAction(CPDF_AAction::kValidate, &fa, pPageView); | 
 |  | 
 |   if (!pWidget) | 
 |     return true; | 
 |  | 
 |   return fa.bRC; | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnCalculate( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget) { | 
 |   if (m_bNotifying) | 
 |     return; | 
 |  | 
 |   ObservedPtr<CPDFSDK_Annot> pObserved(pWidget.Get()); | 
 |   m_pCallbackIface->OnCalculate(pObserved); | 
 | } | 
 |  | 
 | void CFFL_InteractiveFormFiller::OnFormat( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget) { | 
 |   if (m_bNotifying) | 
 |     return; | 
 |  | 
 |   ObservedPtr<CPDFSDK_Annot> pObserved(pWidget.Get()); | 
 |   m_pCallbackIface->OnFormat(pObserved); | 
 | } | 
 |  | 
 | #ifdef PDF_ENABLE_XFA | 
 | bool CFFL_InteractiveFormFiller::OnClick(ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |                                          const CPDFSDK_PageView* pPageView, | 
 |                                          Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return false; | 
 |  | 
 |   if (!pWidget->HasXFAAAction(PDFSDK_XFA_Click)) | 
 |     return false; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |  | 
 |     pWidget->OnXFAAAction(PDFSDK_XFA_Click, &fa, pPageView); | 
 |   } | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) | 
 |     return true; | 
 |   if (nAge == pWidget->GetAppearanceAge()) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (pFormField) | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |   return false; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnFull(ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |                                         const CPDFSDK_PageView* pPageView, | 
 |                                         Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return false; | 
 |  | 
 |   if (!pWidget->HasXFAAAction(PDFSDK_XFA_Full)) | 
 |     return false; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |     pWidget->OnXFAAAction(PDFSDK_XFA_Full, &fa, pPageView); | 
 |   } | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) | 
 |     return true; | 
 |   if (nAge == pWidget->GetAppearanceAge()) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (pFormField) | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnPreOpen(ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |                                            const CPDFSDK_PageView* pPageView, | 
 |                                            Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return false; | 
 |  | 
 |   if (!pWidget->HasXFAAAction(PDFSDK_XFA_PreOpen)) | 
 |     return false; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |     pWidget->OnXFAAAction(PDFSDK_XFA_PreOpen, &fa, pPageView); | 
 |   } | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) | 
 |     return true; | 
 |   if (nAge == pWidget->GetAppearanceAge()) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (pFormField) | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnPostOpen( | 
 |     ObservedPtr<CPDFSDK_Widget>& pWidget, | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (m_bNotifying) | 
 |     return false; | 
 |  | 
 |   if (!pWidget->HasXFAAAction(PDFSDK_XFA_PostOpen)) | 
 |     return false; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |   { | 
 |     AutoRestorer<bool> restorer(&m_bNotifying); | 
 |     m_bNotifying = true; | 
 |  | 
 |     CFFL_FieldAction fa; | 
 |     fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |     fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |     pWidget->OnXFAAAction(PDFSDK_XFA_PostOpen, &fa, pPageView); | 
 |   } | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) | 
 |     return true; | 
 |  | 
 |   if (nAge == pWidget->GetAppearanceAge()) | 
 |     return false; | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |   if (pFormField) | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |   return true; | 
 | } | 
 | #endif  // PDF_ENABLE_XFA | 
 |  | 
 | // static | 
 | bool CFFL_InteractiveFormFiller::IsValidAnnot(const CPDFSDK_PageView* pPageView, | 
 |                                               CPDFSDK_Widget* pWidget) { | 
 |   return pPageView && pPageView->IsValidAnnot(pWidget->GetPDFAnnot()); | 
 | } | 
 |  | 
 | std::pair<bool, bool> CFFL_InteractiveFormFiller::OnBeforeKeyStroke( | 
 |     const IPWL_FillerNotify::PerWindowData* pAttached, | 
 |     WideString& strChange, | 
 |     const WideString& strChangeEx, | 
 |     int nSelStart, | 
 |     int nSelEnd, | 
 |     bool bKeyDown, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 |   // Copy out of private data since the window owning it may not survive. | 
 |   auto* pPrivateData = static_cast<const CFFL_PerWindowData*>(pAttached); | 
 |   const CPDFSDK_PageView* pPageView = pPrivateData->GetPageView(); | 
 |   ObservedPtr<CPDFSDK_Widget> pWidget(pPrivateData->GetWidget()); | 
 |   DCHECK(pWidget); | 
 |  | 
 |   CFFL_FormField* pFormField = GetFormField(pWidget.Get()); | 
 |  | 
 | #ifdef PDF_ENABLE_XFA | 
 |   if (pFormField->IsFieldFull(pPageView)) { | 
 |     if (OnFull(pWidget, pPageView, nFlag) || !pWidget) | 
 |       return {true, true}; | 
 |   } | 
 | #endif  // PDF_ENABLE_XFA | 
 |  | 
 |   if (m_bNotifying || | 
 |       !pWidget->GetAAction(CPDF_AAction::kKeyStroke).GetDict()) { | 
 |     return {true, false}; | 
 |   } | 
 |  | 
 |   AutoRestorer<bool> restorer(&m_bNotifying); | 
 |   m_bNotifying = true; | 
 |  | 
 |   uint32_t nAge = pWidget->GetAppearanceAge(); | 
 |   uint32_t nValueAge = pWidget->GetValueAge(); | 
 |  | 
 |   CFFL_FieldAction fa; | 
 |   fa.bModifier = CPWL_Wnd::IsCTRLKeyDown(nFlag); | 
 |   fa.bShift = CPWL_Wnd::IsSHIFTKeyDown(nFlag); | 
 |   fa.sChange = strChange; | 
 |   fa.sChangeEx = strChangeEx; | 
 |   fa.bKeyDown = bKeyDown; | 
 |   fa.bWillCommit = false; | 
 |   fa.bRC = true; | 
 |   fa.nSelStart = nSelStart; | 
 |   fa.nSelEnd = nSelEnd; | 
 |   pFormField->GetActionData(pPageView, CPDF_AAction::kKeyStroke, fa); | 
 |   pFormField->SavePWLWindowState(pPageView); | 
 |  | 
 |   bool action_status = | 
 |       pWidget->OnAAction(CPDF_AAction::kKeyStroke, &fa, pPageView); | 
 |  | 
 |   if (!pWidget || !IsValidAnnot(pPageView, pWidget.Get())) { | 
 |     return {true, true}; | 
 |   } | 
 |   if (!action_status) | 
 |     return {true, false}; | 
 |  | 
 |   bool bExit = false; | 
 |   if (nAge != pWidget->GetAppearanceAge()) { | 
 |     pFormField->ResetPWLWindowForValueAge(pPageView, pWidget.Get(), nValueAge); | 
 |     pPrivateData = pFormField->GetPerPWLWindowData(pPageView); | 
 |     if (!pPrivateData) | 
 |       return {true, true}; | 
 |  | 
 |     pWidget.Reset(pPrivateData->GetWidget()); | 
 |     pPageView = pPrivateData->GetPageView(); | 
 |     bExit = true; | 
 |   } | 
 |   if (fa.bRC) { | 
 |     pFormField->SetActionData(pPageView, CPDF_AAction::kKeyStroke, fa); | 
 |   } else { | 
 |     pFormField->RecreatePWLWindowFromSavedState(pPageView); | 
 |   } | 
 |   if (m_pCallbackIface->GetFocusAnnot() == pWidget) | 
 |     return {false, bExit}; | 
 |  | 
 |   pFormField->CommitData(pPageView, nFlag); | 
 |   return {false, true}; | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnPopupPreOpen( | 
 |     const IPWL_FillerNotify::PerWindowData* pAttached, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 | #ifdef PDF_ENABLE_XFA | 
 |   auto* pData = static_cast<const CFFL_PerWindowData*>(pAttached); | 
 |   DCHECK(pData->GetWidget()); | 
 |  | 
 |   ObservedPtr<CPDFSDK_Widget> pObserved(pData->GetWidget()); | 
 |   return OnPreOpen(pObserved, pData->GetPageView(), nFlag) || !pObserved; | 
 | #else | 
 |   return false; | 
 | #endif | 
 | } | 
 |  | 
 | bool CFFL_InteractiveFormFiller::OnPopupPostOpen( | 
 |     const IPWL_FillerNotify::PerWindowData* pAttached, | 
 |     Mask<FWL_EVENTFLAG> nFlag) { | 
 | #ifdef PDF_ENABLE_XFA | 
 |   auto* pData = static_cast<const CFFL_PerWindowData*>(pAttached); | 
 |   DCHECK(pData->GetWidget()); | 
 |  | 
 |   ObservedPtr<CPDFSDK_Widget> pObserved(pData->GetWidget()); | 
 |   return OnPostOpen(pObserved, pData->GetPageView(), nFlag) || !pObserved; | 
 | #else | 
 |   return false; | 
 | #endif | 
 | } |