summaryrefslogtreecommitdiff
path: root/core/fpdfdoc/cpdf_interactiveform.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'core/fpdfdoc/cpdf_interactiveform.cpp')
-rw-r--r--core/fpdfdoc/cpdf_interactiveform.cpp1066
1 files changed, 1066 insertions, 0 deletions
diff --git a/core/fpdfdoc/cpdf_interactiveform.cpp b/core/fpdfdoc/cpdf_interactiveform.cpp
new file mode 100644
index 0000000000..e05979d341
--- /dev/null
+++ b/core/fpdfdoc/cpdf_interactiveform.cpp
@@ -0,0 +1,1066 @@
+// Copyright 2016 PDFium Authors. All rights reserved.
+// 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_interactiveform.h"
+
+#include <utility>
+#include <vector>
+
+#include "constants/stream_dict_common.h"
+#include "core/fpdfapi/font/cpdf_font.h"
+#include "core/fpdfapi/font/cpdf_fontencoding.h"
+#include "core/fpdfapi/page/cpdf_page.h"
+#include "core/fpdfapi/parser/cfdf_document.h"
+#include "core/fpdfapi/parser/cpdf_array.h"
+#include "core/fpdfapi/parser/cpdf_dictionary.h"
+#include "core/fpdfapi/parser/cpdf_document.h"
+#include "core/fpdfapi/parser/cpdf_name.h"
+#include "core/fpdfapi/parser/cpdf_reference.h"
+#include "core/fpdfapi/parser/cpdf_string.h"
+#include "core/fpdfapi/parser/fpdf_parser_utility.h"
+#include "core/fpdfdoc/cpdf_filespec.h"
+#include "core/fpdfdoc/cpdf_formcontrol.h"
+#include "core/fxcrt/fx_codepage.h"
+#include "core/fxge/cfx_substfont.h"
+#include "core/fxge/fx_font.h"
+#include "third_party/base/ptr_util.h"
+#include "third_party/base/stl_util.h"
+
+namespace {
+
+const int nMaxRecursion = 32;
+
+ByteString GenerateNewFontResourceName(const CPDF_Dictionary* pResDict,
+ const ByteString& csPrefix) {
+ static const char kDummyFontName[] = "ZiTi";
+ ByteString csStr = csPrefix;
+ if (csStr.IsEmpty())
+ csStr = kDummyFontName;
+
+ const size_t szCount = csStr.GetLength();
+ size_t m = 0;
+ ByteString csTmp;
+ while (m < strlen(kDummyFontName) && m < szCount)
+ csTmp += csStr[m++];
+ while (m < strlen(kDummyFontName)) {
+ csTmp += '0' + m % 10;
+ m++;
+ }
+
+ const CPDF_Dictionary* pDict = pResDict->GetDictFor("Font");
+ ASSERT(pDict);
+
+ int num = 0;
+ ByteString bsNum;
+ while (true) {
+ ByteString csKey = csTmp + bsNum;
+ if (!pDict->KeyExist(csKey))
+ return csKey;
+ if (m < szCount)
+ csTmp += csStr[m++];
+ else
+ bsNum = ByteString::Format("%d", num++);
+
+ m++;
+ }
+ return csTmp;
+}
+
+void AddFont(CPDF_Dictionary*& pFormDict,
+ CPDF_Document* pDocument,
+ const CPDF_Font* pFont,
+ ByteString* csNameTag);
+
+void InitDict(CPDF_Dictionary*& pFormDict, CPDF_Document* pDocument) {
+ if (!pDocument)
+ return;
+
+ if (!pFormDict) {
+ pFormDict = pDocument->NewIndirect<CPDF_Dictionary>();
+ pDocument->GetRoot()->SetFor("AcroForm",
+ pFormDict->MakeReference(pDocument));
+ }
+
+ ByteString csDA;
+ if (!pFormDict->KeyExist("DR")) {
+ ByteString csBaseName;
+ uint8_t charSet = CPDF_InterForm::GetNativeCharSet();
+ CPDF_Font* pFont = CPDF_InterForm::AddStandardFont(
+ pDocument, CFX_Font::kDefaultAnsiFontName);
+ if (pFont)
+ AddFont(pFormDict, pDocument, pFont, &csBaseName);
+
+ if (charSet != FX_CHARSET_ANSI) {
+ ByteString csFontName = CPDF_InterForm::GetNativeFont(charSet, nullptr);
+ if (!pFont || csFontName != CFX_Font::kDefaultAnsiFontName) {
+ pFont = CPDF_InterForm::AddNativeFont(pDocument);
+ if (pFont) {
+ csBaseName.clear();
+ AddFont(pFormDict, pDocument, pFont, &csBaseName);
+ }
+ }
+ }
+ if (pFont)
+ csDA = "/" + PDF_NameEncode(csBaseName) + " 0 Tf";
+ }
+ if (!csDA.IsEmpty())
+ csDA += " ";
+
+ csDA += "0 g";
+ if (!pFormDict->KeyExist("DA"))
+ pFormDict->SetNewFor<CPDF_String>("DA", csDA, false);
+}
+
+CPDF_Font* GetFont(CPDF_Dictionary* pFormDict,
+ CPDF_Document* pDocument,
+ const ByteString& csNameTag) {
+ ByteString csAlias = PDF_NameDecode(csNameTag.AsStringView());
+ if (!pFormDict || csAlias.IsEmpty())
+ return nullptr;
+
+ CPDF_Dictionary* pDR = pFormDict->GetDictFor("DR");
+ if (!pDR)
+ return nullptr;
+
+ CPDF_Dictionary* pFonts = pDR->GetDictFor("Font");
+ if (!pFonts)
+ return nullptr;
+
+ CPDF_Dictionary* pElement = pFonts->GetDictFor(csAlias);
+ if (!pElement)
+ return nullptr;
+
+ if (pElement->GetStringFor("Type") == "Font")
+ return pDocument->LoadFont(pElement);
+ return nullptr;
+}
+
+CPDF_Font* GetNativeFont(CPDF_Dictionary* pFormDict,
+ CPDF_Document* pDocument,
+ uint8_t charSet,
+ ByteString* csNameTag) {
+ if (!pFormDict)
+ return nullptr;
+
+ CPDF_Dictionary* pDR = pFormDict->GetDictFor("DR");
+ if (!pDR)
+ return nullptr;
+
+ CPDF_Dictionary* pFonts = pDR->GetDictFor("Font");
+ if (!pFonts)
+ return nullptr;
+
+ for (const auto& it : *pFonts) {
+ const ByteString& csKey = it.first;
+ if (!it.second)
+ continue;
+
+ CPDF_Dictionary* pElement = ToDictionary(it.second->GetDirect());
+ if (!pElement)
+ continue;
+ if (pElement->GetStringFor("Type") != "Font")
+ continue;
+ CPDF_Font* pFind = pDocument->LoadFont(pElement);
+ if (!pFind)
+ continue;
+
+ CFX_SubstFont* pSubst = pFind->GetSubstFont();
+ if (!pSubst)
+ continue;
+
+ if (pSubst->m_Charset == static_cast<int>(charSet)) {
+ *csNameTag = csKey;
+ return pFind;
+ }
+ }
+ return nullptr;
+}
+
+bool FindFont(CPDF_Dictionary* pFormDict,
+ const CPDF_Font* pFont,
+ ByteString* csNameTag) {
+ if (!pFormDict || !pFont)
+ return false;
+
+ CPDF_Dictionary* pDR = pFormDict->GetDictFor("DR");
+ if (!pDR)
+ return false;
+
+ CPDF_Dictionary* pFonts = pDR->GetDictFor("Font");
+ if (!pFonts)
+ return false;
+
+ for (const auto& it : *pFonts) {
+ const ByteString& csKey = it.first;
+ if (!it.second)
+ continue;
+ CPDF_Dictionary* pElement = ToDictionary(it.second->GetDirect());
+ if (!pElement)
+ continue;
+ if (pElement->GetStringFor("Type") != "Font")
+ continue;
+ if (pFont->GetFontDict() == pElement) {
+ *csNameTag = csKey;
+ return true;
+ }
+ }
+ return false;
+}
+
+bool FindFont(CPDF_Dictionary* pFormDict,
+ CPDF_Document* pDocument,
+ ByteString csFontName,
+ CPDF_Font*& pFont,
+ ByteString* csNameTag) {
+ if (!pFormDict)
+ return false;
+
+ CPDF_Dictionary* pDR = pFormDict->GetDictFor("DR");
+ if (!pDR)
+ return false;
+
+ CPDF_Dictionary* pFonts = pDR->GetDictFor("Font");
+ if (!pFonts)
+ return false;
+
+ if (csFontName.GetLength() > 0)
+ csFontName.Remove(' ');
+
+ for (const auto& it : *pFonts) {
+ const ByteString& csKey = it.first;
+ if (!it.second)
+ continue;
+
+ CPDF_Dictionary* pElement = ToDictionary(it.second->GetDirect());
+ if (!pElement)
+ continue;
+ if (pElement->GetStringFor("Type") != "Font")
+ continue;
+ pFont = pDocument->LoadFont(pElement);
+ if (!pFont)
+ continue;
+
+ ByteString csBaseFont = pFont->GetBaseFont();
+ csBaseFont.Remove(' ');
+ if (csBaseFont == csFontName) {
+ *csNameTag = csKey;
+ return true;
+ }
+ }
+ return false;
+}
+
+void AddFont(CPDF_Dictionary*& pFormDict,
+ CPDF_Document* pDocument,
+ const CPDF_Font* pFont,
+ ByteString* csNameTag) {
+ if (!pFont)
+ return;
+ if (!pFormDict)
+ InitDict(pFormDict, pDocument);
+
+ ByteString csTag;
+ if (FindFont(pFormDict, pFont, &csTag)) {
+ *csNameTag = std::move(csTag);
+ return;
+ }
+ if (!pFormDict)
+ InitDict(pFormDict, pDocument);
+
+ CPDF_Dictionary* pDR = pFormDict->GetDictFor("DR");
+ if (!pDR)
+ pDR = pFormDict->SetNewFor<CPDF_Dictionary>("DR");
+
+ CPDF_Dictionary* pFonts = pDR->GetDictFor("Font");
+ if (!pFonts)
+ pFonts = pDR->SetNewFor<CPDF_Dictionary>("Font");
+
+ if (csNameTag->IsEmpty())
+ *csNameTag = pFont->GetBaseFont();
+
+ csNameTag->Remove(' ');
+ *csNameTag = GenerateNewFontResourceName(pDR, *csNameTag);
+ pFonts->SetFor(*csNameTag, pFont->GetFontDict()->MakeReference(pDocument));
+}
+
+CPDF_Font* AddNativeFont(CPDF_Dictionary*& pFormDict,
+ CPDF_Document* pDocument,
+ uint8_t charSet,
+ ByteString* csNameTag) {
+ if (!pFormDict)
+ InitDict(pFormDict, pDocument);
+
+ ByteString csTemp;
+ CPDF_Font* pFont = GetNativeFont(pFormDict, pDocument, charSet, &csTemp);
+ if (pFont) {
+ *csNameTag = std::move(csTemp);
+ return pFont;
+ }
+ ByteString csFontName = CPDF_InterForm::GetNativeFont(charSet, nullptr);
+ if (!csFontName.IsEmpty() &&
+ FindFont(pFormDict, pDocument, csFontName, pFont, csNameTag)) {
+ return pFont;
+ }
+ pFont = CPDF_InterForm::AddNativeFont(charSet, pDocument);
+ if (pFont)
+ AddFont(pFormDict, pDocument, pFont, csNameTag);
+
+ return pFont;
+}
+
+class CFieldNameExtractor {
+ public:
+ explicit CFieldNameExtractor(const WideString& full_name)
+ : m_FullName(full_name) {
+ m_pCur = m_FullName.c_str();
+ m_pEnd = m_pCur + m_FullName.GetLength();
+ }
+
+ void GetNext(const wchar_t*& pSubName, size_t& size) {
+ pSubName = m_pCur;
+ while (m_pCur < m_pEnd && m_pCur[0] != L'.')
+ m_pCur++;
+
+ size = static_cast<size_t>(m_pCur - pSubName);
+ if (m_pCur < m_pEnd && m_pCur[0] == L'.')
+ m_pCur++;
+ }
+
+ protected:
+ WideString m_FullName;
+ const wchar_t* m_pCur;
+ const wchar_t* m_pEnd;
+};
+
+#if _FX_PLATFORM_ == _FX_PLATFORM_WINDOWS_
+struct PDF_FONTDATA {
+ bool bFind;
+ LOGFONTA lf;
+};
+
+static int CALLBACK EnumFontFamExProc(ENUMLOGFONTEXA* lpelfe,
+ NEWTEXTMETRICEX* lpntme,
+ DWORD FontType,
+ LPARAM lParam) {
+ if (FontType != 0x004 || strchr(lpelfe->elfLogFont.lfFaceName, '@'))
+ return 1;
+
+ PDF_FONTDATA* pData = (PDF_FONTDATA*)lParam;
+ memcpy(&pData->lf, &lpelfe->elfLogFont, sizeof(LOGFONTA));
+ pData->bFind = true;
+ return 0;
+}
+
+bool RetrieveSpecificFont(LOGFONTA& lf) {
+ PDF_FONTDATA fd;
+ memset(&fd, 0, sizeof(PDF_FONTDATA));
+ HDC hDC = ::GetDC(nullptr);
+ EnumFontFamiliesExA(hDC, &lf, (FONTENUMPROCA)EnumFontFamExProc, (LPARAM)&fd,
+ 0);
+ ::ReleaseDC(nullptr, hDC);
+ if (fd.bFind)
+ memcpy(&lf, &fd.lf, sizeof(LOGFONTA));
+
+ return fd.bFind;
+}
+
+bool RetrieveSpecificFont(uint8_t charSet,
+ uint8_t pitchAndFamily,
+ LPCSTR pcsFontName,
+ LOGFONTA& lf) {
+ memset(&lf, 0, sizeof(LOGFONTA));
+ lf.lfCharSet = charSet;
+ lf.lfPitchAndFamily = pitchAndFamily;
+ if (pcsFontName) {
+ // TODO(dsinclair): Should this be strncpy?
+ // NOLINTNEXTLINE(runtime/printf)
+ strcpy(lf.lfFaceName, pcsFontName);
+ }
+ return RetrieveSpecificFont(lf);
+}
+#endif // _FX_PLATFORM_ == _FX_PLATFORM_WINDOWS_
+
+} // namespace
+
+class CFieldTree {
+ public:
+ class Node {
+ public:
+ Node() : m_pField(nullptr), m_level(0) {}
+ Node(const WideString& short_name, int level)
+ : m_ShortName(short_name), m_level(level) {}
+ ~Node() {}
+
+ void AddChildNode(std::unique_ptr<Node> pNode) {
+ m_Children.push_back(std::move(pNode));
+ }
+
+ size_t GetChildrenCount() const { return m_Children.size(); }
+
+ Node* GetChildAt(size_t i) { return m_Children[i].get(); }
+ const Node* GetChildAt(size_t i) const { return m_Children[i].get(); }
+
+ CPDF_FormField* GetFieldAtIndex(size_t index) {
+ size_t nFieldsToGo = index;
+ return GetFieldInternal(&nFieldsToGo);
+ }
+
+ size_t CountFields() const { return CountFieldsInternal(); }
+
+ void SetField(std::unique_ptr<CPDF_FormField> pField) {
+ m_pField = std::move(pField);
+ }
+
+ CPDF_FormField* GetField() const { return m_pField.get(); }
+ WideString GetShortName() const { return m_ShortName; }
+ int GetLevel() const { return m_level; }
+
+ private:
+ CPDF_FormField* GetFieldInternal(size_t* pFieldsToGo) {
+ if (m_pField) {
+ if (*pFieldsToGo == 0)
+ return m_pField.get();
+
+ --*pFieldsToGo;
+ }
+ for (size_t i = 0; i < GetChildrenCount(); ++i) {
+ CPDF_FormField* pField = GetChildAt(i)->GetFieldInternal(pFieldsToGo);
+ if (pField)
+ return pField;
+ }
+ return nullptr;
+ }
+
+ size_t CountFieldsInternal() const {
+ size_t count = 0;
+ if (m_pField)
+ ++count;
+
+ for (size_t i = 0; i < GetChildrenCount(); ++i)
+ count += GetChildAt(i)->CountFieldsInternal();
+ return count;
+ }
+
+ std::vector<std::unique_ptr<Node>> m_Children;
+ WideString m_ShortName;
+ std::unique_ptr<CPDF_FormField> m_pField;
+ const int m_level;
+ };
+
+ CFieldTree();
+ ~CFieldTree();
+
+ bool SetField(const WideString& full_name,
+ std::unique_ptr<CPDF_FormField> pField);
+ CPDF_FormField* GetField(const WideString& full_name);
+
+ Node* FindNode(const WideString& full_name);
+ Node* AddChild(Node* pParent, const WideString& short_name);
+
+ Node* Lookup(Node* pParent, const WideString& short_name);
+
+ Node m_Root;
+};
+
+CFieldTree::CFieldTree() {}
+
+CFieldTree::~CFieldTree() {}
+
+CFieldTree::Node* CFieldTree::AddChild(Node* pParent,
+ const WideString& short_name) {
+ if (!pParent)
+ return nullptr;
+
+ int level = pParent->GetLevel() + 1;
+ if (level > nMaxRecursion)
+ return nullptr;
+
+ auto pNew = pdfium::MakeUnique<Node>(short_name, pParent->GetLevel() + 1);
+ Node* pChild = pNew.get();
+ pParent->AddChildNode(std::move(pNew));
+ return pChild;
+}
+
+CFieldTree::Node* CFieldTree::Lookup(Node* pParent,
+ const WideString& short_name) {
+ if (!pParent)
+ return nullptr;
+
+ for (size_t i = 0; i < pParent->GetChildrenCount(); ++i) {
+ Node* pNode = pParent->GetChildAt(i);
+ if (pNode->GetShortName() == short_name)
+ return pNode;
+ }
+ return nullptr;
+}
+
+bool CFieldTree::SetField(const WideString& full_name,
+ std::unique_ptr<CPDF_FormField> pField) {
+ if (full_name.IsEmpty())
+ return false;
+
+ CFieldNameExtractor name_extractor(full_name);
+ const wchar_t* pName;
+ size_t nLength;
+ name_extractor.GetNext(pName, nLength);
+ Node* pNode = &m_Root;
+ Node* pLast = nullptr;
+ while (nLength > 0) {
+ pLast = pNode;
+ WideString name = WideString(pName, nLength);
+ pNode = Lookup(pLast, name);
+ if (!pNode)
+ pNode = AddChild(pLast, name);
+ if (!pNode)
+ return false;
+
+ name_extractor.GetNext(pName, nLength);
+ }
+ if (pNode == &m_Root)
+ return false;
+
+ pNode->SetField(std::move(pField));
+ return true;
+}
+
+CPDF_FormField* CFieldTree::GetField(const WideString& full_name) {
+ if (full_name.IsEmpty())
+ return nullptr;
+
+ CFieldNameExtractor name_extractor(full_name);
+ const wchar_t* pName;
+ size_t nLength;
+ name_extractor.GetNext(pName, nLength);
+ Node* pNode = &m_Root;
+ Node* pLast = nullptr;
+ while (nLength > 0 && pNode) {
+ pLast = pNode;
+ WideString name = WideString(pName, nLength);
+ pNode = Lookup(pLast, name);
+ name_extractor.GetNext(pName, nLength);
+ }
+ return pNode ? pNode->GetField() : nullptr;
+}
+
+CFieldTree::Node* CFieldTree::FindNode(const WideString& full_name) {
+ if (full_name.IsEmpty())
+ return nullptr;
+
+ CFieldNameExtractor name_extractor(full_name);
+ const wchar_t* pName;
+ size_t nLength;
+ name_extractor.GetNext(pName, nLength);
+ Node* pNode = &m_Root;
+ Node* pLast = nullptr;
+ while (nLength > 0 && pNode) {
+ pLast = pNode;
+ WideString name = WideString(pName, nLength);
+ pNode = Lookup(pLast, name);
+ name_extractor.GetNext(pName, nLength);
+ }
+ return pNode;
+}
+
+CPDF_Font* AddNativeInterFormFont(CPDF_Dictionary*& pFormDict,
+ CPDF_Document* pDocument,
+ ByteString* csNameTag) {
+ uint8_t charSet = CPDF_InterForm::GetNativeCharSet();
+ return AddNativeFont(pFormDict, pDocument, charSet, csNameTag);
+}
+
+// static
+uint8_t CPDF_InterForm::GetNativeCharSet() {
+ return FX_GetCharsetFromCodePage(FXSYS_GetACP());
+}
+
+CPDF_InterForm::CPDF_InterForm(CPDF_Document* pDocument)
+ : m_pDocument(pDocument), m_pFieldTree(pdfium::MakeUnique<CFieldTree>()) {
+ CPDF_Dictionary* pRoot = m_pDocument->GetRoot();
+ if (!pRoot)
+ return;
+
+ m_pFormDict = pRoot->GetDictFor("AcroForm");
+ if (!m_pFormDict)
+ return;
+
+ CPDF_Array* pFields = m_pFormDict->GetArrayFor("Fields");
+ if (!pFields)
+ return;
+
+ for (size_t i = 0; i < pFields->GetCount(); ++i)
+ LoadField(pFields->GetDictAt(i), 0);
+}
+
+CPDF_InterForm::~CPDF_InterForm() {}
+
+bool CPDF_InterForm::s_bUpdateAP = true;
+
+bool CPDF_InterForm::IsUpdateAPEnabled() {
+ return s_bUpdateAP;
+}
+
+void CPDF_InterForm::SetUpdateAP(bool bUpdateAP) {
+ s_bUpdateAP = bUpdateAP;
+}
+
+CPDF_Font* CPDF_InterForm::AddStandardFont(CPDF_Document* pDocument,
+ ByteString csFontName) {
+ if (!pDocument || csFontName.IsEmpty())
+ return nullptr;
+
+ if (csFontName == "ZapfDingbats")
+ return pDocument->AddStandardFont(csFontName.c_str(), nullptr);
+
+ CPDF_FontEncoding encoding(PDFFONT_ENCODING_WINANSI);
+ return pDocument->AddStandardFont(csFontName.c_str(), &encoding);
+}
+
+ByteString CPDF_InterForm::GetNativeFont(uint8_t charSet, void* pLogFont) {
+ ByteString csFontName;
+#if _FX_PLATFORM_ == _FX_PLATFORM_WINDOWS_
+ LOGFONTA lf = {};
+ if (charSet == FX_CHARSET_ANSI) {
+ csFontName = CFX_Font::kDefaultAnsiFontName;
+ return csFontName;
+ }
+ bool bRet = false;
+ const ByteString default_font_name =
+ CFX_Font::GetDefaultFontNameByCharset(charSet);
+ if (!default_font_name.IsEmpty()) {
+ bRet = RetrieveSpecificFont(charSet, DEFAULT_PITCH | FF_DONTCARE,
+ default_font_name.c_str(), lf);
+ }
+ if (!bRet) {
+ bRet = RetrieveSpecificFont(charSet, DEFAULT_PITCH | FF_DONTCARE,
+ CFX_Font::kUniversalDefaultFontName, lf);
+ }
+ if (!bRet) {
+ bRet = RetrieveSpecificFont(charSet, DEFAULT_PITCH | FF_DONTCARE,
+ "Microsoft Sans Serif", lf);
+ }
+ if (!bRet) {
+ bRet =
+ RetrieveSpecificFont(charSet, DEFAULT_PITCH | FF_DONTCARE, nullptr, lf);
+ }
+ if (bRet) {
+ if (pLogFont)
+ memcpy(pLogFont, &lf, sizeof(LOGFONTA));
+
+ csFontName = lf.lfFaceName;
+ return csFontName;
+ }
+#endif
+ return csFontName;
+}
+
+CPDF_Font* CPDF_InterForm::AddNativeFont(uint8_t charSet,
+ CPDF_Document* pDocument) {
+ if (!pDocument)
+ return nullptr;
+
+#if _FX_PLATFORM_ == _FX_PLATFORM_WINDOWS_
+ LOGFONTA lf;
+ ByteString csFontName = GetNativeFont(charSet, &lf);
+ if (!csFontName.IsEmpty()) {
+ if (csFontName == CFX_Font::kDefaultAnsiFontName)
+ return AddStandardFont(pDocument, csFontName);
+ return pDocument->AddWindowsFont(&lf);
+ }
+#endif
+ return nullptr;
+}
+
+CPDF_Font* CPDF_InterForm::AddNativeFont(CPDF_Document* pDocument) {
+ return pDocument ? AddNativeFont(GetNativeCharSet(), pDocument) : nullptr;
+}
+
+size_t CPDF_InterForm::CountFields(const WideString& csFieldName) const {
+ if (csFieldName.IsEmpty())
+ return m_pFieldTree->m_Root.CountFields();
+
+ CFieldTree::Node* pNode = m_pFieldTree->FindNode(csFieldName);
+ return pNode ? pNode->CountFields() : 0;
+}
+
+CPDF_FormField* CPDF_InterForm::GetField(uint32_t index,
+ const WideString& csFieldName) const {
+ if (csFieldName.IsEmpty())
+ return m_pFieldTree->m_Root.GetFieldAtIndex(index);
+
+ CFieldTree::Node* pNode = m_pFieldTree->FindNode(csFieldName);
+ return pNode ? pNode->GetFieldAtIndex(index) : nullptr;
+}
+
+CPDF_FormField* CPDF_InterForm::GetFieldByDict(
+ CPDF_Dictionary* pFieldDict) const {
+ if (!pFieldDict)
+ return nullptr;
+
+ WideString csWName = FPDF_GetFullName(pFieldDict);
+ return m_pFieldTree->GetField(csWName);
+}
+
+CPDF_FormControl* CPDF_InterForm::GetControlAtPoint(CPDF_Page* pPage,
+ const CFX_PointF& point,
+
+ int* z_order) const {
+ CPDF_Array* pAnnotList = pPage->GetDict()->GetArrayFor("Annots");
+ if (!pAnnotList)
+ return nullptr;
+
+ for (size_t i = pAnnotList->GetCount(); i > 0; --i) {
+ size_t annot_index = i - 1;
+ CPDF_Dictionary* pAnnot = pAnnotList->GetDictAt(annot_index);
+ if (!pAnnot)
+ continue;
+
+ const auto it = m_ControlMap.find(pAnnot);
+ if (it == m_ControlMap.end())
+ continue;
+
+ CPDF_FormControl* pControl = it->second.get();
+ if (!pControl->GetRect().Contains(point))
+ continue;
+
+ if (z_order)
+ *z_order = static_cast<int>(annot_index);
+ return pControl;
+ }
+ return nullptr;
+}
+
+CPDF_FormControl* CPDF_InterForm::GetControlByDict(
+ const CPDF_Dictionary* pWidgetDict) const {
+ const auto it = m_ControlMap.find(pWidgetDict);
+ return it != m_ControlMap.end() ? it->second.get() : nullptr;
+}
+
+bool CPDF_InterForm::NeedConstructAP() const {
+ return m_pFormDict && m_pFormDict->GetBooleanFor("NeedAppearances", false);
+}
+
+int CPDF_InterForm::CountFieldsInCalculationOrder() {
+ if (!m_pFormDict)
+ return 0;
+
+ CPDF_Array* pArray = m_pFormDict->GetArrayFor("CO");
+ return pArray ? pArray->GetCount() : 0;
+}
+
+CPDF_FormField* CPDF_InterForm::GetFieldInCalculationOrder(int index) {
+ if (!m_pFormDict || index < 0)
+ return nullptr;
+
+ CPDF_Array* pArray = m_pFormDict->GetArrayFor("CO");
+ if (!pArray)
+ return nullptr;
+
+ CPDF_Dictionary* pElement = ToDictionary(pArray->GetDirectObjectAt(index));
+ return pElement ? GetFieldByDict(pElement) : nullptr;
+}
+
+int CPDF_InterForm::FindFieldInCalculationOrder(const CPDF_FormField* pField) {
+ if (!m_pFormDict || !pField)
+ return -1;
+
+ CPDF_Array* pArray = m_pFormDict->GetArrayFor("CO");
+ if (!pArray)
+ return -1;
+
+ for (size_t i = 0; i < pArray->GetCount(); i++) {
+ CPDF_Object* pElement = pArray->GetDirectObjectAt(i);
+ if (pElement == pField->GetDict())
+ return i;
+ }
+ return -1;
+}
+
+CPDF_Font* CPDF_InterForm::GetFormFont(ByteString csNameTag) const {
+ return GetFont(m_pFormDict.Get(), m_pDocument.Get(), csNameTag);
+}
+
+CPDF_DefaultAppearance CPDF_InterForm::GetDefaultAppearance() const {
+ if (!m_pFormDict)
+ return CPDF_DefaultAppearance();
+ return CPDF_DefaultAppearance(m_pFormDict->GetStringFor("DA"));
+}
+
+int CPDF_InterForm::GetFormAlignment() const {
+ return m_pFormDict ? m_pFormDict->GetIntegerFor("Q", 0) : 0;
+}
+
+void CPDF_InterForm::ResetForm(const std::vector<CPDF_FormField*>& fields,
+ bool bIncludeOrExclude,
+ NotificationOption notify) {
+ size_t nCount = m_pFieldTree->m_Root.CountFields();
+ for (size_t i = 0; i < nCount; ++i) {
+ CPDF_FormField* pField = m_pFieldTree->m_Root.GetFieldAtIndex(i);
+ if (!pField)
+ continue;
+
+ if (bIncludeOrExclude == pdfium::ContainsValue(fields, pField))
+ pField->ResetField(notify);
+ }
+ if (notify == NotificationOption::kNotify && m_pFormNotify)
+ m_pFormNotify->AfterFormReset(this);
+}
+
+void CPDF_InterForm::ResetForm(NotificationOption notify) {
+ size_t nCount = m_pFieldTree->m_Root.CountFields();
+ for (size_t i = 0; i < nCount; ++i) {
+ CPDF_FormField* pField = m_pFieldTree->m_Root.GetFieldAtIndex(i);
+ if (!pField)
+ continue;
+
+ pField->ResetField(notify);
+ }
+ if (notify == NotificationOption::kNotify && m_pFormNotify)
+ m_pFormNotify->AfterFormReset(this);
+}
+
+void CPDF_InterForm::LoadField(CPDF_Dictionary* pFieldDict, int nLevel) {
+ if (nLevel > nMaxRecursion)
+ return;
+ if (!pFieldDict)
+ return;
+
+ uint32_t dwParentObjNum = pFieldDict->GetObjNum();
+ CPDF_Array* pKids = pFieldDict->GetArrayFor("Kids");
+ if (!pKids) {
+ AddTerminalField(pFieldDict);
+ return;
+ }
+
+ CPDF_Dictionary* pFirstKid = pKids->GetDictAt(0);
+ if (!pFirstKid)
+ return;
+
+ if (pFirstKid->KeyExist("T") || pFirstKid->KeyExist("Kids")) {
+ for (size_t i = 0; i < pKids->GetCount(); i++) {
+ CPDF_Dictionary* pChildDict = pKids->GetDictAt(i);
+ if (pChildDict) {
+ if (pChildDict->GetObjNum() != dwParentObjNum)
+ LoadField(pChildDict, nLevel + 1);
+ }
+ }
+ } else {
+ AddTerminalField(pFieldDict);
+ }
+}
+
+bool CPDF_InterForm::HasXFAForm() const {
+ return m_pFormDict && m_pFormDict->GetArrayFor("XFA");
+}
+
+void CPDF_InterForm::FixPageFields(CPDF_Page* pPage) {
+ CPDF_Dictionary* pPageDict = pPage->GetDict();
+ if (!pPageDict)
+ return;
+
+ CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots");
+ if (!pAnnots)
+ return;
+
+ for (size_t i = 0; i < pAnnots->GetCount(); i++) {
+ CPDF_Dictionary* pAnnot = pAnnots->GetDictAt(i);
+ if (pAnnot && pAnnot->GetStringFor("Subtype") == "Widget")
+ LoadField(pAnnot, 0);
+ }
+}
+
+void CPDF_InterForm::AddTerminalField(CPDF_Dictionary* pFieldDict) {
+ if (!pFieldDict->KeyExist("FT")) {
+ // Key "FT" is required for terminal fields, it is also inheritable.
+ CPDF_Dictionary* pParentDict = pFieldDict->GetDictFor("Parent");
+ if (!pParentDict || !pParentDict->KeyExist("FT"))
+ return;
+ }
+
+ CPDF_Dictionary* pDict = pFieldDict;
+ WideString csWName = FPDF_GetFullName(pFieldDict);
+ if (csWName.IsEmpty())
+ return;
+
+ CPDF_FormField* pField = nullptr;
+ pField = m_pFieldTree->GetField(csWName);
+ if (!pField) {
+ CPDF_Dictionary* pParent = pFieldDict;
+ if (!pFieldDict->KeyExist("T") &&
+ pFieldDict->GetStringFor("Subtype") == "Widget") {
+ pParent = pFieldDict->GetDictFor("Parent");
+ if (!pParent)
+ pParent = pFieldDict;
+ }
+
+ if (pParent && pParent != pFieldDict && !pParent->KeyExist("FT")) {
+ if (pFieldDict->KeyExist("FT")) {
+ CPDF_Object* pFTValue = pFieldDict->GetDirectObjectFor("FT");
+ if (pFTValue)
+ pParent->SetFor("FT", pFTValue->Clone());
+ }
+
+ if (pFieldDict->KeyExist("Ff")) {
+ CPDF_Object* pFfValue = pFieldDict->GetDirectObjectFor("Ff");
+ if (pFfValue)
+ pParent->SetFor("Ff", pFfValue->Clone());
+ }
+ }
+
+ auto newField = pdfium::MakeUnique<CPDF_FormField>(this, pParent);
+ pField = newField.get();
+ CPDF_Object* pTObj = pDict->GetObjectFor("T");
+ if (ToReference(pTObj)) {
+ std::unique_ptr<CPDF_Object> pClone = pTObj->CloneDirectObject();
+ if (pClone)
+ pDict->SetFor("T", std::move(pClone));
+ else
+ pDict->SetNewFor<CPDF_Name>("T", "");
+ }
+ if (!m_pFieldTree->SetField(csWName, std::move(newField)))
+ return;
+ }
+
+ CPDF_Array* pKids = pFieldDict->GetArrayFor("Kids");
+ if (pKids) {
+ for (size_t i = 0; i < pKids->GetCount(); i++) {
+ CPDF_Dictionary* pKid = pKids->GetDictAt(i);
+ if (!pKid)
+ continue;
+ if (pKid->GetStringFor("Subtype") != "Widget")
+ continue;
+
+ AddControl(pField, pKid);
+ }
+ } else {
+ if (pFieldDict->GetStringFor("Subtype") == "Widget")
+ AddControl(pField, pFieldDict);
+ }
+}
+
+CPDF_FormControl* CPDF_InterForm::AddControl(CPDF_FormField* pField,
+ CPDF_Dictionary* pWidgetDict) {
+ const auto it = m_ControlMap.find(pWidgetDict);
+ if (it != m_ControlMap.end())
+ return it->second.get();
+
+ auto pNew = pdfium::MakeUnique<CPDF_FormControl>(pField, pWidgetDict);
+ CPDF_FormControl* pControl = pNew.get();
+ m_ControlMap[pWidgetDict] = std::move(pNew);
+ pField->AddFormControl(pControl);
+ return pControl;
+}
+
+bool CPDF_InterForm::CheckRequiredFields(
+ const std::vector<CPDF_FormField*>* fields,
+ bool bIncludeOrExclude) const {
+ size_t nCount = m_pFieldTree->m_Root.CountFields();
+ for (size_t i = 0; i < nCount; ++i) {
+ CPDF_FormField* pField = m_pFieldTree->m_Root.GetFieldAtIndex(i);
+ if (!pField)
+ continue;
+
+ int32_t iType = pField->GetType();
+ if (iType == CPDF_FormField::PushButton ||
+ iType == CPDF_FormField::CheckBox || iType == CPDF_FormField::ListBox) {
+ continue;
+ }
+ if (pField->IsNoExport())
+ continue;
+
+ bool bFind = true;
+ if (fields)
+ bFind = pdfium::ContainsValue(*fields, pField);
+ if (bIncludeOrExclude == bFind) {
+ const CPDF_Dictionary* pFieldDict = pField->GetDict();
+ if (pField->IsRequired() && pFieldDict->GetStringFor("V").IsEmpty())
+ return false;
+ }
+ }
+ return true;
+}
+
+std::unique_ptr<CFDF_Document> CPDF_InterForm::ExportToFDF(
+ const WideString& pdf_path,
+ bool bSimpleFileSpec) const {
+ std::vector<CPDF_FormField*> fields;
+ size_t nCount = m_pFieldTree->m_Root.CountFields();
+ for (size_t i = 0; i < nCount; ++i)
+ fields.push_back(m_pFieldTree->m_Root.GetFieldAtIndex(i));
+ return ExportToFDF(pdf_path, fields, true, bSimpleFileSpec);
+}
+
+std::unique_ptr<CFDF_Document> CPDF_InterForm::ExportToFDF(
+ const WideString& pdf_path,
+ const std::vector<CPDF_FormField*>& fields,
+ bool bIncludeOrExclude,
+ bool bSimpleFileSpec) const {
+ std::unique_ptr<CFDF_Document> pDoc = CFDF_Document::CreateNewDoc();
+ if (!pDoc)
+ return nullptr;
+
+ CPDF_Dictionary* pMainDict = pDoc->GetRoot()->GetDictFor("FDF");
+ if (!pdf_path.IsEmpty()) {
+ if (bSimpleFileSpec) {
+ WideString wsFilePath = CPDF_FileSpec::EncodeFileName(pdf_path);
+ pMainDict->SetNewFor<CPDF_String>(pdfium::stream::kF,
+ wsFilePath.ToDefANSI(), false);
+ pMainDict->SetNewFor<CPDF_String>("UF", PDF_EncodeText(wsFilePath),
+ false);
+ } else {
+ auto pNewDict =
+ pdfium::MakeUnique<CPDF_Dictionary>(pDoc->GetByteStringPool());
+ pNewDict->SetNewFor<CPDF_Name>("Type", "Filespec");
+ CPDF_FileSpec filespec(pNewDict.get());
+ filespec.SetFileName(pdf_path);
+ pMainDict->SetFor("F", std::move(pNewDict));
+ }
+ }
+
+ CPDF_Array* pFields = pMainDict->SetNewFor<CPDF_Array>("Fields");
+ size_t nCount = m_pFieldTree->m_Root.CountFields();
+ for (size_t i = 0; i < nCount; ++i) {
+ CPDF_FormField* pField = m_pFieldTree->m_Root.GetFieldAtIndex(i);
+ if (!pField || pField->GetType() == CPDF_FormField::PushButton)
+ continue;
+
+ uint32_t dwFlags = pField->GetFieldFlags();
+ if (dwFlags & 0x04)
+ continue;
+
+ if (bIncludeOrExclude != pdfium::ContainsValue(fields, pField))
+ continue;
+
+ if ((dwFlags & 0x02) != 0 &&
+ pField->GetDict()->GetStringFor("V").IsEmpty()) {
+ continue;
+ }
+
+ WideString fullname = FPDF_GetFullName(pField->GetFieldDict());
+ auto pFieldDict =
+ pdfium::MakeUnique<CPDF_Dictionary>(pDoc->GetByteStringPool());
+ pFieldDict->SetNewFor<CPDF_String>("T", fullname);
+ if (pField->GetType() == CPDF_FormField::CheckBox ||
+ pField->GetType() == CPDF_FormField::RadioButton) {
+ WideString csExport = pField->GetCheckValue(false);
+ ByteString csBExport = PDF_EncodeText(csExport);
+ CPDF_Object* pOpt = FPDF_GetFieldAttr(pField->GetDict(), "Opt");
+ if (pOpt)
+ pFieldDict->SetNewFor<CPDF_String>("V", csBExport, false);
+ else
+ pFieldDict->SetNewFor<CPDF_Name>("V", csBExport);
+ } else {
+ CPDF_Object* pV = FPDF_GetFieldAttr(pField->GetDict(), "V");
+ if (pV)
+ pFieldDict->SetFor("V", pV->CloneDirectObject());
+ }
+ pFields->Add(std::move(pFieldDict));
+ }
+ return pDoc;
+}
+
+void CPDF_InterForm::SetFormNotify(IPDF_FormNotify* pNotify) {
+ m_pFormNotify = pNotify;
+}