| // Copyright 2016 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/cpdfsdk_widget.h" |
| |
| #include "constants/access_permissions.h" |
| #include "constants/annotation_common.h" |
| #include "constants/appearance.h" |
| #include "constants/form_flags.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/cpdf_document.h" |
| #include "core/fpdfapi/parser/cpdf_reference.h" |
| #include "core/fpdfapi/parser/cpdf_stream.h" |
| #include "core/fpdfapi/parser/cpdf_string.h" |
| #include "core/fpdfdoc/cpdf_bafontmap.h" |
| #include "core/fpdfdoc/cpdf_defaultappearance.h" |
| #include "core/fpdfdoc/cpdf_formcontrol.h" |
| #include "core/fpdfdoc/cpdf_formfield.h" |
| #include "core/fpdfdoc/cpdf_iconfit.h" |
| #include "core/fpdfdoc/cpdf_interactiveform.h" |
| #include "core/fxcrt/check.h" |
| #include "core/fxcrt/notreached.h" |
| #include "core/fxge/cfx_fillrenderoptions.h" |
| #include "core/fxge/cfx_graphstatedata.h" |
| #include "core/fxge/cfx_path.h" |
| #include "core/fxge/cfx_renderdevice.h" |
| #include "fpdfsdk/cpdfsdk_appstream.h" |
| #include "fpdfsdk/cpdfsdk_formfillenvironment.h" |
| #include "fpdfsdk/cpdfsdk_interactiveform.h" |
| #include "fpdfsdk/cpdfsdk_pageview.h" |
| #include "fpdfsdk/formfiller/cffl_fieldaction.h" |
| #include "fpdfsdk/pwl/cpwl_edit.h" |
| |
| #ifdef PDF_ENABLE_XFA |
| #include "fpdfsdk/fpdfxfa/cpdfxfa_context.h" |
| #include "xfa/fxfa/cxfa_eventparam.h" |
| #include "xfa/fxfa/cxfa_ffdocview.h" |
| #include "xfa/fxfa/cxfa_ffwidget.h" |
| #include "xfa/fxfa/cxfa_ffwidgethandler.h" |
| #include "xfa/fxfa/parser/cxfa_node.h" |
| #endif // PDF_ENABLE_XFA |
| |
| CPDFSDK_Widget::CPDFSDK_Widget(CPDF_Annot* pAnnot, |
| CPDFSDK_PageView* pPageView, |
| CPDFSDK_InteractiveForm* pInteractiveForm) |
| : CPDFSDK_BAAnnot(pAnnot, pPageView), |
| m_pInteractiveForm(pInteractiveForm) {} |
| |
| CPDFSDK_Widget::~CPDFSDK_Widget() { |
| GetInteractiveFormFiller()->OnDelete(this); |
| m_pInteractiveForm->RemoveMap(GetFormControl()); |
| } |
| |
| #ifdef PDF_ENABLE_XFA |
| CXFA_FFWidget* CPDFSDK_Widget::GetMixXFAWidget() const { |
| CPDF_Document::Extension* pContext = |
| GetPageView()->GetFormFillEnv()->GetDocExtension(); |
| if (!pContext || !pContext->ContainsExtensionForegroundForm()) |
| return nullptr; |
| |
| CXFA_FFDocView* pDocView = |
| static_cast<CPDFXFA_Context*>(pContext)->GetXFADocView(); |
| if (!pDocView) |
| return nullptr; |
| |
| WideString sName; |
| if (GetFieldType() == FormFieldType::kRadioButton) { |
| sName = GetAnnotName(); |
| if (sName.IsEmpty()) |
| sName = GetName(); |
| } else { |
| sName = GetName(); |
| } |
| |
| if (sName.IsEmpty()) |
| return nullptr; |
| |
| return pDocView->GetWidgetByName(sName, nullptr); |
| } |
| |
| CXFA_FFWidget* CPDFSDK_Widget::GetGroupMixXFAWidget() const { |
| CPDF_Document::Extension* pContext = |
| GetPageView()->GetFormFillEnv()->GetDocExtension(); |
| if (!pContext || !pContext->ContainsExtensionForegroundForm()) |
| return nullptr; |
| |
| CXFA_FFDocView* pDocView = |
| static_cast<CPDFXFA_Context*>(pContext)->GetXFADocView(); |
| if (!pDocView) |
| return nullptr; |
| |
| WideString sName = GetName(); |
| return !sName.IsEmpty() ? pDocView->GetWidgetByName(sName, nullptr) : nullptr; |
| } |
| |
| CXFA_FFWidgetHandler* CPDFSDK_Widget::GetXFAWidgetHandler() const { |
| CPDF_Document::Extension* pContext = |
| GetPageView()->GetFormFillEnv()->GetDocExtension(); |
| if (!pContext || !pContext->ContainsExtensionForegroundForm()) |
| return nullptr; |
| |
| CXFA_FFDocView* pDocView = |
| static_cast<CPDFXFA_Context*>(pContext)->GetXFADocView(); |
| return pDocView ? pDocView->GetWidgetHandler() : nullptr; |
| } |
| |
| static XFA_EVENTTYPE GetXFAEventType(PDFSDK_XFAAActionType eXFAAAT) { |
| XFA_EVENTTYPE eEventType = XFA_EVENT_Unknown; |
| |
| switch (eXFAAAT) { |
| case PDFSDK_XFA_Click: |
| eEventType = XFA_EVENT_Click; |
| break; |
| case PDFSDK_XFA_Full: |
| eEventType = XFA_EVENT_Full; |
| break; |
| case PDFSDK_XFA_PreOpen: |
| eEventType = XFA_EVENT_PreOpen; |
| break; |
| case PDFSDK_XFA_PostOpen: |
| eEventType = XFA_EVENT_PostOpen; |
| break; |
| } |
| |
| return eEventType; |
| } |
| |
| static XFA_EVENTTYPE GetXFAEventType(CPDF_AAction::AActionType eAAT, |
| bool bWillCommit) { |
| XFA_EVENTTYPE eEventType = XFA_EVENT_Unknown; |
| |
| switch (eAAT) { |
| case CPDF_AAction::kCursorEnter: |
| eEventType = XFA_EVENT_MouseEnter; |
| break; |
| case CPDF_AAction::kCursorExit: |
| eEventType = XFA_EVENT_MouseExit; |
| break; |
| case CPDF_AAction::kButtonDown: |
| eEventType = XFA_EVENT_MouseDown; |
| break; |
| case CPDF_AAction::kButtonUp: |
| eEventType = XFA_EVENT_MouseUp; |
| break; |
| case CPDF_AAction::kGetFocus: |
| eEventType = XFA_EVENT_Enter; |
| break; |
| case CPDF_AAction::kLoseFocus: |
| eEventType = XFA_EVENT_Exit; |
| break; |
| case CPDF_AAction::kPageOpen: |
| case CPDF_AAction::kPageClose: |
| case CPDF_AAction::kPageVisible: |
| case CPDF_AAction::kPageInvisible: |
| break; |
| case CPDF_AAction::kKeyStroke: |
| if (!bWillCommit) |
| eEventType = XFA_EVENT_Change; |
| break; |
| case CPDF_AAction::kValidate: |
| eEventType = XFA_EVENT_Validate; |
| break; |
| case CPDF_AAction::kOpenPage: |
| case CPDF_AAction::kClosePage: |
| case CPDF_AAction::kFormat: |
| case CPDF_AAction::kCalculate: |
| case CPDF_AAction::kCloseDocument: |
| case CPDF_AAction::kSaveDocument: |
| case CPDF_AAction::kDocumentSaved: |
| case CPDF_AAction::kPrintDocument: |
| case CPDF_AAction::kDocumentPrinted: |
| break; |
| case CPDF_AAction::kDocumentOpen: |
| case CPDF_AAction::kNumberOfActions: |
| NOTREACHED_NORETURN(); |
| } |
| |
| return eEventType; |
| } |
| |
| bool CPDFSDK_Widget::HasXFAAAction(PDFSDK_XFAAActionType eXFAAAT) const { |
| CXFA_FFWidget* pWidget = GetMixXFAWidget(); |
| if (!pWidget) |
| return false; |
| |
| CXFA_FFWidgetHandler* pXFAWidgetHandler = GetXFAWidgetHandler(); |
| if (!pXFAWidgetHandler) |
| return false; |
| |
| XFA_EVENTTYPE eEventType = GetXFAEventType(eXFAAAT); |
| if ((eEventType == XFA_EVENT_Click || eEventType == XFA_EVENT_Change) && |
| GetFieldType() == FormFieldType::kRadioButton) { |
| CXFA_FFWidget* hGroupWidget = GetGroupMixXFAWidget(); |
| if (hGroupWidget && |
| hGroupWidget->HasEventUnderHandler(eEventType, pXFAWidgetHandler)) { |
| return true; |
| } |
| } |
| |
| return pWidget->HasEventUnderHandler(eEventType, pXFAWidgetHandler); |
| } |
| |
| bool CPDFSDK_Widget::OnXFAAAction(PDFSDK_XFAAActionType eXFAAAT, |
| CFFL_FieldAction* data, |
| const CPDFSDK_PageView* pPageView) { |
| auto* pContext = static_cast<CPDFXFA_Context*>( |
| GetPageView()->GetFormFillEnv()->GetDocExtension()); |
| if (!pContext) |
| return false; |
| |
| CXFA_FFWidget* pWidget = GetMixXFAWidget(); |
| if (!pWidget) |
| return false; |
| |
| XFA_EVENTTYPE eEventType = GetXFAEventType(eXFAAAT); |
| if (eEventType == XFA_EVENT_Unknown) |
| return false; |
| |
| CXFA_FFWidgetHandler* pXFAWidgetHandler = GetXFAWidgetHandler(); |
| if (!pXFAWidgetHandler) |
| return false; |
| |
| CXFA_EventParam param(eEventType); |
| param.m_wsChange = data->sChange; |
| param.m_iCommitKey = 0; |
| param.m_bShift = data->bShift; |
| param.m_iSelStart = data->nSelStart; |
| param.m_iSelEnd = data->nSelEnd; |
| param.m_wsFullText = data->sValue; |
| param.m_bKeyDown = data->bKeyDown; |
| param.m_bModifier = data->bModifier; |
| param.m_wsPrevText = data->sValue; |
| if ((eEventType == XFA_EVENT_Click || eEventType == XFA_EVENT_Change) && |
| GetFieldType() == FormFieldType::kRadioButton) { |
| CXFA_FFWidget* hGroupWidget = GetGroupMixXFAWidget(); |
| if (hGroupWidget && |
| !hGroupWidget->ProcessEventUnderHandler(¶m, pXFAWidgetHandler)) { |
| return false; |
| } |
| } |
| |
| bool ret = pWidget->ProcessEventUnderHandler(¶m, pXFAWidgetHandler); |
| CXFA_FFDocView* pDocView = pContext->GetXFADocView(); |
| if (pDocView) |
| pDocView->UpdateDocView(); |
| |
| return ret; |
| } |
| |
| void CPDFSDK_Widget::Synchronize(bool bSynchronizeElse) { |
| CXFA_FFWidget* hWidget = GetMixXFAWidget(); |
| if (!hWidget) |
| return; |
| |
| CXFA_Node* node = hWidget->GetNode(); |
| if (!node->IsWidgetReady()) |
| return; |
| |
| CPDF_FormField* pFormField = GetFormField(); |
| switch (GetFieldType()) { |
| case FormFieldType::kCheckBox: |
| case FormFieldType::kRadioButton: { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| XFA_CheckState eCheckState = |
| pFormCtrl->IsChecked() ? XFA_CheckState::kOn : XFA_CheckState::kOff; |
| node->SetCheckState(eCheckState); |
| break; |
| } |
| case FormFieldType::kTextField: |
| node->SetValue(XFA_ValuePicture::kEdit, pFormField->GetValue()); |
| break; |
| case FormFieldType::kComboBox: |
| case FormFieldType::kListBox: { |
| node->ClearAllSelections(); |
| for (int i = 0; i < pFormField->CountSelectedItems(); ++i) { |
| int nIndex = pFormField->GetSelectedIndex(i); |
| if (nIndex > -1 && |
| static_cast<size_t>(nIndex) < node->CountChoiceListItems(false)) { |
| node->SetItemState(nIndex, true, false, false); |
| } |
| } |
| if (GetFieldType() == FormFieldType::kComboBox) |
| node->SetValue(XFA_ValuePicture::kEdit, pFormField->GetValue()); |
| break; |
| } |
| default: |
| break; |
| } |
| |
| if (bSynchronizeElse) { |
| auto* context = static_cast<CPDFXFA_Context*>( |
| GetPageView()->GetFormFillEnv()->GetDocExtension()); |
| context->GetXFADocView()->ProcessValueChanged(node); |
| } |
| } |
| |
| bool CPDFSDK_Widget::HandleXFAAAction( |
| CPDF_AAction::AActionType type, |
| CFFL_FieldAction* data, |
| CPDFSDK_FormFillEnvironment* pFormFillEnv) { |
| auto* pContext = |
| static_cast<CPDFXFA_Context*>(pFormFillEnv->GetDocExtension()); |
| if (!pContext) |
| return false; |
| |
| CXFA_FFWidget* hWidget = GetMixXFAWidget(); |
| if (!hWidget) |
| return false; |
| |
| XFA_EVENTTYPE eEventType = GetXFAEventType(type, data->bWillCommit); |
| if (eEventType == XFA_EVENT_Unknown) |
| return false; |
| |
| CXFA_FFWidgetHandler* pXFAWidgetHandler = GetXFAWidgetHandler(); |
| if (!pXFAWidgetHandler) |
| return false; |
| |
| CXFA_EventParam param(eEventType); |
| param.m_wsChange = data->sChange; |
| param.m_iCommitKey = 0; |
| param.m_bShift = data->bShift; |
| param.m_iSelStart = data->nSelStart; |
| param.m_iSelEnd = data->nSelEnd; |
| param.m_wsFullText = data->sValue; |
| param.m_bKeyDown = data->bKeyDown; |
| param.m_bModifier = data->bModifier; |
| param.m_wsPrevText = data->sValue; |
| bool ret = hWidget->ProcessEventUnderHandler(¶m, pXFAWidgetHandler); |
| CXFA_FFDocView* pDocView = pContext->GetXFADocView(); |
| if (pDocView) |
| pDocView->UpdateDocView(); |
| |
| return ret; |
| } |
| #endif // PDF_ENABLE_XFA |
| |
| bool CPDFSDK_Widget::IsWidgetAppearanceValid( |
| CPDF_Annot::AppearanceMode mode) const { |
| RetainPtr<const CPDF_Dictionary> pAP = |
| GetAnnotDict()->GetDictFor(pdfium::annotation::kAP); |
| if (!pAP) |
| return false; |
| |
| // Choose the right sub-ap |
| const char* ap_entry = "N"; |
| if (mode == CPDF_Annot::AppearanceMode::kDown) |
| ap_entry = "D"; |
| else if (mode == CPDF_Annot::AppearanceMode::kRollover) |
| ap_entry = "R"; |
| if (!pAP->KeyExist(ap_entry)) |
| ap_entry = "N"; |
| |
| // Get the AP stream or subdirectory |
| RetainPtr<const CPDF_Object> pSub = pAP->GetDirectObjectFor(ap_entry); |
| if (!pSub) |
| return false; |
| |
| FormFieldType fieldType = GetFieldType(); |
| switch (fieldType) { |
| case FormFieldType::kPushButton: |
| case FormFieldType::kComboBox: |
| case FormFieldType::kListBox: |
| case FormFieldType::kTextField: |
| case FormFieldType::kSignature: |
| return pSub->IsStream(); |
| case FormFieldType::kCheckBox: |
| case FormFieldType::kRadioButton: |
| if (const CPDF_Dictionary* pSubDict = pSub->AsDictionary()) { |
| return !!pSubDict->GetStreamFor(GetAppState()); |
| } |
| return false; |
| default: |
| return true; |
| } |
| } |
| |
| bool CPDFSDK_Widget::IsPushHighlighted() const { |
| return GetFormControl()->GetHighlightingMode() == CPDF_FormControl::kPush; |
| } |
| |
| FormFieldType CPDFSDK_Widget::GetFieldType() const { |
| CPDF_FormField* pField = GetFormField(); |
| return pField ? pField->GetFieldType() : FormFieldType::kUnknown; |
| } |
| |
| void CPDFSDK_Widget::SetRect(const CFX_FloatRect& rect) { |
| DCHECK(rect.right - rect.left >= 1.0f); |
| DCHECK(rect.top - rect.bottom >= 1.0f); |
| GetMutableAnnotDict()->SetRectFor(pdfium::annotation::kRect, rect); |
| } |
| |
| bool CPDFSDK_Widget::IsAppearanceValid() { |
| #ifdef PDF_ENABLE_XFA |
| CPDF_Document::Extension* pContext = |
| GetPageView()->GetFormFillEnv()->GetDocExtension(); |
| if (pContext && pContext->ContainsExtensionFullForm()) |
| return true; |
| #endif // PDF_ENABLE_XFA |
| return CPDFSDK_BAAnnot::IsAppearanceValid(); |
| } |
| |
| int CPDFSDK_Widget::GetLayoutOrder() const { |
| return 2; |
| } |
| |
| int CPDFSDK_Widget::GetFieldFlags() const { |
| return GetFormField()->GetFieldFlags(); |
| } |
| |
| bool CPDFSDK_Widget::IsSignatureWidget() const { |
| return GetFieldType() == FormFieldType::kSignature; |
| } |
| |
| CPDF_FormField* CPDFSDK_Widget::GetFormField() const { |
| CPDF_FormControl* pControl = GetFormControl(); |
| return pControl ? pControl->GetField() : nullptr; |
| } |
| |
| CPDF_FormControl* CPDFSDK_Widget::GetFormControl() const { |
| CPDF_InteractiveForm* pPDFInteractiveForm = |
| m_pInteractiveForm->GetInteractiveForm(); |
| return pPDFInteractiveForm->GetControlByDict(GetAnnotDict()); |
| } |
| |
| int CPDFSDK_Widget::GetRotate() const { |
| CPDF_FormControl* pCtrl = GetFormControl(); |
| return pCtrl->GetRotation() % 360; |
| } |
| |
| #ifdef PDF_ENABLE_XFA |
| WideString CPDFSDK_Widget::GetName() const { |
| return GetFormField()->GetFullName(); |
| } |
| #endif // PDF_ENABLE_XFA |
| |
| std::optional<FX_COLORREF> CPDFSDK_Widget::GetFillColor() const { |
| CFX_Color::TypeAndARGB type_argb_pair = |
| GetFormControl()->GetColorARGB(pdfium::appearance::kBG); |
| |
| if (type_argb_pair.color_type == CFX_Color::Type::kTransparent) |
| return std::nullopt; |
| |
| return ArgbToColorRef(type_argb_pair.argb); |
| } |
| |
| std::optional<FX_COLORREF> CPDFSDK_Widget::GetBorderColor() const { |
| CFX_Color::TypeAndARGB type_argb_pair = |
| GetFormControl()->GetColorARGB(pdfium::appearance::kBC); |
| if (type_argb_pair.color_type == CFX_Color::Type::kTransparent) |
| return std::nullopt; |
| |
| return ArgbToColorRef(type_argb_pair.argb); |
| } |
| |
| std::optional<FX_COLORREF> CPDFSDK_Widget::GetTextColor() const { |
| CPDF_DefaultAppearance da = GetFormControl()->GetDefaultAppearance(); |
| std::optional<CFX_Color::TypeAndARGB> maybe_type_argb_pair = |
| da.GetColorARGB(); |
| |
| if (!maybe_type_argb_pair.has_value()) |
| return std::nullopt; |
| |
| if (maybe_type_argb_pair.value().color_type == CFX_Color::Type::kTransparent) |
| return std::nullopt; |
| |
| return ArgbToColorRef(maybe_type_argb_pair.value().argb); |
| } |
| |
| float CPDFSDK_Widget::GetFontSize() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| CPDF_DefaultAppearance pDa = pFormCtrl->GetDefaultAppearance(); |
| float fFontSize; |
| pDa.GetFont(&fFontSize); |
| return fFontSize; |
| } |
| |
| int CPDFSDK_Widget::GetSelectedIndex(int nIndex) const { |
| #ifdef PDF_ENABLE_XFA |
| if (CXFA_FFWidget* hWidget = GetMixXFAWidget()) { |
| CXFA_Node* node = hWidget->GetNode(); |
| if (node->IsWidgetReady()) { |
| if (nIndex < node->CountSelectedItems()) |
| return node->GetSelectedItem(nIndex); |
| } |
| } |
| #endif // PDF_ENABLE_XFA |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->GetSelectedIndex(nIndex); |
| } |
| |
| WideString CPDFSDK_Widget::GetValue() const { |
| #ifdef PDF_ENABLE_XFA |
| if (CXFA_FFWidget* hWidget = GetMixXFAWidget()) { |
| CXFA_Node* node = hWidget->GetNode(); |
| if (node->IsWidgetReady()) |
| return node->GetValue(XFA_ValuePicture::kDisplay); |
| } |
| #endif // PDF_ENABLE_XFA |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->GetValue(); |
| } |
| |
| WideString CPDFSDK_Widget::GetExportValue() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| return pFormCtrl->GetExportValue(); |
| } |
| |
| WideString CPDFSDK_Widget::GetOptionLabel(int nIndex) const { |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->GetOptionLabel(nIndex); |
| } |
| |
| WideString CPDFSDK_Widget::GetSelectExportText(int nIndex) const { |
| if (nIndex < 0) |
| return WideString(); |
| |
| CPDF_FormField* pFormField = GetFormField(); |
| if (!pFormField) |
| return WideString(); |
| |
| WideString swRet = pFormField->GetOptionValue(nIndex); |
| if (!swRet.IsEmpty()) |
| return swRet; |
| |
| return pFormField->GetOptionLabel(nIndex); |
| } |
| |
| int CPDFSDK_Widget::CountOptions() const { |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->CountOptions(); |
| } |
| |
| bool CPDFSDK_Widget::IsOptionSelected(int nIndex) const { |
| #ifdef PDF_ENABLE_XFA |
| if (CXFA_FFWidget* hWidget = GetMixXFAWidget()) { |
| CXFA_Node* node = hWidget->GetNode(); |
| if (node->IsWidgetReady()) { |
| if (nIndex > -1 && |
| static_cast<size_t>(nIndex) < node->CountChoiceListItems(false)) { |
| return node->GetItemState(nIndex); |
| } |
| return false; |
| } |
| } |
| #endif // PDF_ENABLE_XFA |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->IsItemSelected(nIndex); |
| } |
| |
| int CPDFSDK_Widget::GetTopVisibleIndex() const { |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->GetTopVisibleIndex(); |
| } |
| |
| bool CPDFSDK_Widget::IsChecked() const { |
| #ifdef PDF_ENABLE_XFA |
| if (CXFA_FFWidget* hWidget = GetMixXFAWidget()) { |
| CXFA_Node* node = hWidget->GetNode(); |
| if (node->IsWidgetReady()) |
| return node->GetCheckState() == XFA_CheckState::kOn; |
| } |
| #endif // PDF_ENABLE_XFA |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| return pFormCtrl->IsChecked(); |
| } |
| |
| int CPDFSDK_Widget::GetAlignment() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| return pFormCtrl->GetControlAlignment(); |
| } |
| |
| int CPDFSDK_Widget::GetMaxLen() const { |
| CPDF_FormField* pFormField = GetFormField(); |
| return pFormField->GetMaxLen(); |
| } |
| |
| void CPDFSDK_Widget::SetCheck(bool bChecked) { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| CPDF_FormField* pFormField = pFormCtrl->GetField(); |
| pFormField->CheckControl(pFormField->GetControlIndex(pFormCtrl), bChecked, |
| NotificationOption::kDoNotNotify); |
| #ifdef PDF_ENABLE_XFA |
| if (!IsWidgetAppearanceValid(CPDF_Annot::AppearanceMode::kNormal)) |
| ResetXFAAppearance(CPDFSDK_Widget::kValueChanged); |
| Synchronize(true); |
| #endif // PDF_ENABLE_XFA |
| } |
| |
| void CPDFSDK_Widget::SetValue(const WideString& sValue) { |
| CPDF_FormField* pFormField = GetFormField(); |
| pFormField->SetValue(sValue, NotificationOption::kDoNotNotify); |
| #ifdef PDF_ENABLE_XFA |
| Synchronize(true); |
| #endif // PDF_ENABLE_XFA |
| } |
| |
| void CPDFSDK_Widget::SetOptionSelection(int index) { |
| CPDF_FormField* pFormField = GetFormField(); |
| pFormField->SetItemSelection(index, NotificationOption::kDoNotNotify); |
| #ifdef PDF_ENABLE_XFA |
| Synchronize(true); |
| #endif // PDF_ENABLE_XFA |
| } |
| |
| void CPDFSDK_Widget::ClearSelection() { |
| CPDF_FormField* pFormField = GetFormField(); |
| pFormField->ClearSelection(NotificationOption::kDoNotNotify); |
| #ifdef PDF_ENABLE_XFA |
| Synchronize(true); |
| #endif // PDF_ENABLE_XFA |
| } |
| |
| void CPDFSDK_Widget::SetTopVisibleIndex(int index) {} |
| |
| void CPDFSDK_Widget::SetAppModified() { |
| m_bAppModified = true; |
| } |
| |
| void CPDFSDK_Widget::ClearAppModified() { |
| m_bAppModified = false; |
| } |
| |
| bool CPDFSDK_Widget::IsAppModified() const { |
| return m_bAppModified; |
| } |
| |
| #ifdef PDF_ENABLE_XFA |
| void CPDFSDK_Widget::ResetXFAAppearance(ValueChanged bValueChanged) { |
| switch (GetFieldType()) { |
| case FormFieldType::kTextField: |
| case FormFieldType::kComboBox: { |
| ResetAppearance(OnFormat(), kValueChanged); |
| break; |
| } |
| default: |
| ResetAppearance(std::nullopt, kValueUnchanged); |
| break; |
| } |
| } |
| #endif // PDF_ENABLE_XFA |
| |
| void CPDFSDK_Widget::ResetAppearance(std::optional<WideString> sValue, |
| ValueChanged bValueChanged) { |
| SetAppModified(); |
| |
| m_nAppearanceAge++; |
| if (bValueChanged == kValueChanged) |
| m_nValueAge++; |
| |
| CPDFSDK_AppStream appStream(this, GetAPDict().Get()); |
| switch (GetFieldType()) { |
| case FormFieldType::kPushButton: |
| appStream.SetAsPushButton(); |
| break; |
| case FormFieldType::kCheckBox: |
| appStream.SetAsCheckBox(); |
| break; |
| case FormFieldType::kRadioButton: |
| appStream.SetAsRadioButton(); |
| break; |
| case FormFieldType::kComboBox: |
| appStream.SetAsComboBox(sValue); |
| break; |
| case FormFieldType::kListBox: |
| appStream.SetAsListBox(); |
| break; |
| case FormFieldType::kTextField: |
| appStream.SetAsTextField(sValue); |
| break; |
| default: |
| break; |
| } |
| |
| ClearCachedAnnotAP(); |
| } |
| |
| std::optional<WideString> CPDFSDK_Widget::OnFormat() { |
| CPDF_FormField* pFormField = GetFormField(); |
| DCHECK(pFormField); |
| return m_pInteractiveForm->OnFormat(pFormField); |
| } |
| |
| void CPDFSDK_Widget::ResetFieldAppearance() { |
| CPDF_FormField* pFormField = GetFormField(); |
| DCHECK(pFormField); |
| m_pInteractiveForm->ResetFieldAppearance(pFormField, std::nullopt); |
| } |
| |
| void CPDFSDK_Widget::OnDraw(CFX_RenderDevice* pDevice, |
| const CFX_Matrix& mtUser2Device, |
| bool bDrawAnnots) { |
| if (IsSignatureWidget()) { |
| DrawAppearance(pDevice, mtUser2Device, CPDF_Annot::AppearanceMode::kNormal); |
| return; |
| } |
| |
| GetInteractiveFormFiller()->OnDraw(GetPageView(), this, pDevice, |
| mtUser2Device); |
| } |
| |
| bool CPDFSDK_Widget::DoHitTest(const CFX_PointF& point) { |
| if (IsSignatureWidget() || !IsVisible()) |
| return false; |
| |
| if (GetFieldFlags() & pdfium::form_flags::kReadOnly) |
| return false; |
| |
| bool do_hit_test = GetFieldType() == FormFieldType::kPushButton; |
| if (!do_hit_test) { |
| uint32_t perms = GetPDFPage()->GetDocument()->GetUserPermissions( |
| /*get_owner_perms=*/true); |
| do_hit_test = (perms & pdfium::access_permissions::kFillForm) || |
| (perms & pdfium::access_permissions::kModifyAnnotation); |
| } |
| return do_hit_test && GetViewBBox().Contains(point); |
| } |
| |
| CFX_FloatRect CPDFSDK_Widget::GetViewBBox() { |
| if (IsSignatureWidget()) |
| return CFX_FloatRect(); |
| |
| auto* form_filler = GetInteractiveFormFiller(); |
| return CFX_FloatRect(form_filler->GetViewBBox(GetPageView(), this)); |
| } |
| |
| void CPDFSDK_Widget::OnMouseEnter(Mask<FWL_EVENTFLAG> nFlags) { |
| if (IsSignatureWidget()) |
| return; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| GetInteractiveFormFiller()->OnMouseEnter(GetPageView(), observer, nFlags); |
| } |
| |
| void CPDFSDK_Widget::OnMouseExit(Mask<FWL_EVENTFLAG> nFlags) { |
| if (IsSignatureWidget()) |
| return; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| GetInteractiveFormFiller()->OnMouseExit(GetPageView(), observer, nFlags); |
| } |
| |
| bool CPDFSDK_Widget::OnLButtonDown(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnLButtonDown(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnLButtonUp(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnLButtonUp(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnLButtonDblClk(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnLButtonDblClk(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnMouseMove(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnMouseMove(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnMouseWheel(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point, |
| const CFX_Vector& delta) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnMouseWheel(GetPageView(), observer, |
| nFlags, point, delta); |
| } |
| |
| bool CPDFSDK_Widget::OnRButtonDown(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnRButtonDown(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnRButtonUp(Mask<FWL_EVENTFLAG> nFlags, |
| const CFX_PointF& point) { |
| if (IsSignatureWidget()) |
| return false; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnRButtonUp(GetPageView(), observer, |
| nFlags, point); |
| } |
| |
| bool CPDFSDK_Widget::OnChar(uint32_t nChar, Mask<FWL_EVENTFLAG> nFlags) { |
| return !IsSignatureWidget() && |
| GetInteractiveFormFiller()->OnChar(this, nChar, nFlags); |
| } |
| |
| bool CPDFSDK_Widget::OnKeyDown(FWL_VKEYCODE nKeyCode, |
| Mask<FWL_EVENTFLAG> nFlags) { |
| return !IsSignatureWidget() && |
| GetInteractiveFormFiller()->OnKeyDown(this, nKeyCode, nFlags); |
| } |
| |
| bool CPDFSDK_Widget::OnSetFocus(Mask<FWL_EVENTFLAG> nFlags) { |
| if (!IsFocusableAnnot(GetPDFAnnot()->GetSubtype())) |
| return false; |
| |
| if (IsSignatureWidget()) |
| return true; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnSetFocus(observer, nFlags); |
| } |
| |
| bool CPDFSDK_Widget::OnKillFocus(Mask<FWL_EVENTFLAG> nFlags) { |
| if (!IsFocusableAnnot(GetPDFAnnot()->GetSubtype())) |
| return false; |
| |
| if (IsSignatureWidget()) |
| return true; |
| |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return GetInteractiveFormFiller()->OnKillFocus(observer, nFlags); |
| } |
| |
| bool CPDFSDK_Widget::CanUndo() { |
| return !IsSignatureWidget() && GetInteractiveFormFiller()->CanUndo(this); |
| } |
| |
| bool CPDFSDK_Widget::CanRedo() { |
| return !IsSignatureWidget() && GetInteractiveFormFiller()->CanRedo(this); |
| } |
| |
| bool CPDFSDK_Widget::Undo() { |
| return !IsSignatureWidget() && GetInteractiveFormFiller()->Undo(this); |
| } |
| |
| bool CPDFSDK_Widget::Redo() { |
| return !IsSignatureWidget() && GetInteractiveFormFiller()->Redo(this); |
| } |
| |
| WideString CPDFSDK_Widget::GetText() { |
| if (IsSignatureWidget()) |
| return WideString(); |
| return GetInteractiveFormFiller()->GetText(this); |
| } |
| |
| WideString CPDFSDK_Widget::GetSelectedText() { |
| if (IsSignatureWidget()) |
| return WideString(); |
| return GetInteractiveFormFiller()->GetSelectedText(this); |
| } |
| |
| void CPDFSDK_Widget::ReplaceAndKeepSelection(const WideString& text) { |
| if (IsSignatureWidget()) |
| return; |
| |
| GetInteractiveFormFiller()->ReplaceAndKeepSelection(this, text); |
| } |
| |
| void CPDFSDK_Widget::ReplaceSelection(const WideString& text) { |
| if (IsSignatureWidget()) |
| return; |
| |
| GetInteractiveFormFiller()->ReplaceSelection(this, text); |
| } |
| |
| bool CPDFSDK_Widget::SelectAllText() { |
| return !IsSignatureWidget() && |
| GetInteractiveFormFiller()->SelectAllText(this); |
| } |
| |
| bool CPDFSDK_Widget::SetIndexSelected(int index, bool selected) { |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return !IsSignatureWidget() && GetInteractiveFormFiller()->SetIndexSelected( |
| observer, index, selected); |
| } |
| |
| bool CPDFSDK_Widget::IsIndexSelected(int index) { |
| ObservedPtr<CPDFSDK_Widget> observer(this); |
| return !IsSignatureWidget() && |
| GetInteractiveFormFiller()->IsIndexSelected(observer, index); |
| } |
| |
| void CPDFSDK_Widget::DrawAppearance(CFX_RenderDevice* pDevice, |
| const CFX_Matrix& mtUser2Device, |
| CPDF_Annot::AppearanceMode mode) { |
| FormFieldType fieldType = GetFieldType(); |
| |
| if ((fieldType == FormFieldType::kCheckBox || |
| fieldType == FormFieldType::kRadioButton) && |
| mode == CPDF_Annot::AppearanceMode::kNormal && |
| !IsWidgetAppearanceValid(CPDF_Annot::AppearanceMode::kNormal)) { |
| CFX_GraphStateData gsd; |
| gsd.m_LineWidth = 0.0f; |
| |
| CFX_Path path; |
| path.AppendFloatRect(GetRect()); |
| pDevice->DrawPath(path, &mtUser2Device, &gsd, 0, 0xFFAAAAAA, |
| CFX_FillRenderOptions::EvenOddOptions()); |
| } else { |
| CPDFSDK_BAAnnot::DrawAppearance(pDevice, mtUser2Device, mode); |
| } |
| } |
| |
| void CPDFSDK_Widget::UpdateField() { |
| CPDF_FormField* pFormField = GetFormField(); |
| DCHECK(pFormField); |
| m_pInteractiveForm->UpdateField(pFormField); |
| } |
| |
| void CPDFSDK_Widget::DrawShadow(CFX_RenderDevice* pDevice, |
| CPDFSDK_PageView* pPageView) { |
| FormFieldType fieldType = GetFieldType(); |
| if (!m_pInteractiveForm->IsNeedHighLight(fieldType)) |
| return; |
| |
| CFX_Matrix page2device = pPageView->GetCurrentMatrix(); |
| CFX_FloatRect rcDevice = GetRect(); |
| CFX_PointF tmp = |
| page2device.Transform(CFX_PointF(rcDevice.left, rcDevice.bottom)); |
| rcDevice.left = tmp.x; |
| rcDevice.bottom = tmp.y; |
| |
| tmp = page2device.Transform(CFX_PointF(rcDevice.right, rcDevice.top)); |
| rcDevice.right = tmp.x; |
| rcDevice.top = tmp.y; |
| rcDevice.Normalize(); |
| |
| pDevice->FillRect( |
| rcDevice.ToFxRect(), |
| AlphaAndColorRefToArgb( |
| static_cast<int>(m_pInteractiveForm->GetHighlightAlpha()), |
| m_pInteractiveForm->GetHighlightColor(fieldType))); |
| } |
| |
| CFX_FloatRect CPDFSDK_Widget::GetClientRect() const { |
| CFX_FloatRect rcWindow = GetRotatedRect(); |
| float fBorderWidth = GetBorderWidth(); |
| switch (GetBorderStyle()) { |
| case BorderStyle::kBeveled: |
| case BorderStyle::kInset: |
| fBorderWidth *= 2.0f; |
| break; |
| default: |
| break; |
| } |
| return rcWindow.GetDeflated(fBorderWidth, fBorderWidth); |
| } |
| |
| CFX_FloatRect CPDFSDK_Widget::GetRotatedRect() const { |
| CFX_FloatRect rectAnnot = GetRect(); |
| float fWidth = rectAnnot.Width(); |
| float fHeight = rectAnnot.Height(); |
| |
| CPDF_FormControl* pControl = GetFormControl(); |
| CFX_FloatRect rcPWLWindow; |
| switch (abs(pControl->GetRotation() % 360)) { |
| case 0: |
| case 180: |
| default: |
| rcPWLWindow = CFX_FloatRect(0, 0, fWidth, fHeight); |
| break; |
| case 90: |
| case 270: |
| rcPWLWindow = CFX_FloatRect(0, 0, fHeight, fWidth); |
| break; |
| } |
| |
| return rcPWLWindow; |
| } |
| |
| CFX_Matrix CPDFSDK_Widget::GetMatrix() const { |
| CFX_Matrix mt; |
| CPDF_FormControl* pControl = GetFormControl(); |
| CFX_FloatRect rcAnnot = GetRect(); |
| float fWidth = rcAnnot.Width(); |
| float fHeight = rcAnnot.Height(); |
| |
| switch (abs(pControl->GetRotation() % 360)) { |
| default: |
| case 0: |
| break; |
| case 90: |
| mt = CFX_Matrix(0, 1, -1, 0, fWidth, 0); |
| break; |
| case 180: |
| mt = CFX_Matrix(-1, 0, 0, -1, fWidth, fHeight); |
| break; |
| case 270: |
| mt = CFX_Matrix(0, -1, 1, 0, 0, fHeight); |
| break; |
| } |
| |
| return mt; |
| } |
| |
| CFX_Color CPDFSDK_Widget::GetTextPWLColor() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| std::optional<CFX_Color> crText = |
| pFormCtrl->GetDefaultAppearance().GetColor(); |
| return crText.value_or(CFX_Color(CFX_Color::Type::kGray, 0)); |
| } |
| |
| CFX_Color CPDFSDK_Widget::GetBorderPWLColor() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| return pFormCtrl->GetOriginalBorderColor(); |
| } |
| |
| CFX_Color CPDFSDK_Widget::GetFillPWLColor() const { |
| CPDF_FormControl* pFormCtrl = GetFormControl(); |
| return pFormCtrl->GetOriginalBackgroundColor(); |
| } |
| |
| bool CPDFSDK_Widget::OnAAction(CPDF_AAction::AActionType type, |
| CFFL_FieldAction* data, |
| const CPDFSDK_PageView* pPageView) { |
| CPDFSDK_FormFillEnvironment* pFormFillEnv = pPageView->GetFormFillEnv(); |
| |
| #ifdef PDF_ENABLE_XFA |
| if (HandleXFAAAction(type, data, pFormFillEnv)) |
| return true; |
| #endif // PDF_ENABLE_XFA |
| |
| CPDF_Action action = GetAAction(type); |
| if (action.GetType() != CPDF_Action::Type::kUnknown) { |
| pFormFillEnv->DoActionField(action, type, GetFormField(), data); |
| } |
| return false; |
| } |
| |
| void CPDFSDK_Widget::OnLoad() { |
| ObservedPtr<CPDFSDK_Widget> pObserved(this); |
| if (pObserved->IsSignatureWidget()) { |
| return; |
| } |
| if (!pObserved->IsAppearanceValid()) { |
| pObserved->ResetAppearance(std::nullopt, CPDFSDK_Widget::kValueUnchanged); |
| } |
| FormFieldType field_type = pObserved->GetFieldType(); |
| if (field_type == FormFieldType::kTextField || |
| field_type == FormFieldType::kComboBox) { |
| std::optional<WideString> sValue = pObserved->OnFormat(); |
| if (!pObserved) { |
| return; |
| } |
| if (sValue.has_value() && field_type == FormFieldType::kComboBox) { |
| pObserved->ResetAppearance(sValue, CPDFSDK_Widget::kValueUnchanged); |
| } |
| } |
| #ifdef PDF_ENABLE_XFA |
| auto* pContext = |
| pObserved->GetPageView()->GetFormFillEnv()->GetDocExtension(); |
| if (pContext && pContext->ContainsExtensionForegroundForm()) { |
| if (!pObserved->IsAppearanceValid() && !pObserved->GetValue().IsEmpty()) { |
| pObserved->ResetXFAAppearance(CPDFSDK_Widget::kValueUnchanged); |
| } |
| } |
| #endif // PDF_ENABLE_XFA |
| } |
| |
| CPDF_Action CPDFSDK_Widget::GetAAction(CPDF_AAction::AActionType eAAT) { |
| switch (eAAT) { |
| case CPDF_AAction::kCursorEnter: |
| case CPDF_AAction::kCursorExit: |
| case CPDF_AAction::kButtonDown: |
| case CPDF_AAction::kButtonUp: |
| case CPDF_AAction::kGetFocus: |
| case CPDF_AAction::kLoseFocus: |
| case CPDF_AAction::kPageOpen: |
| case CPDF_AAction::kPageClose: |
| case CPDF_AAction::kPageVisible: |
| case CPDF_AAction::kPageInvisible: |
| return CPDFSDK_BAAnnot::GetAAction(eAAT); |
| |
| case CPDF_AAction::kKeyStroke: |
| case CPDF_AAction::kFormat: |
| case CPDF_AAction::kValidate: |
| case CPDF_AAction::kCalculate: { |
| CPDF_FormField* pField = GetFormField(); |
| if (pField->GetAdditionalAction().HasDict()) |
| return pField->GetAdditionalAction().GetAction(eAAT); |
| return CPDFSDK_BAAnnot::GetAAction(eAAT); |
| } |
| default: |
| break; |
| } |
| |
| return CPDF_Action(nullptr); |
| } |
| |
| CFFL_InteractiveFormFiller* CPDFSDK_Widget::GetInteractiveFormFiller() { |
| return GetPageView()->GetFormFillEnv()->GetInteractiveFormFiller(); |
| } |