// 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 #include #include #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 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 pWidgetDict, CPDF_InteractiveForm* pForm) : m_pField(pField), m_pWidgetDict(std::move(pWidgetDict)), m_pForm(pForm) { DCHECK(m_pWidgetDict); } CPDF_FormControl::~CPDF_FormControl() = default; CFX_FloatRect CPDF_FormControl::GetRect() const { return m_pWidgetDict->GetRectFor("Rect"); } ByteString CPDF_FormControl::GetOnStateName() const { DCHECK(GetType() == CPDF_FormField::kCheckBox || GetType() == CPDF_FormField::kRadioButton); RetainPtr pAP = m_pWidgetDict->GetDictFor("AP"); if (!pAP) return ByteString(); RetainPtr 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(m_pField->GetFieldAttr("Opt"))) csOn = ByteString::FormatInteger(m_pField->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 pArray = ToArray(m_pField->GetFieldAttr("Opt")); if (pArray) csOn = pArray->GetByteStringAt(m_pField->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 = m_pWidgetDict->GetByteStringFor("AS"); return csAS == csOn; } bool CPDF_FormControl::IsDefaultChecked() const { DCHECK(GetType() == CPDF_FormField::kCheckBox || GetType() == CPDF_FormField::kRadioButton); RetainPtr pDV = m_pField->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 = m_pWidgetDict->GetByteStringFor("AS", "Off"); ByteString csAS = "Off"; if (bChecked) csAS = GetOnStateName(); if (csOldAS == csAS) return; m_pWidgetDict->SetNewFor("AS", csAS); } CPDF_FormControl::HighlightingMode CPDF_FormControl::GetHighlightingMode() const { ByteString csH = m_pWidgetDict->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(i); } } return kInvert; } CPDF_ApSettings CPDF_FormControl::GetMK() const { return CPDF_ApSettings(m_pWidgetDict->GetMutableDictFor("MK")); } bool CPDF_FormControl::HasMKEntry(const ByteString& csEntry) const { return GetMK().HasMKEntry(csEntry); } int CPDF_FormControl::GetRotation() const { return GetMK().GetRotation(); } CFX_Color::TypeAndARGB CPDF_FormControl::GetColorARGB( const ByteString& csEntry) { return GetMK().GetColorARGB(csEntry); } float CPDF_FormControl::GetOriginalColorComponent(int index, const ByteString& csEntry) { return GetMK().GetOriginalColorComponent(index, csEntry); } CFX_Color CPDF_FormControl::GetOriginalColor(const ByteString& csEntry) { return GetMK().GetOriginalColor(csEntry); } WideString CPDF_FormControl::GetCaption(const ByteString& csEntry) const { return GetMK().GetCaption(csEntry); } RetainPtr CPDF_FormControl::GetIcon(const ByteString& csEntry) { return GetMK().GetIcon(csEntry); } CPDF_IconFit CPDF_FormControl::GetIconFit() const { return GetMK().GetIconFit(); } int CPDF_FormControl::GetTextPosition() const { return GetMK().GetTextPosition(); } CPDF_DefaultAppearance CPDF_FormControl::GetDefaultAppearance() const { if (m_pWidgetDict->KeyExist(pdfium::form_fields::kDA)) { return CPDF_DefaultAppearance( m_pWidgetDict->GetByteStringFor(pdfium::form_fields::kDA)); } RetainPtr pObj = m_pField->GetFieldAttr(pdfium::form_fields::kDA); if (pObj) return CPDF_DefaultAppearance(pObj->GetString()); return m_pForm->GetDefaultAppearance(); } std::optional CPDF_FormControl::GetDefaultControlFontName() const { RetainPtr pFont = GetDefaultControlFont(); if (!pFont) return std::nullopt; return WideString::FromDefANSI(pFont->GetBaseFontName().AsStringView()); } RetainPtr CPDF_FormControl::GetDefaultControlFont() const { float fFontSize; CPDF_DefaultAppearance cDA = GetDefaultAppearance(); std::optional csFontNameTag = cDA.GetFont(&fFontSize); if (!csFontNameTag.has_value() || csFontNameTag->IsEmpty()) return nullptr; RetainPtr pDRDict = ToDictionary( CPDF_FormField::GetMutableFieldAttrForDict(m_pWidgetDict.Get(), "DR")); if (pDRDict) { RetainPtr pFonts = pDRDict->GetMutableDictFor("Font"); if (ValidateFontResourceDict(pFonts.Get())) { RetainPtr pElement = pFonts->GetMutableDictFor(csFontNameTag.value()); if (pElement) { RetainPtr pFont = m_pForm->GetFontForElement(std::move(pElement)); if (pFont) return pFont; } } } RetainPtr pFormFont = m_pForm->GetFormFont(csFontNameTag.value()); if (pFormFont) return pFormFont; RetainPtr pPageDict = m_pWidgetDict->GetMutableDictFor("P"); RetainPtr pDict = ToDictionary( CPDF_FormField::GetMutableFieldAttrForDict(pPageDict.Get(), "Resources")); if (!pDict) return nullptr; RetainPtr pFonts = pDict->GetMutableDictFor("Font"); if (!ValidateFontResourceDict(pFonts.Get())) return nullptr; RetainPtr pElement = pFonts->GetMutableDictFor(csFontNameTag.value()); if (!pElement) return nullptr; return m_pForm->GetFontForElement(std::move(pElement)); } int CPDF_FormControl::GetControlAlignment() const { if (m_pWidgetDict->KeyExist(pdfium::form_fields::kQ)) return m_pWidgetDict->GetIntegerFor(pdfium::form_fields::kQ, 0); RetainPtr pObj = m_pField->GetFieldAttr(pdfium::form_fields::kQ); if (pObj) return pObj->GetInteger(); return m_pForm->GetFormAlignment(); }