| // 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_formfield.h" | 
 |  | 
 | #include <utility> | 
 |  | 
 | #include "constants/form_flags.h" | 
 | #include "core/fpdfapi/page/cpdf_page.h" | 
 | #include "core/fxge/cfx_renderdevice.h" | 
 | #include "fpdfsdk/cpdfsdk_pageview.h" | 
 | #include "fpdfsdk/cpdfsdk_widget.h" | 
 | #include "fpdfsdk/formfiller/cffl_perwindowdata.h" | 
 | #include "third_party/base/check.h" | 
 |  | 
 | CFFL_FormField::CFFL_FormField(CFFL_InteractiveFormFiller* pFormFiller, | 
 |                                CPDFSDK_Widget* pWidget) | 
 |     : m_pFormFiller(pFormFiller), m_pWidget(pWidget) { | 
 |   DCHECK(m_pFormFiller); | 
 | } | 
 |  | 
 | CFFL_FormField::~CFFL_FormField() { | 
 |   DestroyWindows(); | 
 | } | 
 |  | 
 | void CFFL_FormField::DestroyWindows() { | 
 |   while (!m_Maps.empty()) { | 
 |     auto it = m_Maps.begin(); | 
 |     std::unique_ptr<CPWL_Wnd> pWnd = std::move(it->second); | 
 |     m_Maps.erase(it); | 
 |     pWnd->InvalidateProvider(this); | 
 |     pWnd->Destroy(); | 
 |   } | 
 | } | 
 |  | 
 | FX_RECT CFFL_FormField::GetViewBBox(const CPDFSDK_PageView* pPageView) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   CFX_FloatRect rcAnnot = | 
 |       pWnd ? PWLtoFFL(pWnd->GetWindowRect()) : m_pWidget->GetRect(); | 
 |   CFX_FloatRect rcFocus = GetFocusBox(pPageView); | 
 |  | 
 |   CFX_FloatRect rcWin = rcAnnot; | 
 |   if (!rcFocus.IsEmpty()) | 
 |     rcWin.Union(rcFocus); | 
 |   if (!rcWin.IsEmpty()) { | 
 |     rcWin.Inflate(1, 1); | 
 |     rcWin.Normalize(); | 
 |   } | 
 |  | 
 |   return rcWin.GetOuterRect(); | 
 | } | 
 |  | 
 | void CFFL_FormField::OnDraw(CPDFSDK_PageView* pPageView, | 
 |                             CPDFSDK_Widget* pWidget, | 
 |                             CFX_RenderDevice* pDevice, | 
 |                             const CFX_Matrix& mtUser2Device) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (pWnd) { | 
 |     pWnd->DrawAppearance(pDevice, GetCurMatrix() * mtUser2Device); | 
 |     return; | 
 |   } | 
 |   if (!CFFL_InteractiveFormFiller::IsVisible(pWidget)) | 
 |     return; | 
 |  | 
 |   pWidget->DrawAppearance(pDevice, mtUser2Device, | 
 |                           CPDF_Annot::AppearanceMode::kNormal); | 
 | } | 
 |  | 
 | void CFFL_FormField::OnDrawDeactive(CPDFSDK_PageView* pPageView, | 
 |                                     CPDFSDK_Widget* pWidget, | 
 |                                     CFX_RenderDevice* pDevice, | 
 |                                     const CFX_Matrix& mtUser2Device) { | 
 |   pWidget->DrawAppearance(pDevice, mtUser2Device, | 
 |                           CPDF_Annot::AppearanceMode::kNormal); | 
 | } | 
 |  | 
 | void CFFL_FormField::OnMouseEnter(CPDFSDK_PageView* pPageView) {} | 
 |  | 
 | void CFFL_FormField::OnMouseExit(CPDFSDK_PageView* pPageView) { | 
 |   m_pTimer.reset(); | 
 |   DCHECK(m_pWidget); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnLButtonDown(CPDFSDK_PageView* pPageView, | 
 |                                    CPDFSDK_Widget* pWidget, | 
 |                                    Mask<FWL_EVENTFLAG> nFlags, | 
 |                                    const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return false; | 
 |  | 
 |   m_bValid = true; | 
 |   FX_RECT rect = GetViewBBox(pPageView); | 
 |   InvalidateRect(rect); | 
 |   if (!rect.Contains(static_cast<int>(point.x), static_cast<int>(point.y))) | 
 |     return false; | 
 |   return pWnd->OnLButtonDown(nFlags, FFLtoPWL(point)); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnLButtonUp(CPDFSDK_PageView* pPageView, | 
 |                                  CPDFSDK_Widget* pWidget, | 
 |                                  Mask<FWL_EVENTFLAG> nFlags, | 
 |                                  const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return false; | 
 |  | 
 |   InvalidateRect(GetViewBBox(pPageView)); | 
 |   pWnd->OnLButtonUp(nFlags, FFLtoPWL(point)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnLButtonDblClk(CPDFSDK_PageView* pPageView, | 
 |                                      Mask<FWL_EVENTFLAG> nFlags, | 
 |                                      const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return false; | 
 |  | 
 |   pWnd->OnLButtonDblClk(nFlags, FFLtoPWL(point)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnMouseMove(CPDFSDK_PageView* pPageView, | 
 |                                  Mask<FWL_EVENTFLAG> nFlags, | 
 |                                  const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return false; | 
 |  | 
 |   pWnd->OnMouseMove(nFlags, FFLtoPWL(point)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnMouseWheel(CPDFSDK_PageView* pPageView, | 
 |                                   Mask<FWL_EVENTFLAG> nFlags, | 
 |                                   const CFX_PointF& point, | 
 |                                   const CFX_Vector& delta) { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView); | 
 |   return pWnd && pWnd->OnMouseWheel(nFlags, FFLtoPWL(point), delta); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnRButtonDown(CPDFSDK_PageView* pPageView, | 
 |                                    Mask<FWL_EVENTFLAG> nFlags, | 
 |                                    const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView); | 
 |   return pWnd && pWnd->OnRButtonDown(nFlags, FFLtoPWL(point)); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnRButtonUp(CPDFSDK_PageView* pPageView, | 
 |                                  Mask<FWL_EVENTFLAG> nFlags, | 
 |                                  const CFX_PointF& point) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   return pWnd && pWnd->OnRButtonUp(nFlags, FFLtoPWL(point)); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnKeyDown(FWL_VKEYCODE nKeyCode, | 
 |                                Mask<FWL_EVENTFLAG> nFlags) { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->OnKeyDown(nKeyCode, nFlags); | 
 | } | 
 |  | 
 | bool CFFL_FormField::OnChar(CPDFSDK_Widget* pWidget, | 
 |                             uint32_t nChar, | 
 |                             Mask<FWL_EVENTFLAG> nFlags) { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->OnChar(nChar, nFlags); | 
 | } | 
 |  | 
 | bool CFFL_FormField::SetIndexSelected(int index, bool selected) { | 
 |   return false; | 
 | } | 
 |  | 
 | bool CFFL_FormField::IsIndexSelected(int index) { | 
 |   return false; | 
 | } | 
 |  | 
 | WideString CFFL_FormField::GetText() { | 
 |   if (!IsValid()) | 
 |     return WideString(); | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd ? pWnd->GetText() : WideString(); | 
 | } | 
 |  | 
 | WideString CFFL_FormField::GetSelectedText() { | 
 |   if (!IsValid()) | 
 |     return WideString(); | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd ? pWnd->GetSelectedText() : WideString(); | 
 | } | 
 |  | 
 | void CFFL_FormField::ReplaceSelection(const WideString& text) { | 
 |   if (!IsValid()) | 
 |     return; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   if (!pWnd) | 
 |     return; | 
 |  | 
 |   pWnd->ReplaceSelection(text); | 
 | } | 
 |  | 
 | bool CFFL_FormField::SelectAllText() { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->SelectAllText(); | 
 | } | 
 |  | 
 | bool CFFL_FormField::CanUndo() { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->CanUndo(); | 
 | } | 
 |  | 
 | bool CFFL_FormField::CanRedo() { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->CanRedo(); | 
 | } | 
 |  | 
 | bool CFFL_FormField::Undo() { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->Undo(); | 
 | } | 
 |  | 
 | bool CFFL_FormField::Redo() { | 
 |   if (!IsValid()) | 
 |     return false; | 
 |  | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(GetCurPageView()); | 
 |   return pWnd && pWnd->Redo(); | 
 | } | 
 |  | 
 | void CFFL_FormField::SetFocusForAnnot(CPDFSDK_Widget* pWidget, | 
 |                                       Mask<FWL_EVENTFLAG> nFlag) { | 
 |   CPDFSDK_PageView* pPageView = | 
 |       m_pFormFiller->GetCallbackIface()->GetOrCreatePageView( | 
 |           pWidget->GetPage()); | 
 |   CPWL_Wnd* pWnd = CreateOrUpdatePWLWindow(pPageView); | 
 |   if (pWnd) | 
 |     pWnd->SetFocus(); | 
 |  | 
 |   m_bValid = true; | 
 |   InvalidateRect(GetViewBBox(pPageView)); | 
 | } | 
 |  | 
 | void CFFL_FormField::KillFocusForAnnot(Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!IsValid()) | 
 |     return; | 
 |  | 
 |   CPDFSDK_PageView* pPageView = | 
 |       m_pFormFiller->GetCallbackIface()->GetPageView(m_pWidget->GetPage()); | 
 |   if (!pPageView || !CommitData(pPageView, nFlag)) | 
 |     return; | 
 |   if (CPWL_Wnd* pWnd = GetPWLWindow(pPageView)) | 
 |     pWnd->KillFocus(); | 
 |  | 
 |   bool bDestroyPWLWindow; | 
 |   switch (m_pWidget->GetFieldType()) { | 
 |     case FormFieldType::kPushButton: | 
 |     case FormFieldType::kCheckBox: | 
 |     case FormFieldType::kRadioButton: | 
 |       bDestroyPWLWindow = true; | 
 |       break; | 
 |     default: | 
 |       bDestroyPWLWindow = false; | 
 |       break; | 
 |   } | 
 |   EscapeFiller(pPageView, bDestroyPWLWindow); | 
 | } | 
 |  | 
 | bool CFFL_FormField::IsValid() const { | 
 |   return m_bValid; | 
 | } | 
 |  | 
 | CPWL_Wnd::CreateParams CFFL_FormField::GetCreateParam() { | 
 |   CPWL_Wnd::CreateParams cp( | 
 |       m_pFormFiller->GetCallbackIface()->GetTimerHandler(), m_pFormFiller, | 
 |       this); | 
 |  | 
 |   cp.rcRectWnd = GetPDFAnnotRect(); | 
 |  | 
 |   uint32_t dwCreateFlags = PWS_BORDER | PWS_BACKGROUND | PWS_VISIBLE; | 
 |   uint32_t dwFieldFlag = m_pWidget->GetFieldFlags(); | 
 |   if (dwFieldFlag & pdfium::form_flags::kReadOnly) | 
 |     dwCreateFlags |= PWS_READONLY; | 
 |  | 
 |   absl::optional<FX_COLORREF> color = m_pWidget->GetFillColor(); | 
 |   if (color.has_value()) | 
 |     cp.sBackgroundColor = CFX_Color(color.value()); | 
 |   color = m_pWidget->GetBorderColor(); | 
 |   if (color.has_value()) | 
 |     cp.sBorderColor = CFX_Color(color.value()); | 
 |  | 
 |   cp.sTextColor = CFX_Color(CFX_Color::Type::kGray, 0); | 
 |  | 
 |   color = m_pWidget->GetTextColor(); | 
 |   if (color.has_value()) | 
 |     cp.sTextColor = CFX_Color(color.value()); | 
 |  | 
 |   cp.fFontSize = m_pWidget->GetFontSize(); | 
 |   cp.dwBorderWidth = m_pWidget->GetBorderWidth(); | 
 |  | 
 |   cp.nBorderStyle = m_pWidget->GetBorderStyle(); | 
 |   switch (cp.nBorderStyle) { | 
 |     case BorderStyle::kDash: | 
 |       cp.sDash = CPWL_Dash(3, 3, 0); | 
 |       break; | 
 |     case BorderStyle::kBeveled: | 
 |     case BorderStyle::kInset: | 
 |       cp.dwBorderWidth *= 2; | 
 |       break; | 
 |     default: | 
 |       break; | 
 |   } | 
 |  | 
 |   if (cp.fFontSize <= 0) | 
 |     dwCreateFlags |= PWS_AUTOFONTSIZE; | 
 |  | 
 |   cp.dwFlags = dwCreateFlags; | 
 |   return cp; | 
 | } | 
 |  | 
 | CPWL_Wnd* CFFL_FormField::GetPWLWindow( | 
 |     const CPDFSDK_PageView* pPageView) const { | 
 |   DCHECK(pPageView); | 
 |   auto it = m_Maps.find(pPageView); | 
 |   return it != m_Maps.end() ? it->second.get() : nullptr; | 
 | } | 
 |  | 
 | CPWL_Wnd* CFFL_FormField::CreateOrUpdatePWLWindow( | 
 |     const CPDFSDK_PageView* pPageView) { | 
 |   DCHECK(pPageView); | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) { | 
 |     CPWL_Wnd::CreateParams cp = GetCreateParam(); | 
 |     // TODO(tsepez): maybe pass widget's value age as 4th arg. | 
 |     auto pPrivateData = std::make_unique<CFFL_PerWindowData>( | 
 |         m_pWidget.Get(), pPageView, m_pWidget->GetAppearanceAge(), 0); | 
 |     m_Maps[pPageView] = NewPWLWindow(cp, std::move(pPrivateData)); | 
 |     return m_Maps[pPageView].get(); | 
 |   } | 
 |   const auto* pPrivateData = | 
 |       static_cast<const CFFL_PerWindowData*>(pWnd->GetAttachedData()); | 
 |   if (pPrivateData->AppearanceAgeEquals(m_pWidget->GetAppearanceAge())) | 
 |     return pWnd; | 
 |  | 
 |   return ResetPWLWindowForValueAgeInternal(pPageView, m_pWidget, | 
 |                                            pPrivateData->GetValueAge()); | 
 | } | 
 |  | 
 | void CFFL_FormField::DestroyPWLWindow(const CPDFSDK_PageView* pPageView) { | 
 |   auto it = m_Maps.find(pPageView); | 
 |   if (it == m_Maps.end()) | 
 |     return; | 
 |  | 
 |   std::unique_ptr<CPWL_Wnd> pWnd = std::move(it->second); | 
 |   m_Maps.erase(it); | 
 |   pWnd->Destroy(); | 
 | } | 
 |  | 
 | CFX_Matrix CFFL_FormField::GetWindowMatrix( | 
 |     const IPWL_FillerNotify::PerWindowData* pAttached) { | 
 |   const auto* pPrivateData = static_cast<const CFFL_PerWindowData*>(pAttached); | 
 |   if (!pPrivateData) | 
 |     return CFX_Matrix(); | 
 |  | 
 |   const CPDFSDK_PageView* pPageView = pPrivateData->GetPageView(); | 
 |   if (!pPageView) | 
 |     return CFX_Matrix(); | 
 |  | 
 |   return GetCurMatrix() * pPageView->GetCurrentMatrix(); | 
 | } | 
 |  | 
 | void CFFL_FormField::OnSetFocusForEdit(CPWL_Edit* pEdit) { | 
 |   // Only sub-classes might have a subordinate edit to focus. | 
 | } | 
 |  | 
 | CFX_Matrix CFFL_FormField::GetCurMatrix() { | 
 |   CFX_Matrix mt; | 
 |   CFX_FloatRect rcDA = m_pWidget->GetPDFAnnot()->GetRect(); | 
 |   switch (m_pWidget->GetRotate()) { | 
 |     case 90: | 
 |       mt = CFX_Matrix(0, 1, -1, 0, rcDA.right - rcDA.left, 0); | 
 |       break; | 
 |     case 180: | 
 |       mt = CFX_Matrix(-1, 0, 0, -1, rcDA.right - rcDA.left, | 
 |                       rcDA.top - rcDA.bottom); | 
 |       break; | 
 |     case 270: | 
 |       mt = CFX_Matrix(0, -1, 1, 0, 0, rcDA.top - rcDA.bottom); | 
 |       break; | 
 |     case 0: | 
 |     default: | 
 |       break; | 
 |   } | 
 |   mt.e += rcDA.left; | 
 |   mt.f += rcDA.bottom; | 
 |  | 
 |   return mt; | 
 | } | 
 |  | 
 | CFX_FloatRect CFFL_FormField::GetPDFAnnotRect() const { | 
 |   CFX_FloatRect rectAnnot = m_pWidget->GetPDFAnnot()->GetRect(); | 
 |   float fWidth = rectAnnot.Width(); | 
 |   float fHeight = rectAnnot.Height(); | 
 |   if ((m_pWidget->GetRotate() / 90) & 0x01) | 
 |     std::swap(fWidth, fHeight); | 
 |   return CFX_FloatRect(0, 0, fWidth, fHeight); | 
 | } | 
 |  | 
 | CPDFSDK_PageView* CFFL_FormField::GetCurPageView() { | 
 |   return m_pFormFiller->GetCallbackIface()->GetOrCreatePageView( | 
 |       m_pWidget->GetPage()); | 
 | } | 
 |  | 
 | CFX_FloatRect CFFL_FormField::GetFocusBox(const CPDFSDK_PageView* pPageView) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return CFX_FloatRect(); | 
 |  | 
 |   CFX_FloatRect rcFocus = PWLtoFFL(pWnd->GetFocusRect()); | 
 |   return pPageView->GetPDFPage()->GetBBox().Contains(rcFocus) ? rcFocus | 
 |                                                               : CFX_FloatRect(); | 
 | } | 
 |  | 
 | CFX_FloatRect CFFL_FormField::FFLtoPWL(const CFX_FloatRect& rect) { | 
 |   return GetCurMatrix().GetInverse().TransformRect(rect); | 
 | } | 
 |  | 
 | CFX_FloatRect CFFL_FormField::PWLtoFFL(const CFX_FloatRect& rect) { | 
 |   return GetCurMatrix().TransformRect(rect); | 
 | } | 
 |  | 
 | CFX_PointF CFFL_FormField::FFLtoPWL(const CFX_PointF& point) { | 
 |   return GetCurMatrix().GetInverse().Transform(point); | 
 | } | 
 |  | 
 | CFX_PointF CFFL_FormField::PWLtoFFL(const CFX_PointF& point) { | 
 |   return GetCurMatrix().Transform(point); | 
 | } | 
 |  | 
 | bool CFFL_FormField::CommitData(const CPDFSDK_PageView* pPageView, | 
 |                                 Mask<FWL_EVENTFLAG> nFlag) { | 
 |   if (!IsDataChanged(pPageView)) | 
 |     return true; | 
 |  | 
 |   ObservedPtr<CPDFSDK_Widget> pObserved(m_pWidget.Get()); | 
 |   if (!m_pFormFiller->OnKeyStrokeCommit(pObserved, pPageView, nFlag)) { | 
 |     if (!pObserved) | 
 |       return false; | 
 |     ResetPWLWindow(pPageView); | 
 |     return true; | 
 |   } | 
 |   if (!pObserved) | 
 |     return false; | 
 |  | 
 |   if (!m_pFormFiller->OnValidate(pObserved, pPageView, nFlag)) { | 
 |     if (!pObserved) | 
 |       return false; | 
 |     ResetPWLWindow(pPageView); | 
 |     return true; | 
 |   } | 
 |   if (!pObserved) | 
 |     return false; | 
 |  | 
 |   SaveData(pPageView);  // may invoking JS to delete this widget. | 
 |   if (!pObserved) | 
 |     return false; | 
 |  | 
 |   m_pFormFiller->OnCalculate(pObserved); | 
 |   if (!pObserved) | 
 |     return false; | 
 |  | 
 |   m_pFormFiller->OnFormat(pObserved); | 
 |   if (!pObserved) | 
 |     return false; | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | bool CFFL_FormField::IsDataChanged(const CPDFSDK_PageView* pPageView) { | 
 |   return false; | 
 | } | 
 |  | 
 | void CFFL_FormField::SaveData(const CPDFSDK_PageView* pPageView) {} | 
 |  | 
 | #ifdef PDF_ENABLE_XFA | 
 | bool CFFL_FormField::IsFieldFull(const CPDFSDK_PageView* pPageView) { | 
 |   return false; | 
 | } | 
 | #endif  // PDF_ENABLE_XFA | 
 |  | 
 | void CFFL_FormField::SetChangeMark() { | 
 |   m_pFormFiller->GetCallbackIface()->OnChange(); | 
 | } | 
 |  | 
 | void CFFL_FormField::GetActionData(const CPDFSDK_PageView* pPageView, | 
 |                                    CPDF_AAction::AActionType type, | 
 |                                    CFFL_FieldAction& fa) { | 
 |   fa.sValue = m_pWidget->GetValue(); | 
 | } | 
 |  | 
 | void CFFL_FormField::SetActionData(const CPDFSDK_PageView* pPageView, | 
 |                                    CPDF_AAction::AActionType type, | 
 |                                    const CFFL_FieldAction& fa) {} | 
 |  | 
 | void CFFL_FormField::SavePWLWindowState(const CPDFSDK_PageView* pPageView) {} | 
 |  | 
 | void CFFL_FormField::RecreatePWLWindowFromSavedState( | 
 |     const CPDFSDK_PageView* pPageView) {} | 
 |  | 
 | CFFL_PerWindowData* CFFL_FormField::GetPerPWLWindowData( | 
 |     const CPDFSDK_PageView* pPageView) { | 
 |   CPWL_Wnd* pWnd = GetPWLWindow(pPageView); | 
 |   if (!pWnd) | 
 |     return nullptr; | 
 |  | 
 |   return static_cast<CFFL_PerWindowData*>(pWnd->GetAttachedData()); | 
 | } | 
 |  | 
 | void CFFL_FormField::ResetPWLWindowForValueAge( | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     CPDFSDK_Widget* pWidget, | 
 |     uint32_t nValueAge) { | 
 |   // Don't leak PWL_Wnd result to public callers. | 
 |   ResetPWLWindowForValueAgeInternal(pPageView, pWidget, nValueAge); | 
 | } | 
 |  | 
 | CPWL_Wnd* CFFL_FormField::ResetPWLWindowForValueAgeInternal( | 
 |     const CPDFSDK_PageView* pPageView, | 
 |     CPDFSDK_Widget* pWidget, | 
 |     uint32_t nValueAge) { | 
 |   return nValueAge == pWidget->GetValueAge() ? RestorePWLWindow(pPageView) | 
 |                                              : ResetPWLWindow(pPageView); | 
 | } | 
 |  | 
 | CPWL_Wnd* CFFL_FormField::ResetPWLWindow(const CPDFSDK_PageView* pPageView) { | 
 |   return GetPWLWindow(pPageView); | 
 | } | 
 |  | 
 | CPWL_Wnd* CFFL_FormField::RestorePWLWindow(const CPDFSDK_PageView* pPageView) { | 
 |   return GetPWLWindow(pPageView); | 
 | } | 
 |  | 
 | void CFFL_FormField::OnTimerFired() {} | 
 |  | 
 | void CFFL_FormField::EscapeFiller(CPDFSDK_PageView* pPageView, | 
 |                                   bool bDestroyPWLWindow) { | 
 |   m_bValid = false; | 
 |  | 
 |   InvalidateRect(GetViewBBox(pPageView)); | 
 |   if (bDestroyPWLWindow) | 
 |     DestroyPWLWindow(pPageView); | 
 | } | 
 |  | 
 | void CFFL_FormField::InvalidateRect(const FX_RECT& rect) { | 
 |   m_pFormFiller->GetCallbackIface()->Invalidate(m_pWidget->GetPage(), rect); | 
 | } |