1004 lines
30 KiB
C++
1004 lines
30 KiB
C++
|
|
// Copyright 2014 The PDFium Authors
|
||
|
|
// 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).HasDict()) {
|
||
|
|
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).HasDict()) {
|
||
|
|
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).HasDict()) {
|
||
|
|
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).HasDict())
|
||
|
|
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).HasDict()) {
|
||
|
|
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).HasDict())
|
||
|
|
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;
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFFL_InteractiveFormFiller::OnSetFieldInputFocus(const WideString& text) {
|
||
|
|
m_pCallbackIface->OnSetFieldInputFocus(text);
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFFL_InteractiveFormFiller::Invalidate(IPDF_Page* pPage,
|
||
|
|
const FX_RECT& rect) {
|
||
|
|
m_pCallbackIface->Invalidate(pPage, rect);
|
||
|
|
}
|
||
|
|
|
||
|
|
CPDFSDK_PageView* CFFL_InteractiveFormFiller::GetOrCreatePageView(
|
||
|
|
IPDF_Page* pPage) {
|
||
|
|
return m_pCallbackIface->GetOrCreatePageView(pPage);
|
||
|
|
}
|
||
|
|
|
||
|
|
CPDFSDK_PageView* CFFL_InteractiveFormFiller::GetPageView(IPDF_Page* pPage) {
|
||
|
|
return m_pCallbackIface->GetPageView(pPage);
|
||
|
|
}
|
||
|
|
|
||
|
|
CFX_Timer::HandlerIface* CFFL_InteractiveFormFiller::GetTimerHandler() {
|
||
|
|
return m_pCallbackIface->GetTimerHandler();
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFFL_InteractiveFormFiller::OnChange() {
|
||
|
|
m_pCallbackIface->OnChange();
|
||
|
|
}
|
||
|
|
|
||
|
|
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::ReplaceAndKeepSelection(
|
||
|
|
CPDFSDK_Widget* pWidget,
|
||
|
|
const WideString& text) {
|
||
|
|
CFFL_FormField* pFormField = GetFormField(pWidget);
|
||
|
|
if (!pFormField)
|
||
|
|
return;
|
||
|
|
|
||
|
|
pFormField->ReplaceAndKeepSelection(text);
|
||
|
|
}
|
||
|
|
|
||
|
|
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;
|
||
|
|
|
||
|
|
m_pCallbackIface->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;
|
||
|
|
|
||
|
|
m_pCallbackIface->OutputSelectedRect(pFormField, rect);
|
||
|
|
}
|
||
|
|
|
||
|
|
bool CFFL_InteractiveFormFiller::IsSelectionImplemented() const {
|
||
|
|
return m_pCallbackIface->IsSelectionImplemented();
|
||
|
|
}
|
||
|
|
|
||
|
|
void CFFL_InteractiveFormFiller::SetCursor(CursorStyle nCursorStyle) {
|
||
|
|
m_pCallbackIface->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).HasDict())
|
||
|
|
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).HasDict())
|
||
|
|
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).HasDict()) {
|
||
|
|
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
|
||
|
|
}
|