blob: d71274505102a9ca0e8b68be3fe914c2aeeb0586 [file] [log] [blame]
// 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();
}