| // 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 "core/fpdfdoc/cpdf_formcontrol.h" |
| |
| #include <array> |
| #include <iterator> |
| #include <utility> |
| |
| #include "constants/form_fields.h" |
| #include "core/fpdfapi/font/cpdf_font.h" |
| #include "core/fpdfapi/page/cpdf_docpagedata.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/cpdf_name.h" |
| #include "core/fpdfapi/parser/cpdf_stream.h" |
| #include "core/fpdfapi/parser/fpdf_parser_decode.h" |
| #include "core/fpdfapi/parser/fpdf_parser_utility.h" |
| #include "core/fpdfdoc/cpdf_interactiveform.h" |
| #include "core/fxcrt/check.h" |
| |
| namespace { |
| |
| constexpr std::array<char, 5> kHighlightModes = {{'N', 'I', 'O', 'P', 'T'}}; |
| |
| // Order of |kHighlightModes| must match order of HighlightingMode enum. |
| static_assert(kHighlightModes[CPDF_FormControl::kNone] == 'N', |
| "HighlightingMode mismatch"); |
| static_assert(kHighlightModes[CPDF_FormControl::kInvert] == 'I', |
| "HighlightingMode mismatch"); |
| static_assert(kHighlightModes[CPDF_FormControl::kOutline] == 'O', |
| "HighlightingMode mismatch"); |
| static_assert(kHighlightModes[CPDF_FormControl::kPush] == 'P', |
| "HighlightingMode mismatch"); |
| static_assert(kHighlightModes[CPDF_FormControl::kToggle] == 'T', |
| "HighlightingMode mismatch"); |
| |
| } // namespace |
| |
| CPDF_FormControl::CPDF_FormControl(CPDF_FormField* pField, |
| RetainPtr<CPDF_Dictionary> pWidgetDict, |
| CPDF_InteractiveForm* pForm) |
| : field_(pField), widget_dict_(std::move(pWidgetDict)), form_(pForm) { |
| DCHECK(widget_dict_); |
| } |
| |
| CPDF_FormControl::~CPDF_FormControl() = default; |
| |
| CFX_FloatRect CPDF_FormControl::GetRect() const { |
| return widget_dict_->GetRectFor("Rect"); |
| } |
| |
| ByteString CPDF_FormControl::GetOnStateName() const { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| RetainPtr<const CPDF_Dictionary> pAP = widget_dict_->GetDictFor("AP"); |
| if (!pAP) { |
| return ByteString(); |
| } |
| |
| RetainPtr<const CPDF_Dictionary> pN = pAP->GetDictFor("N"); |
| if (!pN) { |
| return ByteString(); |
| } |
| |
| CPDF_DictionaryLocker locker(pN); |
| for (const auto& it : locker) { |
| if (it.first != "Off") { |
| return it.first; |
| } |
| } |
| return ByteString(); |
| } |
| |
| ByteString CPDF_FormControl::GetCheckedAPState() const { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| ByteString csOn = GetOnStateName(); |
| if (ToArray(field_->GetFieldAttr("Opt"))) { |
| csOn = ByteString::FormatInteger(field_->GetControlIndex(this)); |
| } |
| if (csOn.IsEmpty()) { |
| csOn = "Yes"; |
| } |
| return csOn; |
| } |
| |
| WideString CPDF_FormControl::GetExportValue() const { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| ByteString csOn = GetOnStateName(); |
| RetainPtr<const CPDF_Array> pArray = ToArray(field_->GetFieldAttr("Opt")); |
| if (pArray) { |
| csOn = pArray->GetByteStringAt(field_->GetControlIndex(this)); |
| } |
| if (csOn.IsEmpty()) { |
| csOn = "Yes"; |
| } |
| return PDF_DecodeText(csOn.unsigned_span()); |
| } |
| |
| bool CPDF_FormControl::IsChecked() const { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| ByteString csOn = GetOnStateName(); |
| ByteString csAS = widget_dict_->GetByteStringFor("AS"); |
| return csAS == csOn; |
| } |
| |
| bool CPDF_FormControl::IsDefaultChecked() const { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| RetainPtr<const CPDF_Object> pDV = field_->GetFieldAttr("DV"); |
| if (!pDV) { |
| return false; |
| } |
| |
| ByteString csDV = pDV->GetString(); |
| ByteString csOn = GetOnStateName(); |
| return (csDV == csOn); |
| } |
| |
| void CPDF_FormControl::CheckControl(bool bChecked) { |
| DCHECK(GetType() == CPDF_FormField::kCheckBox || |
| GetType() == CPDF_FormField::kRadioButton); |
| ByteString csOldAS = widget_dict_->GetByteStringFor("AS", "Off"); |
| ByteString csAS = "Off"; |
| if (bChecked) { |
| csAS = GetOnStateName(); |
| } |
| if (csOldAS == csAS) { |
| return; |
| } |
| widget_dict_->SetNewFor<CPDF_Name>("AS", csAS); |
| } |
| |
| CPDF_FormControl::HighlightingMode CPDF_FormControl::GetHighlightingMode() |
| const { |
| ByteString csH = widget_dict_->GetByteStringFor("H", "I"); |
| for (size_t i = 0; i < std::size(kHighlightModes); ++i) { |
| // TODO(tsepez): disambiguate string ctors. |
| if (csH == ByteStringView(kHighlightModes[i])) { |
| return static_cast<HighlightingMode>(i); |
| } |
| } |
| return kInvert; |
| } |
| |
| CPDF_ApSettings CPDF_FormControl::GetMK() const { |
| return CPDF_ApSettings(widget_dict_->GetMutableDictFor("MK")); |
| } |
| |
| bool CPDF_FormControl::HasMKEntry(ByteStringView entry) const { |
| return GetMK().HasMKEntry(entry); |
| } |
| |
| int CPDF_FormControl::GetRotation() const { |
| return GetMK().GetRotation(); |
| } |
| |
| CFX_Color::TypeAndARGB CPDF_FormControl::GetColorARGB(ByteStringView entry) { |
| return GetMK().GetColorARGB(entry); |
| } |
| |
| float CPDF_FormControl::GetOriginalColorComponent(int index, |
| ByteStringView entry) { |
| return GetMK().GetOriginalColorComponent(index, entry); |
| } |
| |
| CFX_Color CPDF_FormControl::GetOriginalColor(ByteStringView entry) { |
| return GetMK().GetOriginalColor(entry); |
| } |
| |
| WideString CPDF_FormControl::GetCaption(ByteStringView entry) const { |
| return GetMK().GetCaption(entry); |
| } |
| |
| RetainPtr<CPDF_Stream> CPDF_FormControl::GetIcon(ByteStringView entry) { |
| return GetMK().GetIcon(entry); |
| } |
| |
| CPDF_IconFit CPDF_FormControl::GetIconFit() const { |
| return GetMK().GetIconFit(); |
| } |
| |
| int CPDF_FormControl::GetTextPosition() const { |
| return GetMK().GetTextPosition(); |
| } |
| |
| CPDF_DefaultAppearance CPDF_FormControl::GetDefaultAppearance() const { |
| if (widget_dict_->KeyExist(pdfium::form_fields::kDA)) { |
| return CPDF_DefaultAppearance( |
| widget_dict_->GetByteStringFor(pdfium::form_fields::kDA)); |
| } |
| RetainPtr<const CPDF_Object> pObj = |
| field_->GetFieldAttr(pdfium::form_fields::kDA); |
| if (pObj) { |
| return CPDF_DefaultAppearance(pObj->GetString()); |
| } |
| |
| return form_->GetDefaultAppearance(); |
| } |
| |
| std::optional<WideString> CPDF_FormControl::GetDefaultControlFontName() const { |
| RetainPtr<CPDF_Font> font = GetDefaultControlFont(); |
| if (!font) { |
| return std::nullopt; |
| } |
| |
| return WideString::FromDefANSI(font->GetBaseFontName().AsStringView()); |
| } |
| |
| RetainPtr<CPDF_Font> CPDF_FormControl::GetDefaultControlFont() const { |
| CPDF_DefaultAppearance default_appearance = GetDefaultAppearance(); |
| auto maybe_font_name_and_size = default_appearance.GetFont(); |
| if (!maybe_font_name_and_size.has_value() || |
| maybe_font_name_and_size.value().name.IsEmpty()) { |
| return nullptr; |
| } |
| |
| const ByteString& font_name = maybe_font_name_and_size.value().name; |
| RetainPtr<CPDF_Dictionary> pDRDict = ToDictionary( |
| CPDF_FormField::GetMutableFieldAttrForDict(widget_dict_.Get(), "DR")); |
| if (pDRDict) { |
| RetainPtr<CPDF_Dictionary> fonts = pDRDict->GetMutableDictFor("Font"); |
| if (ValidateFontResourceDict(fonts.Get())) { |
| RetainPtr<CPDF_Dictionary> pElement = |
| fonts->GetMutableDictFor(font_name.AsStringView()); |
| if (pElement) { |
| RetainPtr<CPDF_Font> font = |
| form_->GetFontForElement(std::move(pElement)); |
| if (font) { |
| return font; |
| } |
| } |
| } |
| } |
| RetainPtr<CPDF_Font> pFormFont = form_->GetFormFont(font_name); |
| if (pFormFont) { |
| return pFormFont; |
| } |
| |
| RetainPtr<CPDF_Dictionary> pPageDict = widget_dict_->GetMutableDictFor("P"); |
| RetainPtr<CPDF_Dictionary> pDict = ToDictionary( |
| CPDF_FormField::GetMutableFieldAttrForDict(pPageDict.Get(), "Resources")); |
| if (!pDict) { |
| return nullptr; |
| } |
| |
| RetainPtr<CPDF_Dictionary> fonts = pDict->GetMutableDictFor("Font"); |
| if (!ValidateFontResourceDict(fonts.Get())) { |
| return nullptr; |
| } |
| |
| RetainPtr<CPDF_Dictionary> pElement = |
| fonts->GetMutableDictFor(font_name.AsStringView()); |
| if (!pElement) { |
| return nullptr; |
| } |
| |
| return form_->GetFontForElement(std::move(pElement)); |
| } |
| |
| int CPDF_FormControl::GetControlAlignment() const { |
| if (widget_dict_->KeyExist(pdfium::form_fields::kQ)) { |
| return widget_dict_->GetIntegerFor(pdfium::form_fields::kQ, 0); |
| } |
| |
| RetainPtr<const CPDF_Object> pObj = |
| field_->GetFieldAttr(pdfium::form_fields::kQ); |
| if (pObj) { |
| return pObj->GetInteger(); |
| } |
| |
| return form_->GetFormAlignment(); |
| } |