blob: bd3c7fa1a31274430d7c731f0408764dd959d80d [file] [log] [blame]
jaepark611adb82016-08-17 11:34:36 -07001// Copyright 2016 PDFium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
dsinclair114e46a2016-09-29 17:18:21 -07007#include "fpdfsdk/cpdfsdk_interform.h"
jaepark611adb82016-08-17 11:34:36 -07008
9#include <algorithm>
10#include <memory>
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -040011#include <sstream>
12#include <string>
Tom Sepeze005dc32018-06-19 17:33:32 +000013#include <utility>
thestig7c292e02016-09-28 14:14:26 -070014#include <vector>
jaepark611adb82016-08-17 11:34:36 -070015
dsinclair41872fa2016-10-04 11:29:35 -070016#include "core/fpdfapi/page/cpdf_page.h"
dsinclair488b7ad2016-10-04 11:55:50 -070017#include "core/fpdfapi/parser/cfdf_document.h"
18#include "core/fpdfapi/parser/cpdf_array.h"
Lei Zhang81535612018-10-09 21:15:17 +000019#include "core/fpdfapi/parser/cpdf_dictionary.h"
dsinclair488b7ad2016-10-04 11:55:50 -070020#include "core/fpdfapi/parser/cpdf_document.h"
21#include "core/fpdfapi/parser/cpdf_stream.h"
dsinclair1727aee2016-09-29 13:12:56 -070022#include "core/fpdfdoc/cpdf_actionfields.h"
23#include "core/fpdfdoc/cpdf_interform.h"
Tom Sepezc22d6712018-06-05 22:33:31 +000024#include "core/fxcrt/autorestorer.h"
dsinclair74a34fc2016-09-29 16:41:42 -070025#include "core/fxge/cfx_graphstatedata.h"
26#include "core/fxge/cfx_pathdata.h"
Dan Sinclair7d125322018-03-28 18:49:34 +000027#include "fpdfsdk/cpdfsdk_actionhandler.h"
dsinclair114e46a2016-09-29 17:18:21 -070028#include "fpdfsdk/cpdfsdk_annot.h"
Dan Sinclaircbf76e62018-03-28 21:00:35 +000029#include "fpdfsdk/cpdfsdk_annotiterator.h"
dsinclair735606d2016-10-05 15:47:02 -070030#include "fpdfsdk/cpdfsdk_formfillenvironment.h"
Dan Sinclair00d47a62018-03-28 18:39:04 +000031#include "fpdfsdk/cpdfsdk_helpers.h"
dsinclair114e46a2016-09-29 17:18:21 -070032#include "fpdfsdk/cpdfsdk_pageview.h"
33#include "fpdfsdk/cpdfsdk_widget.h"
jaepark611adb82016-08-17 11:34:36 -070034#include "fpdfsdk/formfiller/cffl_formfiller.h"
dsinclair114e46a2016-09-29 17:18:21 -070035#include "fpdfsdk/ipdfsdk_annothandler.h"
Dan Sinclaire0345a42017-10-30 20:20:42 +000036#include "fxjs/ijs_event_context.h"
37#include "fxjs/ijs_runtime.h"
Lei Zhang99f5bbb2018-10-09 21:31:28 +000038#include "third_party/base/ptr_util.h"
jaepark611adb82016-08-17 11:34:36 -070039#include "third_party/base/stl_util.h"
40
41#ifdef PDF_ENABLE_XFA
dsinclair114e46a2016-09-29 17:18:21 -070042#include "fpdfsdk/cpdfsdk_xfawidget.h"
dsinclair521b7502016-11-02 13:02:28 -070043#include "fpdfsdk/fpdfxfa/cpdfxfa_context.h"
dsinclair4d29e782016-10-04 14:02:47 -070044#include "fpdfsdk/fpdfxfa/cxfa_fwladaptertimermgr.h"
dsinclair5b493092016-09-29 20:20:24 -070045#include "xfa/fxfa/cxfa_eventparam.h"
Dan Sinclair80c48782017-03-23 12:11:20 -040046#include "xfa/fxfa/cxfa_ffdocview.h"
47#include "xfa/fxfa/cxfa_ffwidget.h"
48#include "xfa/fxfa/cxfa_ffwidgethandler.h"
jaepark611adb82016-08-17 11:34:36 -070049#endif // PDF_ENABLE_XFA
50
Ryan Harrison9baf31f2018-01-12 18:36:30 +000051namespace {
52
Lei Zhang4f7479a2018-03-21 13:37:06 +000053constexpr uint32_t kWhiteBGR = FXSYS_BGR(255, 255, 255);
54
Ryan Harrison9baf31f2018-01-12 18:36:30 +000055bool IsFormFieldTypeComboOrText(FormFieldType fieldType) {
56 switch (fieldType) {
57 case FormFieldType::kComboBox:
58 case FormFieldType::kTextField:
59 return true;
60 default:
61 return false;
62 }
63}
64
Ryan Harrison2056fac2018-01-16 16:06:45 +000065#ifdef PDF_ENABLE_XFA
66bool IsFormFieldTypeXFA(FormFieldType fieldType) {
67 switch (fieldType) {
68 case FormFieldType::kXFA:
69 case FormFieldType::kXFA_CheckBox:
70 case FormFieldType::kXFA_ComboBox:
71 case FormFieldType::kXFA_ImageField:
72 case FormFieldType::kXFA_ListBox:
73 case FormFieldType::kXFA_PushButton:
74 case FormFieldType::kXFA_Signature:
75 case FormFieldType::kXFA_TextField:
76 return true;
77 default:
78 return false;
79 }
80}
81#endif // PDF_ENABLE_XFA
82
Lei Zhang2ee811f2018-08-13 21:32:50 +000083bool FDFToURLEncodedData(std::vector<uint8_t>* pBuffer) {
Lei Zhangf0260b22018-08-13 22:50:02 +000084 std::unique_ptr<CFDF_Document> pFDF = CFDF_Document::ParseMemory(*pBuffer);
Lei Zhang4f1aa692018-08-13 21:31:18 +000085 if (!pFDF)
86 return true;
87
88 CPDF_Dictionary* pMainDict = pFDF->GetRoot()->GetDictFor("FDF");
89 if (!pMainDict)
90 return false;
91
92 CPDF_Array* pFields = pMainDict->GetArrayFor("Fields");
93 if (!pFields)
94 return false;
95
96 std::ostringstream fdfEncodedData;
97 for (uint32_t i = 0; i < pFields->GetCount(); i++) {
98 CPDF_Dictionary* pField = pFields->GetDictAt(i);
99 if (!pField)
100 continue;
101 WideString name;
102 name = pField->GetUnicodeTextFor("T");
103 ByteString name_b = name.ToDefANSI();
104 ByteString csBValue = pField->GetStringFor("V");
105 WideString csWValue = PDF_DecodeText(csBValue);
106 ByteString csValue_b = csWValue.ToDefANSI();
107 fdfEncodedData << name_b << "=" << csValue_b;
108 if (i != pFields->GetCount() - 1)
109 fdfEncodedData << "&";
110 }
111
Lei Zhang2ee811f2018-08-13 21:32:50 +0000112 size_t nBufSize = fdfEncodedData.tellp();
Lei Zhang4f1aa692018-08-13 21:31:18 +0000113 if (nBufSize <= 0)
114 return false;
115
Lei Zhang2ee811f2018-08-13 21:32:50 +0000116 pBuffer->resize(nBufSize);
117 memcpy(pBuffer->data(), fdfEncodedData.str().c_str(), nBufSize);
Lei Zhang4f1aa692018-08-13 21:31:18 +0000118 return true;
119}
120
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000121} // namespace
122
dsinclair690c0332016-10-11 09:13:01 -0700123CPDFSDK_InterForm::CPDFSDK_InterForm(CPDFSDK_FormFillEnvironment* pFormFillEnv)
124 : m_pFormFillEnv(pFormFillEnv),
Dan Sinclair0bb13332017-03-30 16:12:02 -0400125 m_pInterForm(
126 pdfium::MakeUnique<CPDF_InterForm>(m_pFormFillEnv->GetPDFDocument())),
jaepark611adb82016-08-17 11:34:36 -0700127#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700128 m_bXfaCalculate(true),
129 m_bXfaValidationsEnabled(true),
jaepark611adb82016-08-17 11:34:36 -0700130#endif // PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700131 m_bCalculate(true),
132 m_bBusy(false),
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000133 m_HighlightAlpha(0) {
jaepark611adb82016-08-17 11:34:36 -0700134 m_pInterForm->SetFormNotify(this);
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000135 RemoveAllHighLights();
jaepark611adb82016-08-17 11:34:36 -0700136}
137
138CPDFSDK_InterForm::~CPDFSDK_InterForm() {
139 m_Map.clear();
140#ifdef PDF_ENABLE_XFA
141 m_XFAMap.clear();
142#endif // PDF_ENABLE_XFA
143}
144
tsepez4cf55152016-11-02 14:37:54 -0700145bool CPDFSDK_InterForm::HighlightWidgets() {
146 return false;
jaepark611adb82016-08-17 11:34:36 -0700147}
148
149CPDFSDK_Widget* CPDFSDK_InterForm::GetSibling(CPDFSDK_Widget* pWidget,
tsepez4cf55152016-11-02 14:37:54 -0700150 bool bNext) const {
Dan Sinclaircbf76e62018-03-28 21:00:35 +0000151 auto pIterator = pdfium::MakeUnique<CPDFSDK_AnnotIterator>(
Dan Sinclair0bb13332017-03-30 16:12:02 -0400152 pWidget->GetPageView(), CPDF_Annot::Subtype::WIDGET);
jaepark611adb82016-08-17 11:34:36 -0700153
Tom Sepez4ef943b2018-07-26 19:06:06 +0000154 return ToCPDFSDKWidget(bNext ? pIterator->GetNextAnnot(pWidget)
155 : pIterator->GetPrevAnnot(pWidget));
jaepark611adb82016-08-17 11:34:36 -0700156}
157
dsinclairc5267c52016-11-04 15:35:12 -0700158CPDFSDK_Widget* CPDFSDK_InterForm::GetWidget(CPDF_FormControl* pControl) const {
jaepark611adb82016-08-17 11:34:36 -0700159 if (!pControl || !m_pInterForm)
160 return nullptr;
161
162 CPDFSDK_Widget* pWidget = nullptr;
163 const auto it = m_Map.find(pControl);
164 if (it != m_Map.end())
165 pWidget = it->second;
166 if (pWidget)
167 return pWidget;
jaepark611adb82016-08-17 11:34:36 -0700168
169 CPDF_Dictionary* pControlDict = pControl->GetWidget();
dsinclair7cbe68e2016-10-12 11:56:23 -0700170 CPDF_Document* pDocument = m_pFormFillEnv->GetPDFDocument();
jaepark611adb82016-08-17 11:34:36 -0700171 CPDFSDK_PageView* pPage = nullptr;
172
dsinclair38fd8442016-09-15 10:15:32 -0700173 if (CPDF_Dictionary* pPageDict = pControlDict->GetDictFor("P")) {
jaepark611adb82016-08-17 11:34:36 -0700174 int nPageIndex = pDocument->GetPageIndex(pPageDict->GetObjNum());
175 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700176 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700177 }
178
179 if (!pPage) {
180 int nPageIndex = GetPageIndexByAnnotDict(pDocument, pControlDict);
181 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700182 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700183 }
184
Tom Sepez4ef943b2018-07-26 19:06:06 +0000185 return pPage ? ToCPDFSDKWidget(pPage->GetAnnotByDict(pControlDict)) : nullptr;
jaepark611adb82016-08-17 11:34:36 -0700186}
187
188void CPDFSDK_InterForm::GetWidgets(
Ryan Harrison275e2602017-09-18 14:23:18 -0400189 const WideString& sFieldName,
tsepez8fa82792017-01-11 09:32:33 -0800190 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700191 for (int i = 0, sz = m_pInterForm->CountFields(sFieldName); i < sz; ++i) {
192 CPDF_FormField* pFormField = m_pInterForm->GetField(i, sFieldName);
193 ASSERT(pFormField);
194 GetWidgets(pFormField, widgets);
195 }
196}
197
198void CPDFSDK_InterForm::GetWidgets(
199 CPDF_FormField* pField,
tsepez8fa82792017-01-11 09:32:33 -0800200 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700201 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
202 CPDF_FormControl* pFormCtrl = pField->GetControl(i);
203 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700204 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
jaepark611adb82016-08-17 11:34:36 -0700205 if (pWidget)
tsepez8fa82792017-01-11 09:32:33 -0800206 widgets->emplace_back(pWidget);
jaepark611adb82016-08-17 11:34:36 -0700207 }
208}
209
210int CPDFSDK_InterForm::GetPageIndexByAnnotDict(
211 CPDF_Document* pDocument,
212 CPDF_Dictionary* pAnnotDict) const {
213 ASSERT(pAnnotDict);
214
215 for (int i = 0, sz = pDocument->GetPageCount(); i < sz; i++) {
Tom Sepez967aa072018-05-08 13:40:20 +0000216 if (CPDF_Dictionary* pPageDict = pDocument->GetPageDictionary(i)) {
dsinclair38fd8442016-09-15 10:15:32 -0700217 if (CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots")) {
jaepark611adb82016-08-17 11:34:36 -0700218 for (int j = 0, jsz = pAnnots->GetCount(); j < jsz; j++) {
219 CPDF_Object* pDict = pAnnots->GetDirectObjectAt(j);
220 if (pAnnotDict == pDict)
221 return i;
222 }
223 }
224 }
225 }
226
227 return -1;
228}
229
230void CPDFSDK_InterForm::AddMap(CPDF_FormControl* pControl,
231 CPDFSDK_Widget* pWidget) {
232 m_Map[pControl] = pWidget;
233}
234
235void CPDFSDK_InterForm::RemoveMap(CPDF_FormControl* pControl) {
236 m_Map.erase(pControl);
237}
238
tsepez4cf55152016-11-02 14:37:54 -0700239void CPDFSDK_InterForm::EnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700240 m_bCalculate = bEnabled;
241}
242
tsepez4cf55152016-11-02 14:37:54 -0700243bool CPDFSDK_InterForm::IsCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700244 return m_bCalculate;
245}
246
247#ifdef PDF_ENABLE_XFA
248void CPDFSDK_InterForm::AddXFAMap(CXFA_FFWidget* hWidget,
249 CPDFSDK_XFAWidget* pWidget) {
250 ASSERT(hWidget);
251 m_XFAMap[hWidget] = pWidget;
252}
253
254void CPDFSDK_InterForm::RemoveXFAMap(CXFA_FFWidget* hWidget) {
255 ASSERT(hWidget);
256 m_XFAMap.erase(hWidget);
257}
258
259CPDFSDK_XFAWidget* CPDFSDK_InterForm::GetXFAWidget(CXFA_FFWidget* hWidget) {
260 ASSERT(hWidget);
261 auto it = m_XFAMap.find(hWidget);
262 return it != m_XFAMap.end() ? it->second : nullptr;
263}
264
tsepez4cf55152016-11-02 14:37:54 -0700265void CPDFSDK_InterForm::XfaEnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700266 m_bXfaCalculate = bEnabled;
267}
tsepez4cf55152016-11-02 14:37:54 -0700268bool CPDFSDK_InterForm::IsXfaCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700269 return m_bXfaCalculate;
270}
271
tsepez4cf55152016-11-02 14:37:54 -0700272bool CPDFSDK_InterForm::IsXfaValidationsEnabled() {
jaepark611adb82016-08-17 11:34:36 -0700273 return m_bXfaValidationsEnabled;
274}
tsepez4cf55152016-11-02 14:37:54 -0700275void CPDFSDK_InterForm::XfaSetValidationsEnabled(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700276 m_bXfaValidationsEnabled = bEnabled;
277}
278
Dan Sinclairce047a62018-01-30 18:15:02 +0000279void CPDFSDK_InterForm::SynchronizeField(CPDF_FormField* pFormField) {
jaepark611adb82016-08-17 11:34:36 -0700280 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
281 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
dsinclairc5267c52016-11-04 15:35:12 -0700282 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
Dan Sinclairce047a62018-01-30 18:15:02 +0000283 pWidget->Synchronize(false);
jaepark611adb82016-08-17 11:34:36 -0700284 }
285}
286#endif // PDF_ENABLE_XFA
287
288void CPDFSDK_InterForm::OnCalculate(CPDF_FormField* pFormField) {
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000289 if (!m_pFormFillEnv->IsJSPlatformPresent())
jaepark611adb82016-08-17 11:34:36 -0700290 return;
291
292 if (m_bBusy)
293 return;
294
Tom Sepezc22d6712018-06-05 22:33:31 +0000295 AutoRestorer<bool> restorer(&m_bBusy);
tsepez4cf55152016-11-02 14:37:54 -0700296 m_bBusy = true;
jaepark611adb82016-08-17 11:34:36 -0700297
Tom Sepezc22d6712018-06-05 22:33:31 +0000298 if (!IsCalculateEnabled())
jaepark611adb82016-08-17 11:34:36 -0700299 return;
jaepark611adb82016-08-17 11:34:36 -0700300
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000301 IJS_Runtime* pRuntime = m_pFormFillEnv->GetIJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700302 int nSize = m_pInterForm->CountFieldsInCalculationOrder();
303 for (int i = 0; i < nSize; i++) {
304 CPDF_FormField* pField = m_pInterForm->GetFieldInCalculationOrder(i);
305 if (!pField)
306 continue;
307
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000308 FormFieldType fieldType = pField->GetFieldType();
309 if (!IsFormFieldTypeComboOrText(fieldType))
jaepark611adb82016-08-17 11:34:36 -0700310 continue;
311
312 CPDF_AAction aAction = pField->GetAdditionalAction();
313 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Calculate))
314 continue;
315
316 CPDF_Action action = aAction.GetAction(CPDF_AAction::Calculate);
317 if (!action.GetDict())
318 continue;
319
Ryan Harrison275e2602017-09-18 14:23:18 -0400320 WideString csJS = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700321 if (csJS.IsEmpty())
322 continue;
323
Ryan Harrison275e2602017-09-18 14:23:18 -0400324 WideString sOldValue = pField->GetValue();
325 WideString sValue = sOldValue;
tsepez4cf55152016-11-02 14:37:54 -0700326 bool bRC = true;
Tom Sepezc22d6712018-06-05 22:33:31 +0000327 IJS_Runtime::ScopedEventContext pContext(pRuntime);
Tom Sepez3d813152018-06-21 17:28:24 +0000328 pContext->OnField_Calculate(pFormField, pField, &sValue, &bRC);
jaepark611adb82016-08-17 11:34:36 -0700329
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000330 Optional<IJS_Runtime::JS_Error> err = pContext->RunScript(csJS);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000331 if (!err && bRC && sValue.Compare(sOldValue) != 0)
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000332 pField->SetValue(sValue, NotificationOption::kNotify);
jaepark611adb82016-08-17 11:34:36 -0700333 }
jaepark611adb82016-08-17 11:34:36 -0700334}
335
Ryan Harrison275e2602017-09-18 14:23:18 -0400336WideString CPDFSDK_InterForm::OnFormat(CPDF_FormField* pFormField,
337 bool& bFormatted) {
338 WideString sValue = pFormField->GetValue();
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000339 if (!m_pFormFillEnv->IsJSPlatformPresent()) {
tsepez4cf55152016-11-02 14:37:54 -0700340 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700341 return sValue;
342 }
343
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000344 IJS_Runtime* pRuntime = m_pFormFillEnv->GetIJSRuntime();
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000345 if (pFormField->GetFieldType() == FormFieldType::kComboBox &&
jaepark611adb82016-08-17 11:34:36 -0700346 pFormField->CountSelectedItems() > 0) {
347 int index = pFormField->GetSelectedIndex(0);
348 if (index >= 0)
349 sValue = pFormField->GetOptionLabel(index);
350 }
351
tsepez4cf55152016-11-02 14:37:54 -0700352 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700353
354 CPDF_AAction aAction = pFormField->GetAdditionalAction();
355 if (aAction.GetDict() && aAction.ActionExist(CPDF_AAction::Format)) {
356 CPDF_Action action = aAction.GetAction(CPDF_AAction::Format);
357 if (action.GetDict()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400358 WideString script = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700359 if (!script.IsEmpty()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400360 WideString Value = sValue;
Tom Sepezc22d6712018-06-05 22:33:31 +0000361 IJS_Runtime::ScopedEventContext pContext(pRuntime);
Tom Sepez3d813152018-06-21 17:28:24 +0000362 pContext->OnField_Format(pFormField, &Value, true);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000363 Optional<IJS_Runtime::JS_Error> err = pContext->RunScript(script);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000364 if (!err) {
Tom Sepeze005dc32018-06-19 17:33:32 +0000365 sValue = std::move(Value);
tsepez4cf55152016-11-02 14:37:54 -0700366 bFormatted = true;
jaepark611adb82016-08-17 11:34:36 -0700367 }
368 }
369 }
370 }
jaepark611adb82016-08-17 11:34:36 -0700371 return sValue;
372}
373
374void CPDFSDK_InterForm::ResetFieldAppearance(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400375 const WideString* sValue,
tsepez4cf55152016-11-02 14:37:54 -0700376 bool bValueChanged) {
jaepark611adb82016-08-17 11:34:36 -0700377 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
378 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
379 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700380 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700381 pWidget->ResetAppearance(sValue, bValueChanged);
382 }
383}
384
385void CPDFSDK_InterForm::UpdateField(CPDF_FormField* pFormField) {
Lei Zhang375c2762017-03-10 14:37:14 -0800386 auto* formfiller = m_pFormFillEnv->GetInteractiveFormFiller();
jaepark611adb82016-08-17 11:34:36 -0700387 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
388 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
389 ASSERT(pFormCtrl);
390
Lei Zhang77f9bff2017-08-29 11:34:12 -0700391 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
392 if (!pWidget)
393 continue;
394
Tom Sepez101535f2018-06-12 13:36:05 +0000395 IPDF_Page* pPage = pWidget->GetPage();
Lei Zhang77f9bff2017-08-29 11:34:12 -0700396 FX_RECT rect = formfiller->GetViewBBox(
397 m_pFormFillEnv->GetPageView(pPage, false), pWidget);
398 m_pFormFillEnv->Invalidate(pPage, rect);
jaepark611adb82016-08-17 11:34:36 -0700399 }
400}
401
tsepez4cf55152016-11-02 14:37:54 -0700402bool CPDFSDK_InterForm::OnKeyStrokeCommit(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400403 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700404 CPDF_AAction aAction = pFormField->GetAdditionalAction();
405 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::KeyStroke))
tsepez4cf55152016-11-02 14:37:54 -0700406 return true;
jaepark611adb82016-08-17 11:34:36 -0700407
408 CPDF_Action action = aAction.GetAction(CPDF_AAction::KeyStroke);
409 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700410 return true;
jaepark611adb82016-08-17 11:34:36 -0700411
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700412 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
Dan Sinclair7d125322018-03-28 18:49:34 +0000413 CPDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700414 fa.bModifier = false;
415 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700416 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700417 pActionHandler->DoAction_FieldJavaScript(
Dan Sinclair8cdea722018-01-30 18:56:50 +0000418 action, CPDF_AAction::KeyStroke, m_pFormFillEnv.Get(), pFormField, &fa);
jaepark611adb82016-08-17 11:34:36 -0700419 return fa.bRC;
420}
421
tsepez4cf55152016-11-02 14:37:54 -0700422bool CPDFSDK_InterForm::OnValidate(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400423 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700424 CPDF_AAction aAction = pFormField->GetAdditionalAction();
425 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Validate))
tsepez4cf55152016-11-02 14:37:54 -0700426 return true;
jaepark611adb82016-08-17 11:34:36 -0700427
428 CPDF_Action action = aAction.GetAction(CPDF_AAction::Validate);
429 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700430 return true;
jaepark611adb82016-08-17 11:34:36 -0700431
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700432 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
Dan Sinclair7d125322018-03-28 18:49:34 +0000433 CPDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700434 fa.bModifier = false;
435 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700436 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700437 pActionHandler->DoAction_FieldJavaScript(
Dan Sinclair8cdea722018-01-30 18:56:50 +0000438 action, CPDF_AAction::Validate, m_pFormFillEnv.Get(), pFormField, &fa);
jaepark611adb82016-08-17 11:34:36 -0700439 return fa.bRC;
440}
441
tsepez4cf55152016-11-02 14:37:54 -0700442bool CPDFSDK_InterForm::DoAction_Hide(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700443 ASSERT(action.GetDict());
444
445 CPDF_ActionFields af(&action);
Lei Zhang5cee3f22018-05-25 21:48:49 +0000446 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700447 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
448
449 bool bHide = action.GetHideStatus();
tsepez4cf55152016-11-02 14:37:54 -0700450 bool bChanged = false;
jaepark611adb82016-08-17 11:34:36 -0700451
452 for (CPDF_FormField* pField : fields) {
453 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
454 CPDF_FormControl* pControl = pField->GetControl(i);
455 ASSERT(pControl);
456
dsinclairc5267c52016-11-04 15:35:12 -0700457 if (CPDFSDK_Widget* pWidget = GetWidget(pControl)) {
jaepark611adb82016-08-17 11:34:36 -0700458 uint32_t nFlags = pWidget->GetFlags();
459 nFlags &= ~ANNOTFLAG_INVISIBLE;
460 nFlags &= ~ANNOTFLAG_NOVIEW;
461 if (bHide)
462 nFlags |= ANNOTFLAG_HIDDEN;
463 else
464 nFlags &= ~ANNOTFLAG_HIDDEN;
465 pWidget->SetFlags(nFlags);
466 pWidget->GetPageView()->UpdateView(pWidget);
tsepez4cf55152016-11-02 14:37:54 -0700467 bChanged = true;
jaepark611adb82016-08-17 11:34:36 -0700468 }
469 }
470 }
471
472 return bChanged;
473}
474
tsepez4cf55152016-11-02 14:37:54 -0700475bool CPDFSDK_InterForm::DoAction_SubmitForm(const CPDF_Action& action) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400476 WideString sDestination = action.GetFilePath();
jaepark611adb82016-08-17 11:34:36 -0700477 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700478 return false;
jaepark611adb82016-08-17 11:34:36 -0700479
Lei Zhang5cee3f22018-05-25 21:48:49 +0000480 const CPDF_Dictionary* pActionDict = action.GetDict();
jaepark611adb82016-08-17 11:34:36 -0700481 if (pActionDict->KeyExist("Fields")) {
482 CPDF_ActionFields af(&action);
483 uint32_t dwFlags = action.GetFlags();
Lei Zhang5cee3f22018-05-25 21:48:49 +0000484 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700485 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
486 if (!fields.empty()) {
487 bool bIncludeOrExclude = !(dwFlags & 0x01);
Nicolas Penaa478dc52017-01-23 15:48:51 -0500488 if (!m_pInterForm->CheckRequiredFields(&fields, bIncludeOrExclude))
tsepez4cf55152016-11-02 14:37:54 -0700489 return false;
jaepark611adb82016-08-17 11:34:36 -0700490
491 return SubmitFields(sDestination, fields, bIncludeOrExclude, false);
492 }
493 }
Nicolas Penaa478dc52017-01-23 15:48:51 -0500494 if (!m_pInterForm->CheckRequiredFields(nullptr, true))
tsepez4cf55152016-11-02 14:37:54 -0700495 return false;
jaepark611adb82016-08-17 11:34:36 -0700496
tsepez4cf55152016-11-02 14:37:54 -0700497 return SubmitForm(sDestination, false);
jaepark611adb82016-08-17 11:34:36 -0700498}
499
Ryan Harrison275e2602017-09-18 14:23:18 -0400500bool CPDFSDK_InterForm::SubmitFields(const WideString& csDestination,
tsepez4cf55152016-11-02 14:37:54 -0700501 const std::vector<CPDF_FormField*>& fields,
502 bool bIncludeOrExclude,
503 bool bUrlEncoded) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400504 ByteString textBuf = ExportFieldsToFDFTextBuf(fields, bIncludeOrExclude);
Tom Sepez65be34f2018-09-14 20:58:36 +0000505 if (textBuf.IsEmpty())
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400506 return false;
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400507
Tom Sepez65be34f2018-09-14 20:58:36 +0000508 std::vector<uint8_t> buffer(textBuf.begin(), textBuf.end());
Lei Zhang2ee811f2018-08-13 21:32:50 +0000509 if (bUrlEncoded && !FDFToURLEncodedData(&buffer))
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400510 return false;
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400511
Lei Zhang2ee811f2018-08-13 21:32:50 +0000512 m_pFormFillEnv->JS_docSubmitForm(buffer.data(), buffer.size(), csDestination);
tsepez4cf55152016-11-02 14:37:54 -0700513 return true;
jaepark611adb82016-08-17 11:34:36 -0700514}
515
Ryan Harrison275e2602017-09-18 14:23:18 -0400516ByteString CPDFSDK_InterForm::ExportFieldsToFDFTextBuf(
jaepark611adb82016-08-17 11:34:36 -0700517 const std::vector<CPDF_FormField*>& fields,
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400518 bool bIncludeOrExclude) {
Tom Sepez690d4562017-05-18 11:42:46 -0700519 std::unique_ptr<CFDF_Document> pFDF = m_pInterForm->ExportToFDF(
520 m_pFormFillEnv->JS_docGetFilePath(), fields, bIncludeOrExclude, false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400521
Ryan Harrison275e2602017-09-18 14:23:18 -0400522 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700523}
524
Ryan Harrison275e2602017-09-18 14:23:18 -0400525bool CPDFSDK_InterForm::SubmitForm(const WideString& sDestination,
tsepez4cf55152016-11-02 14:37:54 -0700526 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700527 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700528 return false;
jaepark611adb82016-08-17 11:34:36 -0700529
dsinclair7cbe68e2016-10-12 11:56:23 -0700530 if (!m_pFormFillEnv || !m_pInterForm)
tsepez4cf55152016-11-02 14:37:54 -0700531 return false;
jaepark611adb82016-08-17 11:34:36 -0700532
Tom Sepez690d4562017-05-18 11:42:46 -0700533 std::unique_ptr<CFDF_Document> pFDFDoc =
534 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
jaepark611adb82016-08-17 11:34:36 -0700535 if (!pFDFDoc)
tsepez4cf55152016-11-02 14:37:54 -0700536 return false;
jaepark611adb82016-08-17 11:34:36 -0700537
Ryan Harrison275e2602017-09-18 14:23:18 -0400538 ByteString fdfBuffer = pFDFDoc->WriteToString();
Ryan Harrison875e98c2017-09-27 10:53:11 -0400539 if (fdfBuffer.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700540 return false;
jaepark611adb82016-08-17 11:34:36 -0700541
Tom Sepez65be34f2018-09-14 20:58:36 +0000542 std::vector<uint8_t> buffer(fdfBuffer.begin(), fdfBuffer.end());
Lei Zhang2ee811f2018-08-13 21:32:50 +0000543 if (bUrlEncoded && !FDFToURLEncodedData(&buffer))
tsepez4cf55152016-11-02 14:37:54 -0700544 return false;
jaepark611adb82016-08-17 11:34:36 -0700545
Lei Zhang2ee811f2018-08-13 21:32:50 +0000546 m_pFormFillEnv->JS_docSubmitForm(buffer.data(), buffer.size(), sDestination);
tsepez4cf55152016-11-02 14:37:54 -0700547 return true;
jaepark611adb82016-08-17 11:34:36 -0700548}
549
Ryan Harrison275e2602017-09-18 14:23:18 -0400550ByteString CPDFSDK_InterForm::ExportFormToFDFTextBuf() {
Tom Sepez690d4562017-05-18 11:42:46 -0700551 std::unique_ptr<CFDF_Document> pFDF =
552 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400553
Ryan Harrison275e2602017-09-18 14:23:18 -0400554 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700555}
556
dan sinclair7544a4b2018-03-08 15:19:50 +0000557void CPDFSDK_InterForm::DoAction_ResetForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700558 ASSERT(action.GetDict());
Lei Zhang5cee3f22018-05-25 21:48:49 +0000559 const CPDF_Dictionary* pActionDict = action.GetDict();
dan sinclair7544a4b2018-03-08 15:19:50 +0000560 if (!pActionDict->KeyExist("Fields")) {
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000561 m_pInterForm->ResetForm(NotificationOption::kNotify);
dan sinclair7544a4b2018-03-08 15:19:50 +0000562 return;
563 }
jaepark611adb82016-08-17 11:34:36 -0700564 CPDF_ActionFields af(&action);
565 uint32_t dwFlags = action.GetFlags();
Lei Zhang5cee3f22018-05-25 21:48:49 +0000566 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700567 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000568 m_pInterForm->ResetForm(fields, !(dwFlags & 0x01),
569 NotificationOption::kNotify);
jaepark611adb82016-08-17 11:34:36 -0700570}
571
jaepark611adb82016-08-17 11:34:36 -0700572std::vector<CPDF_FormField*> CPDFSDK_InterForm::GetFieldFromObjects(
Lei Zhang5cee3f22018-05-25 21:48:49 +0000573 const std::vector<const CPDF_Object*>& objects) const {
jaepark611adb82016-08-17 11:34:36 -0700574 std::vector<CPDF_FormField*> fields;
Lei Zhang5cee3f22018-05-25 21:48:49 +0000575 for (const CPDF_Object* pObject : objects) {
Dan Sinclairce047a62018-01-30 18:15:02 +0000576 if (!pObject || !pObject->IsString())
577 continue;
578
579 WideString csName = pObject->GetUnicodeText();
580 CPDF_FormField* pField = m_pInterForm->GetField(0, csName);
581 if (pField)
582 fields.push_back(pField);
jaepark611adb82016-08-17 11:34:36 -0700583 }
584 return fields;
585}
586
dan sinclair507fb4e2018-03-08 15:14:09 +0000587bool CPDFSDK_InterForm::BeforeValueChange(CPDF_FormField* pField,
588 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000589 FormFieldType fieldType = pField->GetFieldType();
590 if (!IsFormFieldTypeComboOrText(fieldType))
dan sinclair507fb4e2018-03-08 15:14:09 +0000591 return true;
jaepark611adb82016-08-17 11:34:36 -0700592 if (!OnKeyStrokeCommit(pField, csValue))
dan sinclair507fb4e2018-03-08 15:14:09 +0000593 return false;
594 return OnValidate(pField, csValue);
jaepark611adb82016-08-17 11:34:36 -0700595}
596
597void CPDFSDK_InterForm::AfterValueChange(CPDF_FormField* pField) {
598#ifdef PDF_ENABLE_XFA
Dan Sinclairce047a62018-01-30 18:15:02 +0000599 SynchronizeField(pField);
jaepark611adb82016-08-17 11:34:36 -0700600#endif // PDF_ENABLE_XFA
Dan Sinclairce047a62018-01-30 18:15:02 +0000601
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000602 FormFieldType fieldType = pField->GetFieldType();
Dan Sinclairce047a62018-01-30 18:15:02 +0000603 if (!IsFormFieldTypeComboOrText(fieldType))
604 return;
605
606 OnCalculate(pField);
607 bool bFormatted = false;
608 WideString sValue = OnFormat(pField, bFormatted);
609 ResetFieldAppearance(pField, bFormatted ? &sValue : nullptr, true);
610 UpdateField(pField);
jaepark611adb82016-08-17 11:34:36 -0700611}
612
dan sinclair507fb4e2018-03-08 15:14:09 +0000613bool CPDFSDK_InterForm::BeforeSelectionChange(CPDF_FormField* pField,
614 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000615 if (pField->GetFieldType() != FormFieldType::kListBox)
dan sinclair507fb4e2018-03-08 15:14:09 +0000616 return true;
jaepark611adb82016-08-17 11:34:36 -0700617 if (!OnKeyStrokeCommit(pField, csValue))
dan sinclair507fb4e2018-03-08 15:14:09 +0000618 return false;
619 return OnValidate(pField, csValue);
jaepark611adb82016-08-17 11:34:36 -0700620}
621
622void CPDFSDK_InterForm::AfterSelectionChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000623 if (pField->GetFieldType() != FormFieldType::kListBox)
jaepark611adb82016-08-17 11:34:36 -0700624 return;
625
626 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700627 ResetFieldAppearance(pField, nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700628 UpdateField(pField);
629}
630
631void CPDFSDK_InterForm::AfterCheckedStatusChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000632 FormFieldType fieldType = pField->GetFieldType();
633 if (fieldType != FormFieldType::kCheckBox &&
634 fieldType != FormFieldType::kRadioButton)
jaepark611adb82016-08-17 11:34:36 -0700635 return;
636
637 OnCalculate(pField);
638 UpdateField(pField);
639}
640
jaepark611adb82016-08-17 11:34:36 -0700641void CPDFSDK_InterForm::AfterFormReset(CPDF_InterForm* pForm) {
642 OnCalculate(nullptr);
643}
644
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000645bool CPDFSDK_InterForm::IsNeedHighLight(FormFieldType fieldType) {
646 if (fieldType == FormFieldType::kUnknown)
tsepez4cf55152016-11-02 14:37:54 -0700647 return false;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000648
Ryan Harrison2056fac2018-01-16 16:06:45 +0000649#ifdef PDF_ENABLE_XFA
650 // For the XFA fields, we need to return if the specific field type has
651 // highlight enabled or if the general XFA field type has it enabled.
652 if (IsFormFieldTypeXFA(fieldType)) {
653 if (!m_NeedsHighlight[static_cast<size_t>(fieldType)])
654 return m_NeedsHighlight[static_cast<size_t>(FormFieldType::kXFA)];
655 }
656#endif // PDF_ENABLE_XFA
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000657 return m_NeedsHighlight[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700658}
659
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000660void CPDFSDK_InterForm::RemoveAllHighLights() {
661 std::fill(m_HighlightColor, m_HighlightColor + kFormFieldTypeCount,
Lei Zhang4f7479a2018-03-21 13:37:06 +0000662 kWhiteBGR);
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000663 std::fill(m_NeedsHighlight, m_NeedsHighlight + kFormFieldTypeCount, false);
jaepark611adb82016-08-17 11:34:36 -0700664}
665
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000666void CPDFSDK_InterForm::SetHighlightColor(FX_COLORREF clr,
667 FormFieldType fieldType) {
668 if (fieldType == FormFieldType::kUnknown)
jaepark611adb82016-08-17 11:34:36 -0700669 return;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000670
671 m_HighlightColor[static_cast<size_t>(fieldType)] = clr;
672 m_NeedsHighlight[static_cast<size_t>(fieldType)] = true;
673}
674
675void CPDFSDK_InterForm::SetAllHighlightColors(FX_COLORREF clr) {
676 for (auto type : kFormFieldTypes) {
677 m_HighlightColor[static_cast<size_t>(type)] = clr;
678 m_NeedsHighlight[static_cast<size_t>(type)] = true;
jaepark611adb82016-08-17 11:34:36 -0700679 }
680}
681
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000682FX_COLORREF CPDFSDK_InterForm::GetHighlightColor(FormFieldType fieldType) {
683 if (fieldType == FormFieldType::kUnknown)
Lei Zhang4f7479a2018-03-21 13:37:06 +0000684 return kWhiteBGR;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000685
Ryan Harrison2056fac2018-01-16 16:06:45 +0000686#ifdef PDF_ENABLE_XFA
687 // For the XFA fields, we need to return the specific field type highlight
688 // colour or the general XFA field type colour if present.
689 if (IsFormFieldTypeXFA(fieldType)) {
690 if (!m_NeedsHighlight[static_cast<size_t>(fieldType)] &&
691 m_NeedsHighlight[static_cast<size_t>(FormFieldType::kXFA)]) {
692 return m_HighlightColor[static_cast<size_t>(FormFieldType::kXFA)];
693 }
694 }
695#endif // PDF_ENABLE_XFA
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000696 return m_HighlightColor[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700697}