// 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 "fxjs/cjs_field.h" #include #include #include #include "constants/access_permissions.h" #include "constants/annotation_flags.h" #include "constants/form_flags.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfdoc/cpdf_formcontrol.h" #include "core/fpdfdoc/cpdf_formfield.h" #include "core/fpdfdoc/cpdf_interactiveform.h" #include "fpdfsdk/cpdfsdk_formfillenvironment.h" #include "fpdfsdk/cpdfsdk_interactiveform.h" #include "fpdfsdk/cpdfsdk_pageview.h" #include "fpdfsdk/cpdfsdk_widget.h" #include "fxjs/cjs_color.h" #include "fxjs/cjs_delaydata.h" #include "fxjs/cjs_document.h" #include "fxjs/cjs_icon.h" #include "fxjs/fxv8.h" #include "fxjs/js_resources.h" #include "third_party/abseil-cpp/absl/types/optional.h" #include "third_party/base/check.h" #include "third_party/base/notreached.h" #include "v8/include/v8-container.h" namespace { constexpr wchar_t kCheckSelector = L'4'; constexpr wchar_t kCircleSelector = L'l'; constexpr wchar_t kCrossSelector = L'8'; constexpr wchar_t kDiamondSelector = L'u'; constexpr wchar_t kSquareSelector = L'n'; constexpr wchar_t kStarSelector = L'H'; bool IsCheckBoxOrRadioButton(const CPDF_FormField* pFormField) { return pFormField->GetFieldType() == FormFieldType::kCheckBox || pFormField->GetFieldType() == FormFieldType::kRadioButton; } bool IsComboBoxOrListBox(const CPDF_FormField* pFormField) { return pFormField->GetFieldType() == FormFieldType::kComboBox || pFormField->GetFieldType() == FormFieldType::kListBox; } bool IsComboBoxOrTextField(const CPDF_FormField* pFormField) { return pFormField->GetFieldType() == FormFieldType::kComboBox || pFormField->GetFieldType() == FormFieldType::kTextField; } void UpdateFormField(CPDFSDK_FormFillEnvironment* pFormFillEnv, CPDF_FormField* pFormField, bool bResetAP) { CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm(); if (bResetAP) { std::vector> widgets; pForm->GetWidgets(pFormField, &widgets); if (IsComboBoxOrTextField(pFormField)) { for (auto& pWidget : widgets) { if (pWidget) { absl::optional sValue = pWidget->OnFormat(); if (pWidget) { // Not redundant, may be clobbered by OnFormat. pWidget->ResetAppearance(sValue, CPDFSDK_Widget::kValueUnchanged); } } } } else { for (auto& pWidget : widgets) { if (pWidget) { pWidget->ResetAppearance(absl::nullopt, CPDFSDK_Widget::kValueUnchanged); } } } } // Refresh the widget list. The calls in |bResetAP| may have caused widgets // to be removed from the list. We need to call |GetWidgets| again to be // sure none of the widgets have been deleted. std::vector> widgets; pForm->GetWidgets(pFormField, &widgets); for (auto& pWidget : widgets) { if (pWidget) pFormFillEnv->UpdateAllViews(pWidget.Get()); } pFormFillEnv->SetChangeMark(); } void UpdateFormControl(CPDFSDK_FormFillEnvironment* pFormFillEnv, CPDF_FormControl* pFormControl, bool bResetAP) { DCHECK(pFormControl); CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm(); CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl); if (pWidget) { ObservedPtr observed_widget(pWidget); if (bResetAP) { FormFieldType fieldType = pWidget->GetFieldType(); if (fieldType == FormFieldType::kComboBox || fieldType == FormFieldType::kTextField) { absl::optional sValue = pWidget->OnFormat(); if (!observed_widget) return; pWidget->ResetAppearance(sValue, CPDFSDK_Widget::kValueUnchanged); } else { pWidget->ResetAppearance(absl::nullopt, CPDFSDK_Widget::kValueUnchanged); } if (!observed_widget) return; } pFormFillEnv->UpdateAllViews(pWidget); } pFormFillEnv->SetChangeMark(); } struct FieldNameData { FieldNameData(WideString field_name_in, int control_index_in) : field_name(field_name_in), control_index(control_index_in) {} WideString field_name; int control_index; }; absl::optional ParseFieldName(const WideString& field_name) { auto reverse_it = field_name.rbegin(); while (reverse_it != field_name.rend()) { if (*reverse_it == L'.') break; ++reverse_it; } if (reverse_it == field_name.rend()) { return absl::nullopt; } WideString suffixal = field_name.Last(reverse_it - field_name.rbegin()); int control_index = FXSYS_wtoi(suffixal.c_str()); if (control_index == 0) { suffixal.TrimRight(L' '); if (suffixal != L"0") { return absl::nullopt; } } return FieldNameData(field_name.First(field_name.rend() - reverse_it - 1), control_index); } std::vector GetFormFieldsForName( CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& csFieldName) { std::vector fields; CPDFSDK_InteractiveForm* pReaderForm = pFormFillEnv->GetInteractiveForm(); CPDF_InteractiveForm* pForm = pReaderForm->GetInteractiveForm(); const size_t sz = pForm->CountFields(csFieldName); for (size_t i = 0; i < sz; ++i) { CPDF_FormField* pFormField = pForm->GetField(i, csFieldName); if (pFormField) fields.push_back(pFormField); } return fields; } CFX_Color GetFormControlColor(CPDF_FormControl* pFormControl, const ByteString& entry) { switch (pFormControl->GetColorARGB(entry).color_type) { case CFX_Color::Type::kTransparent: return CFX_Color(CFX_Color::Type::kTransparent); case CFX_Color::Type::kGray: return CFX_Color(CFX_Color::Type::kGray, pFormControl->GetOriginalColorComponent(0, entry)); case CFX_Color::Type::kRGB: return CFX_Color(CFX_Color::Type::kRGB, pFormControl->GetOriginalColorComponent(0, entry), pFormControl->GetOriginalColorComponent(1, entry), pFormControl->GetOriginalColorComponent(2, entry)); case CFX_Color::Type::kCMYK: return CFX_Color(CFX_Color::Type::kCMYK, pFormControl->GetOriginalColorComponent(0, entry), pFormControl->GetOriginalColorComponent(1, entry), pFormControl->GetOriginalColorComponent(2, entry), pFormControl->GetOriginalColorComponent(3, entry)); } } bool SetWidgetDisplayStatus(CPDFSDK_Widget* pWidget, int value) { if (!pWidget) return false; uint32_t dwFlag = pWidget->GetFlags(); switch (value) { case 0: dwFlag &= ~pdfium::annotation_flags::kInvisible; dwFlag &= ~pdfium::annotation_flags::kHidden; dwFlag &= ~pdfium::annotation_flags::kNoView; dwFlag |= pdfium::annotation_flags::kPrint; break; case 1: dwFlag &= ~pdfium::annotation_flags::kInvisible; dwFlag &= ~pdfium::annotation_flags::kNoView; dwFlag |= (pdfium::annotation_flags::kHidden | pdfium::annotation_flags::kPrint); break; case 2: dwFlag &= ~pdfium::annotation_flags::kInvisible; dwFlag &= ~pdfium::annotation_flags::kPrint; dwFlag &= ~pdfium::annotation_flags::kHidden; dwFlag &= ~pdfium::annotation_flags::kNoView; break; case 3: dwFlag |= pdfium::annotation_flags::kNoView; dwFlag |= pdfium::annotation_flags::kPrint; dwFlag &= ~pdfium::annotation_flags::kHidden; break; } if (dwFlag != pWidget->GetFlags()) { pWidget->SetFlags(dwFlag); return true; } return false; } void SetBorderStyle(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, const ByteString& bsString) { DCHECK(pFormFillEnv); BorderStyle nBorderStyle; if (bsString == "solid") nBorderStyle = BorderStyle::kSolid; else if (bsString == "beveled") nBorderStyle = BorderStyle::kBeveled; else if (bsString == "dashed") nBorderStyle = BorderStyle::kDash; else if (bsString == "inset") nBorderStyle = BorderStyle::kInset; else if (bsString == "underline") nBorderStyle = BorderStyle::kUnderline; else return; std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); auto* pForm = pFormFillEnv->GetInteractiveForm(); for (CPDF_FormField* pFormField : FieldArray) { if (nControlIndex < 0) { bool bSet = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; ++i) { CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormField->GetControl(i)); if (pWidget) { if (pWidget->GetBorderStyle() != nBorderStyle) { pWidget->SetBorderStyle(nBorderStyle); bSet = true; } } } if (bSet) UpdateFormField(pFormFillEnv, pFormField, true); } else { if (nControlIndex >= pFormField->CountControls()) return; if (CPDF_FormControl* pFormControl = pFormField->GetControl(nControlIndex)) { CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl); if (pWidget) { if (pWidget->GetBorderStyle() != nBorderStyle) { pWidget->SetBorderStyle(nBorderStyle); UpdateFormControl(pFormFillEnv, pFormControl, true); } } } } } } void SetCurrentValueIndices(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, const std::vector& array) { DCHECK(pFormFillEnv); std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); for (CPDF_FormField* pFormField : FieldArray) { if (!IsComboBoxOrListBox(pFormField)) continue; uint32_t dwFieldFlags = pFormField->GetFieldFlags(); pFormField->ClearSelection(NotificationOption::kNotify); for (size_t i = 0; i < array.size(); ++i) { if (i != 0 && !(dwFieldFlags & pdfium::form_flags::kChoiceMultiSelect)) break; if (array[i] < static_cast(pFormField->CountOptions()) && !pFormField->IsItemSelected(array[i])) { pFormField->SetItemSelection(array[i], NotificationOption::kDoNotNotify); } } UpdateFormField(pFormFillEnv, pFormField, true); } } void SetDisplay(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, int number) { CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm(); std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); for (CPDF_FormField* pFormField : FieldArray) { if (nControlIndex < 0) { bool bAnySet = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; ++i) { CPDF_FormControl* pFormControl = pFormField->GetControl(i); DCHECK(pFormControl); CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl); if (SetWidgetDisplayStatus(pWidget, number)) bAnySet = true; } if (bAnySet) UpdateFormField(pFormFillEnv, pFormField, false); } else { if (nControlIndex >= pFormField->CountControls()) return; CPDF_FormControl* pFormControl = pFormField->GetControl(nControlIndex); if (!pFormControl) return; CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl); if (SetWidgetDisplayStatus(pWidget, number)) UpdateFormControl(pFormFillEnv, pFormControl, false); } } } void SetHidden(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, bool b) { int display = b ? 1 /*Hidden*/ : 0 /*Visible*/; SetDisplay(pFormFillEnv, swFieldName, nControlIndex, display); } void SetLineWidth(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, int number) { CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm(); std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); for (CPDF_FormField* pFormField : FieldArray) { if (nControlIndex < 0) { bool bSet = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; ++i) { CPDF_FormControl* pFormControl = pFormField->GetControl(i); DCHECK(pFormControl); if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl)) { if (number != pWidget->GetBorderWidth()) { pWidget->SetBorderWidth(number); bSet = true; } } } if (bSet) UpdateFormField(pFormFillEnv, pFormField, true); } else { if (nControlIndex >= pFormField->CountControls()) return; if (CPDF_FormControl* pFormControl = pFormField->GetControl(nControlIndex)) { if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl)) { if (number != pWidget->GetBorderWidth()) { pWidget->SetBorderWidth(number); UpdateFormControl(pFormFillEnv, pFormControl, true); } } } } } } void SetRect(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, const CFX_FloatRect& rect) { CPDFSDK_InteractiveForm* pForm = pFormFillEnv->GetInteractiveForm(); std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); for (CPDF_FormField* pFormField : FieldArray) { if (nControlIndex < 0) { bool bSet = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; ++i) { CPDF_FormControl* pFormControl = pFormField->GetControl(i); DCHECK(pFormControl); if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl)) { CFX_FloatRect crRect = rect; CPDF_Page* pPDFPage = pWidget->GetPDFPage(); crRect.Intersect(pPDFPage->GetBBox()); if (!crRect.IsEmpty()) { CFX_FloatRect rcOld = pWidget->GetRect(); if (crRect.left != rcOld.left || crRect.right != rcOld.right || crRect.top != rcOld.top || crRect.bottom != rcOld.bottom) { pWidget->SetRect(crRect); bSet = true; } } } } if (bSet) UpdateFormField(pFormFillEnv, pFormField, true); continue; } if (nControlIndex >= pFormField->CountControls()) return; if (CPDF_FormControl* pFormControl = pFormField->GetControl(nControlIndex)) { if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl)) { CFX_FloatRect crRect = rect; CPDF_Page* pPDFPage = pWidget->GetPDFPage(); crRect.Intersect(pPDFPage->GetBBox()); if (!crRect.IsEmpty()) { CFX_FloatRect rcOld = pWidget->GetRect(); if (crRect.left != rcOld.left || crRect.right != rcOld.right || crRect.top != rcOld.top || crRect.bottom != rcOld.bottom) { pWidget->SetRect(crRect); UpdateFormControl(pFormFillEnv, pFormControl, true); } } } } } } void SetFieldValue(CPDFSDK_FormFillEnvironment* pFormFillEnv, const WideString& swFieldName, int nControlIndex, const std::vector& strArray) { DCHECK(pFormFillEnv); if (strArray.empty()) return; std::vector FieldArray = GetFormFieldsForName(pFormFillEnv, swFieldName); for (CPDF_FormField* pFormField : FieldArray) { if (pFormField->GetFullName() != swFieldName) continue; switch (pFormField->GetFieldType()) { case FormFieldType::kTextField: case FormFieldType::kComboBox: if (pFormField->GetValue() != strArray[0]) { pFormField->SetValue(strArray[0], NotificationOption::kNotify); UpdateFormField(pFormFillEnv, pFormField, false); } break; case FormFieldType::kCheckBox: case FormFieldType::kRadioButton: if (pFormField->GetValue() != strArray[0]) { pFormField->SetValue(strArray[0], NotificationOption::kNotify); UpdateFormField(pFormFillEnv, pFormField, false); } break; case FormFieldType::kListBox: { bool bModified = false; for (const auto& str : strArray) { if (!pFormField->IsItemSelected(pFormField->FindOption(str))) { bModified = true; break; } } if (bModified) { pFormField->ClearSelection(NotificationOption::kNotify); for (const auto& str : strArray) { int index = pFormField->FindOption(str); if (!pFormField->IsItemSelected(index)) pFormField->SetItemSelection(index, NotificationOption::kNotify); } UpdateFormField(pFormFillEnv, pFormField, false); } break; } default: break; } } } wchar_t GetSelectorFromCaptionForFieldType(const WideString& caption, CPDF_FormField::Type type) { if (!caption.IsEmpty()) return caption[0]; switch (type) { case CPDF_FormField::kRadioButton: return kCircleSelector; default: return kCheckSelector; } } } // namespace const JSPropertySpec CJS_Field::PropertySpecs[] = { {"alignment", get_alignment_static, set_alignment_static}, {"borderStyle", get_border_style_static, set_border_style_static}, {"buttonAlignX", get_button_align_x_static, set_button_align_x_static}, {"buttonAlignY", get_button_align_y_static, set_button_align_y_static}, {"buttonFitBounds", get_button_fit_bounds_static, set_button_fit_bounds_static}, {"buttonPosition", get_button_position_static, set_button_position_static}, {"buttonScaleHow", get_button_scale_how_static, set_button_scale_how_static}, {"buttonScaleWhen", get_button_scale_when_static, set_button_scale_when_static}, {"calcOrderIndex", get_calc_order_index_static, set_calc_order_index_static}, {"charLimit", get_char_limit_static, set_char_limit_static}, {"comb", get_comb_static, set_comb_static}, {"commitOnSelChange", get_commit_on_sel_change_static, set_commit_on_sel_change_static}, {"currentValueIndices", get_current_value_indices_static, set_current_value_indices_static}, {"defaultStyle", get_default_style_static, set_default_style_static}, {"defaultValue", get_default_value_static, set_default_value_static}, {"doNotScroll", get_do_not_scroll_static, set_do_not_scroll_static}, {"doNotSpellCheck", get_do_not_spell_check_static, set_do_not_spell_check_static}, {"delay", get_delay_static, set_delay_static}, {"display", get_display_static, set_display_static}, {"doc", get_doc_static, set_doc_static}, {"editable", get_editable_static, set_editable_static}, {"exportValues", get_export_values_static, set_export_values_static}, {"hidden", get_hidden_static, set_hidden_static}, {"fileSelect", get_file_select_static, set_file_select_static}, {"fillColor", get_fill_color_static, set_fill_color_static}, {"lineWidth", get_line_width_static, set_line_width_static}, {"highlight", get_highlight_static, set_highlight_static}, {"multiline", get_multiline_static, set_multiline_static}, {"multipleSelection", get_multiple_selection_static, set_multiple_selection_static}, {"name", get_name_static, set_name_static}, {"numItems", get_num_items_static, set_num_items_static}, {"page", get_page_static, set_page_static}, {"password", get_password_static, set_password_static}, {"print", get_print_static, set_print_static}, {"radiosInUnison", get_radios_in_unison_static, set_radios_in_unison_static}, {"readonly", get_readonly_static, set_readonly_static}, {"rect", get_rect_static, set_rect_static}, {"required", get_required_static, set_required_static}, {"richText", get_rich_text_static, set_rich_text_static}, {"richValue", get_rich_value_static, set_rich_value_static}, {"rotation", get_rotation_static, set_rotation_static}, {"source", get_source_static, set_source_static}, {"strokeColor", get_stroke_color_static, set_stroke_color_static}, {"style", get_style_static, set_style_static}, {"submitName", get_submit_name_static, set_submit_name_static}, {"textColor", get_text_color_static, set_text_color_static}, {"textFont", get_text_font_static, set_text_font_static}, {"textSize", get_text_size_static, set_text_size_static}, {"type", get_type_static, set_type_static}, {"userName", get_user_name_static, set_user_name_static}, {"value", get_value_static, set_value_static}, {"valueAsString", get_value_as_string_static, set_value_as_string_static}}; const JSMethodSpec CJS_Field::MethodSpecs[] = { {"browseForFileToSubmit", browseForFileToSubmit_static}, {"buttonGetCaption", buttonGetCaption_static}, {"buttonGetIcon", buttonGetIcon_static}, {"buttonImportIcon", buttonImportIcon_static}, {"buttonSetCaption", buttonSetCaption_static}, {"buttonSetIcon", buttonSetIcon_static}, {"checkThisBox", checkThisBox_static}, {"clearItems", clearItems_static}, {"defaultIsChecked", defaultIsChecked_static}, {"deleteItemAt", deleteItemAt_static}, {"getArray", getArray_static}, {"getItemAt", getItemAt_static}, {"getLock", getLock_static}, {"insertItemAt", insertItemAt_static}, {"isBoxChecked", isBoxChecked_static}, {"isDefaultChecked", isDefaultChecked_static}, {"setAction", setAction_static}, {"setFocus", setFocus_static}, {"setItems", setItems_static}, {"setLock", setLock_static}, {"signatureGetModifications", signatureGetModifications_static}, {"signatureGetSeedValue", signatureGetSeedValue_static}, {"signatureInfo", signatureInfo_static}, {"signatureSetSeedValue", signatureSetSeedValue_static}, {"signatureSign", signatureSign_static}, {"signatureValidate", signatureValidate_static}}; uint32_t CJS_Field::ObjDefnID = 0; const char CJS_Field::kName[] = "Field"; // static uint32_t CJS_Field::GetObjDefnID() { return ObjDefnID; } // static void CJS_Field::DefineJSObjects(CFXJS_Engine* pEngine) { ObjDefnID = pEngine->DefineObj(CJS_Field::kName, FXJSOBJTYPE_DYNAMIC, JSConstructor, JSDestructor); DefineProps(pEngine, ObjDefnID, PropertySpecs); DefineMethods(pEngine, ObjDefnID, MethodSpecs); } CJS_Field::CJS_Field(v8::Local pObject, CJS_Runtime* pRuntime) : CJS_Object(pObject, pRuntime) {} CJS_Field::~CJS_Field() = default; bool CJS_Field::AttachField(CJS_Document* pDocument, const WideString& csFieldName) { m_pJSDoc.Reset(pDocument); m_pFormFillEnv.Reset(pDocument->GetFormFillEnv()); m_bCanSet = m_pFormFillEnv->HasPermissions( pdfium::access_permissions::kFillForm | pdfium::access_permissions::kModifyAnnotation | pdfium::access_permissions::kModifyContent); CPDFSDK_InteractiveForm* pRDForm = m_pFormFillEnv->GetInteractiveForm(); CPDF_InteractiveForm* pForm = pRDForm->GetInteractiveForm(); WideString swFieldNameTemp = csFieldName; swFieldNameTemp.Replace(L"..", L"."); if (pForm->CountFields(swFieldNameTemp) <= 0) { absl::optional parsed_data = ParseFieldName(swFieldNameTemp); if (!parsed_data.has_value()) return false; m_FieldName = parsed_data.value().field_name; m_nFormControlIndex = parsed_data.value().control_index; return true; } m_FieldName = swFieldNameTemp; m_nFormControlIndex = -1; return true; } std::vector CJS_Field::GetFormFields() const { return GetFormFieldsForName(m_pFormFillEnv.Get(), m_FieldName); } CPDF_FormField* CJS_Field::GetFirstFormField() const { std::vector fields = GetFormFields(); return fields.empty() ? nullptr : fields[0]; } CPDF_FormControl* CJS_Field::GetSmartFieldControl(CPDF_FormField* pFormField) { if (!pFormField->CountControls() || m_nFormControlIndex >= pFormField->CountControls()) return nullptr; if (m_nFormControlIndex < 0) return pFormField->GetControl(0); return pFormField->GetControl(m_nFormControlIndex); } CJS_Result CJS_Field::get_alignment(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); switch (pFormControl->GetControlAlignment()) { case 0: return CJS_Result::Success(pRuntime->NewString("left")); case 1: return CJS_Result::Success(pRuntime->NewString("center")); case 2: return CJS_Result::Success(pRuntime->NewString("right")); } return CJS_Result::Success(pRuntime->NewString("")); } CJS_Result CJS_Field::set_alignment(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_border_style(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_Widget* pWidget = m_pFormFillEnv->GetInteractiveForm()->GetWidget( GetSmartFieldControl(pFormField)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); switch (pWidget->GetBorderStyle()) { case BorderStyle::kSolid: return CJS_Result::Success(pRuntime->NewString("solid")); case BorderStyle::kDash: return CJS_Result::Success(pRuntime->NewString("dashed")); case BorderStyle::kBeveled: return CJS_Result::Success(pRuntime->NewString("beveled")); case BorderStyle::kInset: return CJS_Result::Success(pRuntime->NewString("inset")); case BorderStyle::kUnderline: return CJS_Result::Success(pRuntime->NewString("underline")); } return CJS_Result::Success(pRuntime->NewString("")); } CJS_Result CJS_Field::set_border_style(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); ByteString byte_str = pRuntime->ToByteString(vp); if (m_bDelay) { AddDelay_String(FP_BORDERSTYLE, byte_str); } else { SetBorderStyle(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, byte_str); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_align_x(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_IconFit IconFit = pFormControl->GetIconFit(); CFX_PointF pos = IconFit.GetIconBottomLeftPosition(); return CJS_Result::Success(pRuntime->NewNumber(static_cast(pos.x))); } CJS_Result CJS_Field::set_button_align_x(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_align_y(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_IconFit IconFit = pFormControl->GetIconFit(); CFX_PointF pos = IconFit.GetIconBottomLeftPosition(); return CJS_Result::Success(pRuntime->NewNumber(static_cast(pos.y))); } CJS_Result CJS_Field::set_button_align_y(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_fit_bounds(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success( pRuntime->NewBoolean(pFormControl->GetIconFit().GetFittingBounds())); } CJS_Result CJS_Field::set_button_fit_bounds(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_position(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success( pRuntime->NewNumber(pFormControl->GetTextPosition())); } CJS_Result CJS_Field::set_button_position(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_scale_how(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success( pRuntime->NewBoolean(!pFormControl->GetIconFit().IsProportionalScale())); } CJS_Result CJS_Field::set_button_scale_how(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_button_scale_when(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_IconFit IconFit = pFormControl->GetIconFit(); CPDF_IconFit::ScaleMethod scale_method = IconFit.GetScaleMethod(); switch (scale_method) { case CPDF_IconFit::ScaleMethod::kAlways: case CPDF_IconFit::ScaleMethod::kBigger: case CPDF_IconFit::ScaleMethod::kNever: case CPDF_IconFit::ScaleMethod::kSmaller: return CJS_Result::Success( pRuntime->NewNumber(static_cast(scale_method))); } } CJS_Result CJS_Field::set_button_scale_when(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_calc_order_index(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrTextField(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDFSDK_InteractiveForm* pRDForm = m_pFormFillEnv->GetInteractiveForm(); CPDF_InteractiveForm* pForm = pRDForm->GetInteractiveForm(); return CJS_Result::Success( pRuntime->NewNumber(pForm->FindFieldInCalculationOrder(pFormField))); } CJS_Result CJS_Field::set_calc_order_index(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_char_limit(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success( pRuntime->NewNumber(static_cast(pFormField->GetMaxLen()))); } CJS_Result CJS_Field::set_char_limit(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_comb(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextComb))); } CJS_Result CJS_Field::set_comb(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_commit_on_sel_change(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrListBox(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); uint32_t dwFieldFlags = pFormField->GetFieldFlags(); return CJS_Result::Success(pRuntime->NewBoolean( !!(dwFieldFlags & pdfium::form_flags::kChoiceCommitOnSelChange))); } CJS_Result CJS_Field::set_commit_on_sel_change(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_current_value_indices(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrListBox(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); int count = pFormField->CountSelectedItems(); if (count <= 0) return CJS_Result::Success(pRuntime->NewNumber(-1)); if (count == 1) return CJS_Result::Success( pRuntime->NewNumber(pFormField->GetSelectedIndex(0))); v8::Local SelArray = pRuntime->NewArray(); for (int i = 0; i < count; i++) { pRuntime->PutArrayElement( SelArray, i, pRuntime->NewNumber(pFormField->GetSelectedIndex(i))); } if (SelArray.IsEmpty()) return CJS_Result::Success(pRuntime->NewArray()); return CJS_Result::Success(SelArray); } CJS_Result CJS_Field::set_current_value_indices(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); std::vector array; if (vp->IsNumber()) { array.push_back(pRuntime->ToInt32(vp)); } else if (fxv8::IsArray(vp)) { v8::Local SelArray = pRuntime->ToArray(vp); for (size_t i = 0; i < pRuntime->GetArrayLength(SelArray); i++) { array.push_back( pRuntime->ToInt32(pRuntime->GetArrayElement(SelArray, i))); } } if (m_bDelay) { AddDelay_WordArray(FP_CURRENTVALUEINDICES, array); } else { SetCurrentValueIndices(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, array); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_default_style(CJS_Runtime* pRuntime) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::set_default_style(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::get_default_value(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() == FormFieldType::kPushButton || pFormField->GetFieldType() == FormFieldType::kSignature) { return CJS_Result::Failure(JSMessage::kObjectTypeError); } return CJS_Result::Success( pRuntime->NewString(pFormField->GetDefaultValue().AsStringView())); } CJS_Result CJS_Field::set_default_value(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_do_not_scroll(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextDoNotScroll))); } CJS_Result CJS_Field::set_do_not_scroll(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_do_not_spell_check(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrTextField(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); uint32_t dwFieldFlags = pFormField->GetFieldFlags(); return CJS_Result::Success(pRuntime->NewBoolean( !!(dwFieldFlags & pdfium::form_flags::kTextDoNotSpellCheck))); } CJS_Result CJS_Field::set_do_not_spell_check(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_delay(CJS_Runtime* pRuntime) { return CJS_Result::Success(pRuntime->NewBoolean(m_bDelay)); } CJS_Result CJS_Field::set_delay(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); SetDelay(pRuntime->ToBoolean(vp)); return CJS_Result::Success(); } CJS_Result CJS_Field::get_display(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); CPDFSDK_Widget* pWidget = pForm->GetWidget(GetSmartFieldControl(pFormField)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); uint32_t dwFlag = pWidget->GetFlags(); if (pdfium::annotation_flags::kInvisible & dwFlag || pdfium::annotation_flags::kHidden & dwFlag) { return CJS_Result::Success(pRuntime->NewNumber(1)); } if (pdfium::annotation_flags::kPrint & dwFlag) { if (pdfium::annotation_flags::kNoView & dwFlag) return CJS_Result::Success(pRuntime->NewNumber(3)); return CJS_Result::Success(pRuntime->NewNumber(0)); } return CJS_Result::Success(pRuntime->NewNumber(2)); } CJS_Result CJS_Field::set_display(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (m_bDelay) { AddDelay_Int(FP_DISPLAY, pRuntime->ToInt32(vp)); } else { SetDisplay(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, pRuntime->ToInt32(vp)); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_doc(CJS_Runtime* pRuntime) { return CJS_Result::Success(m_pJSDoc->ToV8Object()); } CJS_Result CJS_Field::set_doc(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::get_editable(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kComboBox) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kChoiceEdit))); } CJS_Result CJS_Field::set_editable(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_export_values(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsCheckBoxOrRadioButton(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); v8::Local ExportValuesArray = pRuntime->NewArray(); if (m_nFormControlIndex < 0) { for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) { CPDF_FormControl* pFormControl = pFormField->GetControl(i); pRuntime->PutArrayElement( ExportValuesArray, i, pRuntime->NewString(pFormControl->GetExportValue().AsStringView())); } } else { if (m_nFormControlIndex >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); CPDF_FormControl* pFormControl = pFormField->GetControl(m_nFormControlIndex); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); pRuntime->PutArrayElement( ExportValuesArray, 0, pRuntime->NewString(pFormControl->GetExportValue().AsStringView())); } return CJS_Result::Success(ExportValuesArray); } CJS_Result CJS_Field::set_export_values(CJS_Runtime* pRuntime, v8::Local vp) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsCheckBoxOrRadioButton(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (!fxv8::IsArray(vp)) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_file_select(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextFileSelect))); } CJS_Result CJS_Field::set_file_select(CJS_Runtime* pRuntime, v8::Local vp) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_fill_color(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CFX_Color color = GetFormControlColor(pFormControl, pdfium::appearance::kBG); v8::Local array = CJS_Color::ConvertPWLColorToArray(pRuntime, color); if (array.IsEmpty()) return CJS_Result::Success(pRuntime->NewArray()); return CJS_Result::Success(array); } CJS_Result CJS_Field::set_fill_color(CJS_Runtime* pRuntime, v8::Local vp) { std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (!fxv8::IsArray(vp)) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_hidden(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); CPDFSDK_Widget* pWidget = pForm->GetWidget(GetSmartFieldControl(pFormField)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); uint32_t dwFlags = pWidget->GetFlags(); return CJS_Result::Success( pRuntime->NewBoolean(pdfium::annotation_flags::kInvisible & dwFlags || pdfium::annotation_flags::kHidden & dwFlags)); } CJS_Result CJS_Field::set_hidden(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (m_bDelay) { AddDelay_Bool(FP_HIDDEN, pRuntime->ToBoolean(vp)); } else { SetHidden(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, pRuntime->ToBoolean(vp)); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_highlight(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); int eHM = pFormControl->GetHighlightingMode(); switch (eHM) { case CPDF_FormControl::kNone: return CJS_Result::Success(pRuntime->NewString("none")); case CPDF_FormControl::kPush: return CJS_Result::Success(pRuntime->NewString("push")); case CPDF_FormControl::kInvert: return CJS_Result::Success(pRuntime->NewString("invert")); case CPDF_FormControl::kOutline: return CJS_Result::Success(pRuntime->NewString("outline")); case CPDF_FormControl::kToggle: return CJS_Result::Success(pRuntime->NewString("toggle")); } return CJS_Result::Success(); } CJS_Result CJS_Field::set_highlight(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_line_width(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); if (!pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormField->GetControl(0)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewNumber(pWidget->GetBorderWidth())); } CJS_Result CJS_Field::set_line_width(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (m_bDelay) { AddDelay_Int(FP_LINEWIDTH, pRuntime->ToInt32(vp)); } else { SetLineWidth(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, pRuntime->ToInt32(vp)); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_multiline(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextMultiline))); } CJS_Result CJS_Field::set_multiline(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_multiple_selection(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kListBox) return CJS_Result::Failure(JSMessage::kObjectTypeError); uint32_t dwFieldFlags = pFormField->GetFieldFlags(); return CJS_Result::Success(pRuntime->NewBoolean( !!(dwFieldFlags & pdfium::form_flags::kChoiceMultiSelect))); } CJS_Result CJS_Field::set_multiple_selection(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_name(CJS_Runtime* pRuntime) { std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewString(m_FieldName.AsStringView())); } CJS_Result CJS_Field::set_name(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::get_num_items(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrListBox(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewNumber(pFormField->CountOptions())); } CJS_Result CJS_Field::set_num_items(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::get_page(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); std::vector> widgets; m_pFormFillEnv->GetInteractiveForm()->GetWidgets(pFormField, &widgets); if (widgets.empty()) return CJS_Result::Success(pRuntime->NewNumber(-1)); v8::Local PageArray = pRuntime->NewArray(); int i = 0; for (const auto& pWidget : widgets) { if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); pRuntime->PutArrayElement( PageArray, i, pRuntime->NewNumber(pWidget->GetPageView()->GetPageIndex())); ++i; } return CJS_Result::Success(PageArray); } CJS_Result CJS_Field::set_page(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kReadOnlyError); } CJS_Result CJS_Field::get_password(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextPassword))); } CJS_Result CJS_Field::set_password(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_print(CJS_Runtime* pRuntime) { CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_Widget* pWidget = pForm->GetWidget(GetSmartFieldControl(pFormField)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pWidget->GetFlags() & pdfium::annotation_flags::kPrint))); } CJS_Result CJS_Field::set_print(CJS_Runtime* pRuntime, v8::Local vp) { CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); for (CPDF_FormField* pFormField : FieldArray) { if (m_nFormControlIndex < 0) { bool bSet = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; ++i) { if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormField->GetControl(i))) { uint32_t dwFlags = pWidget->GetFlags(); if (pRuntime->ToBoolean(vp)) dwFlags |= pdfium::annotation_flags::kPrint; else dwFlags &= ~pdfium::annotation_flags::kPrint; if (dwFlags != pWidget->GetFlags()) { pWidget->SetFlags(dwFlags); bSet = true; } } } if (bSet) UpdateFormField(m_pFormFillEnv.Get(), pFormField, false); continue; } if (m_nFormControlIndex >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); if (CPDF_FormControl* pFormControl = pFormField->GetControl(m_nFormControlIndex)) { if (CPDFSDK_Widget* pWidget = pForm->GetWidget(pFormControl)) { uint32_t dwFlags = pWidget->GetFlags(); if (pRuntime->ToBoolean(vp)) dwFlags |= pdfium::annotation_flags::kPrint; else dwFlags &= ~pdfium::annotation_flags::kPrint; if (dwFlags != pWidget->GetFlags()) { pWidget->SetFlags(dwFlags); UpdateFormControl(m_pFormFillEnv.Get(), pFormField->GetControl(m_nFormControlIndex), false); } } } } return CJS_Result::Success(); } CJS_Result CJS_Field::get_radios_in_unison(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kRadioButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); uint32_t dwFieldFlags = pFormField->GetFieldFlags(); return CJS_Result::Success(pRuntime->NewBoolean( !!(dwFieldFlags & pdfium::form_flags::kButtonRadiosInUnison))); } CJS_Result CJS_Field::set_radios_in_unison(CJS_Runtime* pRuntime, v8::Local vp) { std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_readonly(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kReadOnly))); } CJS_Result CJS_Field::set_readonly(CJS_Runtime* pRuntime, v8::Local vp) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); const bool bReadOnly = pRuntime->ToBoolean(vp); const uint32_t dwFlags = pFormField->GetFieldFlags(); const uint32_t dwNewFlags = bReadOnly ? (dwFlags | pdfium::form_flags::kReadOnly) : (dwFlags & ~pdfium::form_flags::kReadOnly); if (dwNewFlags != dwFlags) pFormField->SetFieldFlags(dwNewFlags); return CJS_Result::Success(); } CJS_Result CJS_Field::get_rect(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); CPDFSDK_Widget* pWidget = pForm->GetWidget(GetSmartFieldControl(pFormField)); if (!pWidget) return CJS_Result::Failure(JSMessage::kBadObjectError); CFX_FloatRect crRect = pWidget->GetRect(); v8::Local rcArray = pRuntime->NewArray(); pRuntime->PutArrayElement( rcArray, 0, pRuntime->NewNumber(static_cast(crRect.left))); pRuntime->PutArrayElement( rcArray, 1, pRuntime->NewNumber(static_cast(crRect.top))); pRuntime->PutArrayElement( rcArray, 2, pRuntime->NewNumber(static_cast(crRect.right))); pRuntime->PutArrayElement( rcArray, 3, pRuntime->NewNumber(static_cast(crRect.bottom))); return CJS_Result::Success(rcArray); } CJS_Result CJS_Field::set_rect(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (!fxv8::IsArray(vp)) return CJS_Result::Failure(JSMessage::kValueError); v8::Local rcArray = pRuntime->ToArray(vp); if (pRuntime->GetArrayLength(rcArray) < 4) return CJS_Result::Failure(JSMessage::kValueError); float f0 = static_cast( pRuntime->ToInt32(pRuntime->GetArrayElement(rcArray, 0))); float f1 = static_cast( pRuntime->ToInt32(pRuntime->GetArrayElement(rcArray, 1))); float f2 = static_cast( pRuntime->ToInt32(pRuntime->GetArrayElement(rcArray, 2))); float f3 = static_cast( pRuntime->ToInt32(pRuntime->GetArrayElement(rcArray, 3))); CFX_FloatRect crRect(f0, f1, f2, f3); if (m_bDelay) { AddDelay_Rect(FP_RECT, crRect); } else { SetRect(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, crRect); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_required(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() == FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kRequired))); } CJS_Result CJS_Field::set_required(CJS_Runtime* pRuntime, v8::Local vp) { std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_rich_text(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kTextField) return CJS_Result::Failure(JSMessage::kObjectTypeError); return CJS_Result::Success(pRuntime->NewBoolean( !!(pFormField->GetFieldFlags() & pdfium::form_flags::kTextRichText))); } CJS_Result CJS_Field::set_rich_text(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_rich_value(CJS_Runtime* pRuntime) { return CJS_Result::Success(); } CJS_Result CJS_Field::set_rich_value(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Success(); } CJS_Result CJS_Field::get_rotation(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewNumber(pFormControl->GetRotation())); } CJS_Result CJS_Field::set_rotation(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_source(CJS_Runtime* pRuntime) { return CJS_Result::Success(); } CJS_Result CJS_Field::set_source(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Success(); } CJS_Result CJS_Field::get_stroke_color(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CFX_Color color = GetFormControlColor(pFormControl, pdfium::appearance::kBC); v8::Local array = CJS_Color::ConvertPWLColorToArray(pRuntime, color); if (array.IsEmpty()) return CJS_Result::Success(pRuntime->NewArray()); return CJS_Result::Success(array); } CJS_Result CJS_Field::set_stroke_color(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (!fxv8::IsArray(vp)) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_style(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsCheckBoxOrRadioButton(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); wchar_t selector = GetSelectorFromCaptionForFieldType( pFormControl->GetNormalCaption(), pFormControl->GetType()); ByteString csBCaption; switch (selector) { case kCircleSelector: csBCaption = "circle"; break; case kCrossSelector: csBCaption = "cross"; break; case kDiamondSelector: csBCaption = "diamond"; break; case kSquareSelector: csBCaption = "square"; break; case kStarSelector: csBCaption = "star"; break; case kCheckSelector: default: csBCaption = "check"; break; } return CJS_Result::Success(pRuntime->NewString(csBCaption.AsStringView())); } CJS_Result CJS_Field::set_style(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_submit_name(CJS_Runtime* pRuntime) { return CJS_Result::Success(); } CJS_Result CJS_Field::set_submit_name(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Success(); } CJS_Result CJS_Field::get_text_color(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_DefaultAppearance FieldAppearance = pFormControl->GetDefaultAppearance(); absl::optional maybe_type_argb_pair = FieldAppearance.GetColorARGB(); CFX_Color crRet; if (maybe_type_argb_pair.has_value() && maybe_type_argb_pair.value().color_type != CFX_Color::Type::kTransparent) { int32_t a; int32_t r; int32_t g; int32_t b; std::tie(a, r, g, b) = ArgbDecode(maybe_type_argb_pair.value().argb); crRet = CFX_Color(CFX_Color::Type::kRGB, r / 255.0f, g / 255.0f, b / 255.0f); } v8::Local array = CJS_Color::ConvertPWLColorToArray(pRuntime, crRet); if (array.IsEmpty()) return CJS_Result::Success(pRuntime->NewArray()); return CJS_Result::Success(array); } CJS_Result CJS_Field::set_text_color(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (!fxv8::IsArray(vp)) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_text_font(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); FormFieldType fieldType = pFormField->GetFieldType(); if (fieldType != FormFieldType::kPushButton && fieldType != FormFieldType::kComboBox && fieldType != FormFieldType::kListBox && fieldType != FormFieldType::kTextField) { return CJS_Result::Failure(JSMessage::kObjectTypeError); } absl::optional wsFontName = pFormControl->GetDefaultControlFontName(); if (!wsFontName.has_value()) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success( pRuntime->NewString(wsFontName.value().AsStringView())); } CJS_Result CJS_Field::set_text_font(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (pRuntime->ToByteString(vp).IsEmpty()) return CJS_Result::Failure(JSMessage::kValueError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_text_size(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); float fFontSize; CPDF_DefaultAppearance FieldAppearance = pFormControl->GetDefaultAppearance(); FieldAppearance.GetFont(&fFontSize); return CJS_Result::Success(pRuntime->NewNumber(static_cast(fFontSize))); } CJS_Result CJS_Field::set_text_size(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_type(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); switch (pFormField->GetFieldType()) { case FormFieldType::kUnknown: return CJS_Result::Success(pRuntime->NewString("unknown")); case FormFieldType::kPushButton: return CJS_Result::Success(pRuntime->NewString("button")); case FormFieldType::kCheckBox: return CJS_Result::Success(pRuntime->NewString("checkbox")); case FormFieldType::kRadioButton: return CJS_Result::Success(pRuntime->NewString("radiobutton")); case FormFieldType::kComboBox: return CJS_Result::Success(pRuntime->NewString("combobox")); case FormFieldType::kListBox: return CJS_Result::Success(pRuntime->NewString("listbox")); case FormFieldType::kTextField: return CJS_Result::Success(pRuntime->NewString("text")); case FormFieldType::kSignature: return CJS_Result::Success(pRuntime->NewString("signature")); #ifdef PDF_ENABLE_XFA default: return CJS_Result::Success(pRuntime->NewString("unknown")); #endif } } CJS_Result CJS_Field::set_type(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::get_user_name(CJS_Runtime* pRuntime) { DCHECK(m_pFormFillEnv); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success( pRuntime->NewString(pFormField->GetAlternateName().AsStringView())); } CJS_Result CJS_Field::set_user_name(CJS_Runtime* pRuntime, v8::Local vp) { DCHECK(m_pFormFillEnv); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); return CJS_Result::Success(); } CJS_Result CJS_Field::get_value(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); v8::Local ret; switch (pFormField->GetFieldType()) { case FormFieldType::kPushButton: return CJS_Result::Failure(JSMessage::kObjectTypeError); case FormFieldType::kComboBox: case FormFieldType::kTextField: ret = pRuntime->NewString(pFormField->GetValue().AsStringView()); break; case FormFieldType::kListBox: { if (pFormField->CountSelectedItems() > 1) { v8::Local ValueArray = pRuntime->NewArray(); v8::Local ElementValue; int iIndex; for (int i = 0, sz = pFormField->CountSelectedItems(); i < sz; i++) { iIndex = pFormField->GetSelectedIndex(i); ElementValue = pRuntime->NewString( pFormField->GetOptionValue(iIndex).AsStringView()); if (pRuntime->ToWideString(ElementValue).IsEmpty()) { ElementValue = pRuntime->NewString( pFormField->GetOptionLabel(iIndex).AsStringView()); } pRuntime->PutArrayElement(ValueArray, i, ElementValue); } ret = ValueArray; } else { ret = pRuntime->NewString(pFormField->GetValue().AsStringView()); } break; } case FormFieldType::kCheckBox: case FormFieldType::kRadioButton: { bool bFind = false; for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) { if (pFormField->GetControl(i)->IsChecked()) { ret = pRuntime->NewString( pFormField->GetControl(i)->GetExportValue().AsStringView()); bFind = true; break; } } if (!bFind) ret = pRuntime->NewString("Off"); break; } default: ret = pRuntime->NewString(pFormField->GetValue().AsStringView()); break; } return CJS_Result::Success(pRuntime->MaybeCoerceToNumber(ret)); } CJS_Result CJS_Field::set_value(CJS_Runtime* pRuntime, v8::Local vp) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); std::vector strArray; if (fxv8::IsArray(vp)) { v8::Local ValueArray = pRuntime->ToArray(vp); for (size_t i = 0; i < pRuntime->GetArrayLength(ValueArray); i++) { strArray.push_back( pRuntime->ToWideString(pRuntime->GetArrayElement(ValueArray, i))); } } else { strArray.push_back(pRuntime->ToWideString(vp)); } if (m_bDelay) { AddDelay_WideStringArray(FP_VALUE, strArray); } else { SetFieldValue(m_pFormFillEnv.Get(), m_FieldName, m_nFormControlIndex, strArray); } return CJS_Result::Success(); } CJS_Result CJS_Field::get_value_as_string(CJS_Runtime* pRuntime) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() == FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); if (pFormField->GetFieldType() == FormFieldType::kCheckBox) { if (!pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kBadObjectError); return CJS_Result::Success(pRuntime->NewString( pFormField->GetControl(0)->IsChecked() ? L"Yes" : L"Off")); } if (pFormField->GetFieldType() == FormFieldType::kRadioButton && !(pFormField->GetFieldFlags() & pdfium::form_flags::kButtonRadiosInUnison)) { for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) { if (pFormField->GetControl(i)->IsChecked()) { return CJS_Result::Success(pRuntime->NewString( pFormField->GetControl(i)->GetExportValue().AsStringView())); } } return CJS_Result::Success(pRuntime->NewString("Off")); } if (pFormField->GetFieldType() == FormFieldType::kListBox && (pFormField->CountSelectedItems() > 1)) { return CJS_Result::Success(pRuntime->NewString("")); } return CJS_Result::Success( pRuntime->NewString(pFormField->GetValue().AsStringView())); } CJS_Result CJS_Field::set_value_as_string(CJS_Runtime* pRuntime, v8::Local vp) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::browseForFileToSubmit( CJS_Runtime* pRuntime, const std::vector>& params) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if ((pFormField->GetFieldFlags() & pdfium::form_flags::kTextFileSelect) && (pFormField->GetFieldType() == FormFieldType::kTextField)) { WideString wsFileName = m_pFormFillEnv->JS_fieldBrowse(); if (!wsFileName.IsEmpty()) { pFormField->SetValue(wsFileName, NotificationOption::kDoNotNotify); UpdateFormField(m_pFormFillEnv.Get(), pFormField, true); } return CJS_Result::Success(); } return CJS_Result::Failure(JSMessage::kObjectTypeError); } CJS_Result CJS_Field::buttonGetCaption( CJS_Runtime* pRuntime, const std::vector>& params) { int nface = 0; if (params.size() >= 1) nface = pRuntime->ToInt32(params[0]); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); if (nface == 0) { return CJS_Result::Success( pRuntime->NewString(pFormControl->GetNormalCaption().AsStringView())); } if (nface == 1) { return CJS_Result::Success( pRuntime->NewString(pFormControl->GetDownCaption().AsStringView())); } if (nface == 2) { return CJS_Result::Success( pRuntime->NewString(pFormControl->GetRolloverCaption().AsStringView())); } return CJS_Result::Failure(JSMessage::kValueError); } CJS_Result CJS_Field::buttonGetIcon( CJS_Runtime* pRuntime, const std::vector>& params) { if (params.size() >= 1) { int nFace = pRuntime->ToInt32(params[0]); if (nFace < 0 || nFace > 2) return CJS_Result::Failure(JSMessage::kValueError); } CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (pFormField->GetFieldType() != FormFieldType::kPushButton) return CJS_Result::Failure(JSMessage::kObjectTypeError); CPDF_FormControl* pFormControl = GetSmartFieldControl(pFormField); if (!pFormControl) return CJS_Result::Failure(JSMessage::kBadObjectError); v8::Local pObj = pRuntime->NewFXJSBoundObject( CJS_Icon::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pObj.IsEmpty()) return CJS_Result::Failure(JSMessage::kBadObjectError); auto* pJS_Icon = static_cast( CFXJS_Engine::GetObjectPrivate(pRuntime->GetIsolate(), pObj)); return pJS_Icon ? CJS_Result::Success(pJS_Icon->ToV8Object()) : CJS_Result::Failure(JSMessage::kBadObjectError); } CJS_Result CJS_Field::buttonImportIcon( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::buttonSetCaption( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::buttonSetIcon( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::checkThisBox( CJS_Runtime* pRuntime, const std::vector>& params) { const size_t nSize = params.size(); if (nSize == 0) return CJS_Result::Failure(JSMessage::kParamError); if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); int nWidget = pRuntime->ToInt32(params[0]); bool bCheckit = true; if (nSize >= 2) bCheckit = pRuntime->ToBoolean(params[1]); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsCheckBoxOrRadioButton(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); if (nWidget < 0 || nWidget >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); // TODO(weili): Check whether anything special needed for radio button. // (When pFormField->GetFieldType() == FormFieldType::kRadioButton.) pFormField->CheckControl(nWidget, bCheckit, NotificationOption::kNotify); UpdateFormField(m_pFormFillEnv.Get(), pFormField, true); return CJS_Result::Success(); } CJS_Result CJS_Field::clearItems( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::defaultIsChecked( CJS_Runtime* pRuntime, const std::vector>& params) { if (!m_bCanSet) return CJS_Result::Failure(JSMessage::kReadOnlyError); if (params.empty()) return CJS_Result::Failure(JSMessage::kParamError); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); int nWidget = pRuntime->ToInt32(params[0]); if (nWidget < 0 || nWidget >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); return CJS_Result::Success( pRuntime->NewBoolean(IsCheckBoxOrRadioButton(pFormField))); } CJS_Result CJS_Field::deleteItemAt( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::getArray( CJS_Runtime* pRuntime, const std::vector>& params) { std::vector FieldArray = GetFormFields(); if (FieldArray.empty()) return CJS_Result::Failure(JSMessage::kBadObjectError); std::vector> swSort; for (CPDF_FormField* pFormField : FieldArray) { swSort.push_back(std::make_unique(pFormField->GetFullName())); } std::sort(swSort.begin(), swSort.end(), [](const std::unique_ptr& p1, const std::unique_ptr& p2) { return *p1 < *p2; }); v8::Local FormFieldArray = pRuntime->NewArray(); int j = 0; for (const auto& pStr : swSort) { v8::Local pObj = pRuntime->NewFXJSBoundObject( CJS_Field::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC); if (pObj.IsEmpty()) return CJS_Result::Failure(JSMessage::kBadObjectError); auto* pJSField = static_cast( CFXJS_Engine::GetObjectPrivate(pRuntime->GetIsolate(), pObj)); pJSField->AttachField(m_pJSDoc.Get(), *pStr); pRuntime->PutArrayElement(FormFieldArray, j++, pJSField ? v8::Local(pJSField->ToV8Object()) : v8::Local()); } return CJS_Result::Success(FormFieldArray); } CJS_Result CJS_Field::getItemAt( CJS_Runtime* pRuntime, const std::vector>& params) { const size_t nSize = params.size(); int nIdx = -1; if (nSize >= 1) nIdx = pRuntime->ToInt32(params[0]); bool bExport = true; if (nSize >= 2) bExport = pRuntime->ToBoolean(params[1]); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (!IsComboBoxOrListBox(pFormField)) return CJS_Result::Failure(JSMessage::kObjectTypeError); if (nIdx == -1 || nIdx > pFormField->CountOptions()) nIdx = pFormField->CountOptions() - 1; if (!bExport) { return CJS_Result::Success( pRuntime->NewString(pFormField->GetOptionLabel(nIdx).AsStringView())); } WideString strval = pFormField->GetOptionValue(nIdx); if (strval.IsEmpty()) { return CJS_Result::Success( pRuntime->NewString(pFormField->GetOptionLabel(nIdx).AsStringView())); } return CJS_Result::Success(pRuntime->NewString(strval.AsStringView())); } CJS_Result CJS_Field::getLock(CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::insertItemAt( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::isBoxChecked( CJS_Runtime* pRuntime, const std::vector>& params) { int nIndex = -1; if (params.size() >= 1) nIndex = pRuntime->ToInt32(params[0]); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (nIndex < 0 || nIndex >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); return CJS_Result::Success( pRuntime->NewBoolean((IsCheckBoxOrRadioButton(pFormField) && pFormField->GetControl(nIndex)->IsChecked() != 0))); } CJS_Result CJS_Field::isDefaultChecked( CJS_Runtime* pRuntime, const std::vector>& params) { int nIndex = -1; if (params.size() >= 1) nIndex = pRuntime->ToInt32(params[0]); CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); if (nIndex < 0 || nIndex >= pFormField->CountControls()) return CJS_Result::Failure(JSMessage::kValueError); return CJS_Result::Success(pRuntime->NewBoolean( (IsCheckBoxOrRadioButton(pFormField) && pFormField->GetControl(nIndex)->IsDefaultChecked() != 0))); } CJS_Result CJS_Field::setAction( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::setFocus( CJS_Runtime* pRuntime, const std::vector>& params) { CPDF_FormField* pFormField = GetFirstFormField(); if (!pFormField) return CJS_Result::Failure(JSMessage::kBadObjectError); int32_t nCount = pFormField->CountControls(); if (nCount < 1) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_InteractiveForm* pForm = m_pFormFillEnv->GetInteractiveForm(); CPDFSDK_Widget* pWidget = nullptr; if (nCount == 1) { pWidget = pForm->GetWidget(pFormField->GetControl(0)); } else { IPDF_Page* pPage = m_pFormFillEnv->GetCurrentPage(); if (!pPage) return CJS_Result::Failure(JSMessage::kBadObjectError); CPDFSDK_PageView* pCurPageView = m_pFormFillEnv->GetOrCreatePageView(pPage); for (int32_t i = 0; i < nCount; i++) { if (CPDFSDK_Widget* pTempWidget = pForm->GetWidget(pFormField->GetControl(i))) { if (pTempWidget->GetPDFPage() == pCurPageView->GetPDFPage()) { pWidget = pTempWidget; break; } } } } if (pWidget) { ObservedPtr pObserved(pWidget); m_pFormFillEnv->SetFocusAnnot(pObserved); } return CJS_Result::Success(); } CJS_Result CJS_Field::setItems( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Success(); } CJS_Result CJS_Field::setLock(CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureGetModifications( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureGetSeedValue( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureInfo( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureSetSeedValue( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureSign( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } CJS_Result CJS_Field::signatureValidate( CJS_Runtime* pRuntime, const std::vector>& params) { return CJS_Result::Failure(JSMessage::kNotSupportedError); } void CJS_Field::SetDelay(bool bDelay) { m_bDelay = bDelay; if (m_bDelay) return; if (m_pJSDoc) m_pJSDoc->DoFieldDelay(m_FieldName, m_nFormControlIndex); } void CJS_Field::AddDelay_Int(FIELD_PROP prop, int32_t n) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->num = n; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::AddDelay_Bool(FIELD_PROP prop, bool b) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->b = b; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::AddDelay_String(FIELD_PROP prop, const ByteString& str) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->bytestring = str; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::AddDelay_Rect(FIELD_PROP prop, const CFX_FloatRect& rect) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->rect = rect; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::AddDelay_WordArray(FIELD_PROP prop, const std::vector& array) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->wordarray = array; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::AddDelay_WideStringArray(FIELD_PROP prop, const std::vector& array) { auto pNewData = std::make_unique(prop, m_nFormControlIndex, m_FieldName); pNewData->widestringarray = array; m_pJSDoc->AddDelayData(std::move(pNewData)); } void CJS_Field::DoDelay(CPDFSDK_FormFillEnvironment* pFormFillEnv, CJS_DelayData* pData) { DCHECK(pFormFillEnv); switch (pData->eProp) { case FP_BORDERSTYLE: SetBorderStyle(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->bytestring); break; case FP_CURRENTVALUEINDICES: SetCurrentValueIndices(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->wordarray); break; case FP_DISPLAY: SetDisplay(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->num); break; case FP_HIDDEN: SetHidden(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->b); break; case FP_LINEWIDTH: SetLineWidth(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->num); break; case FP_RECT: SetRect(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->rect); break; case FP_VALUE: SetFieldValue(pFormFillEnv, pData->sFieldName, pData->nControlIndex, pData->widestringarray); break; } }