blob: d8f999ec722ab1e532c2ef0c20d978b2b9135c47 [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"
19#include "core/fpdfapi/parser/cpdf_document.h"
20#include "core/fpdfapi/parser/cpdf_stream.h"
dsinclair1727aee2016-09-29 13:12:56 -070021#include "core/fpdfdoc/cpdf_actionfields.h"
22#include "core/fpdfdoc/cpdf_interform.h"
Tom Sepezc22d6712018-06-05 22:33:31 +000023#include "core/fxcrt/autorestorer.h"
dsinclair74a34fc2016-09-29 16:41:42 -070024#include "core/fxge/cfx_graphstatedata.h"
25#include "core/fxge/cfx_pathdata.h"
Dan Sinclair7d125322018-03-28 18:49:34 +000026#include "fpdfsdk/cpdfsdk_actionhandler.h"
dsinclair114e46a2016-09-29 17:18:21 -070027#include "fpdfsdk/cpdfsdk_annot.h"
Dan Sinclaircbf76e62018-03-28 21:00:35 +000028#include "fpdfsdk/cpdfsdk_annotiterator.h"
dsinclair735606d2016-10-05 15:47:02 -070029#include "fpdfsdk/cpdfsdk_formfillenvironment.h"
Dan Sinclair00d47a62018-03-28 18:39:04 +000030#include "fpdfsdk/cpdfsdk_helpers.h"
dsinclair114e46a2016-09-29 17:18:21 -070031#include "fpdfsdk/cpdfsdk_pageview.h"
32#include "fpdfsdk/cpdfsdk_widget.h"
jaepark611adb82016-08-17 11:34:36 -070033#include "fpdfsdk/formfiller/cffl_formfiller.h"
dsinclair114e46a2016-09-29 17:18:21 -070034#include "fpdfsdk/ipdfsdk_annothandler.h"
Dan Sinclaire0345a42017-10-30 20:20:42 +000035#include "fxjs/ijs_event_context.h"
36#include "fxjs/ijs_runtime.h"
jaepark611adb82016-08-17 11:34:36 -070037#include "third_party/base/stl_util.h"
38
39#ifdef PDF_ENABLE_XFA
dsinclair114e46a2016-09-29 17:18:21 -070040#include "fpdfsdk/cpdfsdk_xfawidget.h"
dsinclair521b7502016-11-02 13:02:28 -070041#include "fpdfsdk/fpdfxfa/cpdfxfa_context.h"
dsinclair4d29e782016-10-04 14:02:47 -070042#include "fpdfsdk/fpdfxfa/cxfa_fwladaptertimermgr.h"
dsinclair5b493092016-09-29 20:20:24 -070043#include "xfa/fxfa/cxfa_eventparam.h"
Dan Sinclair80c48782017-03-23 12:11:20 -040044#include "xfa/fxfa/cxfa_ffdocview.h"
45#include "xfa/fxfa/cxfa_ffwidget.h"
46#include "xfa/fxfa/cxfa_ffwidgethandler.h"
jaepark611adb82016-08-17 11:34:36 -070047#endif // PDF_ENABLE_XFA
48
Ryan Harrison9baf31f2018-01-12 18:36:30 +000049namespace {
50
Lei Zhang4f7479a2018-03-21 13:37:06 +000051constexpr uint32_t kWhiteBGR = FXSYS_BGR(255, 255, 255);
52
Ryan Harrison9baf31f2018-01-12 18:36:30 +000053bool IsFormFieldTypeComboOrText(FormFieldType fieldType) {
54 switch (fieldType) {
55 case FormFieldType::kComboBox:
56 case FormFieldType::kTextField:
57 return true;
58 default:
59 return false;
60 }
61}
62
Ryan Harrison2056fac2018-01-16 16:06:45 +000063#ifdef PDF_ENABLE_XFA
64bool IsFormFieldTypeXFA(FormFieldType fieldType) {
65 switch (fieldType) {
66 case FormFieldType::kXFA:
67 case FormFieldType::kXFA_CheckBox:
68 case FormFieldType::kXFA_ComboBox:
69 case FormFieldType::kXFA_ImageField:
70 case FormFieldType::kXFA_ListBox:
71 case FormFieldType::kXFA_PushButton:
72 case FormFieldType::kXFA_Signature:
73 case FormFieldType::kXFA_TextField:
74 return true;
75 default:
76 return false;
77 }
78}
79#endif // PDF_ENABLE_XFA
80
Lei Zhang2ee811f2018-08-13 21:32:50 +000081bool FDFToURLEncodedData(std::vector<uint8_t>* pBuffer) {
Lei Zhangf0260b22018-08-13 22:50:02 +000082 std::unique_ptr<CFDF_Document> pFDF = CFDF_Document::ParseMemory(*pBuffer);
Lei Zhang4f1aa692018-08-13 21:31:18 +000083 if (!pFDF)
84 return true;
85
86 CPDF_Dictionary* pMainDict = pFDF->GetRoot()->GetDictFor("FDF");
87 if (!pMainDict)
88 return false;
89
90 CPDF_Array* pFields = pMainDict->GetArrayFor("Fields");
91 if (!pFields)
92 return false;
93
94 std::ostringstream fdfEncodedData;
95 for (uint32_t i = 0; i < pFields->GetCount(); i++) {
96 CPDF_Dictionary* pField = pFields->GetDictAt(i);
97 if (!pField)
98 continue;
99 WideString name;
100 name = pField->GetUnicodeTextFor("T");
101 ByteString name_b = name.ToDefANSI();
102 ByteString csBValue = pField->GetStringFor("V");
103 WideString csWValue = PDF_DecodeText(csBValue);
104 ByteString csValue_b = csWValue.ToDefANSI();
105 fdfEncodedData << name_b << "=" << csValue_b;
106 if (i != pFields->GetCount() - 1)
107 fdfEncodedData << "&";
108 }
109
Lei Zhang2ee811f2018-08-13 21:32:50 +0000110 size_t nBufSize = fdfEncodedData.tellp();
Lei Zhang4f1aa692018-08-13 21:31:18 +0000111 if (nBufSize <= 0)
112 return false;
113
Lei Zhang2ee811f2018-08-13 21:32:50 +0000114 pBuffer->resize(nBufSize);
115 memcpy(pBuffer->data(), fdfEncodedData.str().c_str(), nBufSize);
Lei Zhang4f1aa692018-08-13 21:31:18 +0000116 return true;
117}
118
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000119} // namespace
120
dsinclair690c0332016-10-11 09:13:01 -0700121CPDFSDK_InterForm::CPDFSDK_InterForm(CPDFSDK_FormFillEnvironment* pFormFillEnv)
122 : m_pFormFillEnv(pFormFillEnv),
Dan Sinclair0bb13332017-03-30 16:12:02 -0400123 m_pInterForm(
124 pdfium::MakeUnique<CPDF_InterForm>(m_pFormFillEnv->GetPDFDocument())),
jaepark611adb82016-08-17 11:34:36 -0700125#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700126 m_bXfaCalculate(true),
127 m_bXfaValidationsEnabled(true),
jaepark611adb82016-08-17 11:34:36 -0700128#endif // PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700129 m_bCalculate(true),
130 m_bBusy(false),
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000131 m_HighlightAlpha(0) {
jaepark611adb82016-08-17 11:34:36 -0700132 m_pInterForm->SetFormNotify(this);
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000133 RemoveAllHighLights();
jaepark611adb82016-08-17 11:34:36 -0700134}
135
136CPDFSDK_InterForm::~CPDFSDK_InterForm() {
137 m_Map.clear();
138#ifdef PDF_ENABLE_XFA
139 m_XFAMap.clear();
140#endif // PDF_ENABLE_XFA
141}
142
tsepez4cf55152016-11-02 14:37:54 -0700143bool CPDFSDK_InterForm::HighlightWidgets() {
144 return false;
jaepark611adb82016-08-17 11:34:36 -0700145}
146
147CPDFSDK_Widget* CPDFSDK_InterForm::GetSibling(CPDFSDK_Widget* pWidget,
tsepez4cf55152016-11-02 14:37:54 -0700148 bool bNext) const {
Dan Sinclaircbf76e62018-03-28 21:00:35 +0000149 auto pIterator = pdfium::MakeUnique<CPDFSDK_AnnotIterator>(
Dan Sinclair0bb13332017-03-30 16:12:02 -0400150 pWidget->GetPageView(), CPDF_Annot::Subtype::WIDGET);
jaepark611adb82016-08-17 11:34:36 -0700151
Tom Sepez4ef943b2018-07-26 19:06:06 +0000152 return ToCPDFSDKWidget(bNext ? pIterator->GetNextAnnot(pWidget)
153 : pIterator->GetPrevAnnot(pWidget));
jaepark611adb82016-08-17 11:34:36 -0700154}
155
dsinclairc5267c52016-11-04 15:35:12 -0700156CPDFSDK_Widget* CPDFSDK_InterForm::GetWidget(CPDF_FormControl* pControl) const {
jaepark611adb82016-08-17 11:34:36 -0700157 if (!pControl || !m_pInterForm)
158 return nullptr;
159
160 CPDFSDK_Widget* pWidget = nullptr;
161 const auto it = m_Map.find(pControl);
162 if (it != m_Map.end())
163 pWidget = it->second;
164 if (pWidget)
165 return pWidget;
jaepark611adb82016-08-17 11:34:36 -0700166
167 CPDF_Dictionary* pControlDict = pControl->GetWidget();
dsinclair7cbe68e2016-10-12 11:56:23 -0700168 CPDF_Document* pDocument = m_pFormFillEnv->GetPDFDocument();
jaepark611adb82016-08-17 11:34:36 -0700169 CPDFSDK_PageView* pPage = nullptr;
170
dsinclair38fd8442016-09-15 10:15:32 -0700171 if (CPDF_Dictionary* pPageDict = pControlDict->GetDictFor("P")) {
jaepark611adb82016-08-17 11:34:36 -0700172 int nPageIndex = pDocument->GetPageIndex(pPageDict->GetObjNum());
173 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700174 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700175 }
176
177 if (!pPage) {
178 int nPageIndex = GetPageIndexByAnnotDict(pDocument, pControlDict);
179 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700180 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700181 }
182
Tom Sepez4ef943b2018-07-26 19:06:06 +0000183 return pPage ? ToCPDFSDKWidget(pPage->GetAnnotByDict(pControlDict)) : nullptr;
jaepark611adb82016-08-17 11:34:36 -0700184}
185
186void CPDFSDK_InterForm::GetWidgets(
Ryan Harrison275e2602017-09-18 14:23:18 -0400187 const WideString& sFieldName,
tsepez8fa82792017-01-11 09:32:33 -0800188 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700189 for (int i = 0, sz = m_pInterForm->CountFields(sFieldName); i < sz; ++i) {
190 CPDF_FormField* pFormField = m_pInterForm->GetField(i, sFieldName);
191 ASSERT(pFormField);
192 GetWidgets(pFormField, widgets);
193 }
194}
195
196void CPDFSDK_InterForm::GetWidgets(
197 CPDF_FormField* pField,
tsepez8fa82792017-01-11 09:32:33 -0800198 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700199 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
200 CPDF_FormControl* pFormCtrl = pField->GetControl(i);
201 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700202 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
jaepark611adb82016-08-17 11:34:36 -0700203 if (pWidget)
tsepez8fa82792017-01-11 09:32:33 -0800204 widgets->emplace_back(pWidget);
jaepark611adb82016-08-17 11:34:36 -0700205 }
206}
207
208int CPDFSDK_InterForm::GetPageIndexByAnnotDict(
209 CPDF_Document* pDocument,
210 CPDF_Dictionary* pAnnotDict) const {
211 ASSERT(pAnnotDict);
212
213 for (int i = 0, sz = pDocument->GetPageCount(); i < sz; i++) {
Tom Sepez967aa072018-05-08 13:40:20 +0000214 if (CPDF_Dictionary* pPageDict = pDocument->GetPageDictionary(i)) {
dsinclair38fd8442016-09-15 10:15:32 -0700215 if (CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots")) {
jaepark611adb82016-08-17 11:34:36 -0700216 for (int j = 0, jsz = pAnnots->GetCount(); j < jsz; j++) {
217 CPDF_Object* pDict = pAnnots->GetDirectObjectAt(j);
218 if (pAnnotDict == pDict)
219 return i;
220 }
221 }
222 }
223 }
224
225 return -1;
226}
227
228void CPDFSDK_InterForm::AddMap(CPDF_FormControl* pControl,
229 CPDFSDK_Widget* pWidget) {
230 m_Map[pControl] = pWidget;
231}
232
233void CPDFSDK_InterForm::RemoveMap(CPDF_FormControl* pControl) {
234 m_Map.erase(pControl);
235}
236
tsepez4cf55152016-11-02 14:37:54 -0700237void CPDFSDK_InterForm::EnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700238 m_bCalculate = bEnabled;
239}
240
tsepez4cf55152016-11-02 14:37:54 -0700241bool CPDFSDK_InterForm::IsCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700242 return m_bCalculate;
243}
244
245#ifdef PDF_ENABLE_XFA
246void CPDFSDK_InterForm::AddXFAMap(CXFA_FFWidget* hWidget,
247 CPDFSDK_XFAWidget* pWidget) {
248 ASSERT(hWidget);
249 m_XFAMap[hWidget] = pWidget;
250}
251
252void CPDFSDK_InterForm::RemoveXFAMap(CXFA_FFWidget* hWidget) {
253 ASSERT(hWidget);
254 m_XFAMap.erase(hWidget);
255}
256
257CPDFSDK_XFAWidget* CPDFSDK_InterForm::GetXFAWidget(CXFA_FFWidget* hWidget) {
258 ASSERT(hWidget);
259 auto it = m_XFAMap.find(hWidget);
260 return it != m_XFAMap.end() ? it->second : nullptr;
261}
262
tsepez4cf55152016-11-02 14:37:54 -0700263void CPDFSDK_InterForm::XfaEnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700264 m_bXfaCalculate = bEnabled;
265}
tsepez4cf55152016-11-02 14:37:54 -0700266bool CPDFSDK_InterForm::IsXfaCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700267 return m_bXfaCalculate;
268}
269
tsepez4cf55152016-11-02 14:37:54 -0700270bool CPDFSDK_InterForm::IsXfaValidationsEnabled() {
jaepark611adb82016-08-17 11:34:36 -0700271 return m_bXfaValidationsEnabled;
272}
tsepez4cf55152016-11-02 14:37:54 -0700273void CPDFSDK_InterForm::XfaSetValidationsEnabled(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700274 m_bXfaValidationsEnabled = bEnabled;
275}
276
Dan Sinclairce047a62018-01-30 18:15:02 +0000277void CPDFSDK_InterForm::SynchronizeField(CPDF_FormField* pFormField) {
jaepark611adb82016-08-17 11:34:36 -0700278 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
279 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
dsinclairc5267c52016-11-04 15:35:12 -0700280 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
Dan Sinclairce047a62018-01-30 18:15:02 +0000281 pWidget->Synchronize(false);
jaepark611adb82016-08-17 11:34:36 -0700282 }
283}
284#endif // PDF_ENABLE_XFA
285
286void CPDFSDK_InterForm::OnCalculate(CPDF_FormField* pFormField) {
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000287 if (!m_pFormFillEnv->IsJSPlatformPresent())
jaepark611adb82016-08-17 11:34:36 -0700288 return;
289
290 if (m_bBusy)
291 return;
292
Tom Sepezc22d6712018-06-05 22:33:31 +0000293 AutoRestorer<bool> restorer(&m_bBusy);
tsepez4cf55152016-11-02 14:37:54 -0700294 m_bBusy = true;
jaepark611adb82016-08-17 11:34:36 -0700295
Tom Sepezc22d6712018-06-05 22:33:31 +0000296 if (!IsCalculateEnabled())
jaepark611adb82016-08-17 11:34:36 -0700297 return;
jaepark611adb82016-08-17 11:34:36 -0700298
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000299 IJS_Runtime* pRuntime = m_pFormFillEnv->GetIJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700300 int nSize = m_pInterForm->CountFieldsInCalculationOrder();
301 for (int i = 0; i < nSize; i++) {
302 CPDF_FormField* pField = m_pInterForm->GetFieldInCalculationOrder(i);
303 if (!pField)
304 continue;
305
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000306 FormFieldType fieldType = pField->GetFieldType();
307 if (!IsFormFieldTypeComboOrText(fieldType))
jaepark611adb82016-08-17 11:34:36 -0700308 continue;
309
310 CPDF_AAction aAction = pField->GetAdditionalAction();
311 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Calculate))
312 continue;
313
314 CPDF_Action action = aAction.GetAction(CPDF_AAction::Calculate);
315 if (!action.GetDict())
316 continue;
317
Ryan Harrison275e2602017-09-18 14:23:18 -0400318 WideString csJS = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700319 if (csJS.IsEmpty())
320 continue;
321
Ryan Harrison275e2602017-09-18 14:23:18 -0400322 WideString sOldValue = pField->GetValue();
323 WideString sValue = sOldValue;
tsepez4cf55152016-11-02 14:37:54 -0700324 bool bRC = true;
Tom Sepezc22d6712018-06-05 22:33:31 +0000325 IJS_Runtime::ScopedEventContext pContext(pRuntime);
Tom Sepez3d813152018-06-21 17:28:24 +0000326 pContext->OnField_Calculate(pFormField, pField, &sValue, &bRC);
jaepark611adb82016-08-17 11:34:36 -0700327
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000328 Optional<IJS_Runtime::JS_Error> err = pContext->RunScript(csJS);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000329 if (!err && bRC && sValue.Compare(sOldValue) != 0)
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000330 pField->SetValue(sValue, NotificationOption::kNotify);
jaepark611adb82016-08-17 11:34:36 -0700331 }
jaepark611adb82016-08-17 11:34:36 -0700332}
333
Ryan Harrison275e2602017-09-18 14:23:18 -0400334WideString CPDFSDK_InterForm::OnFormat(CPDF_FormField* pFormField,
335 bool& bFormatted) {
336 WideString sValue = pFormField->GetValue();
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000337 if (!m_pFormFillEnv->IsJSPlatformPresent()) {
tsepez4cf55152016-11-02 14:37:54 -0700338 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700339 return sValue;
340 }
341
Tom Sepezf5ca90c2018-02-01 02:15:44 +0000342 IJS_Runtime* pRuntime = m_pFormFillEnv->GetIJSRuntime();
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000343 if (pFormField->GetFieldType() == FormFieldType::kComboBox &&
jaepark611adb82016-08-17 11:34:36 -0700344 pFormField->CountSelectedItems() > 0) {
345 int index = pFormField->GetSelectedIndex(0);
346 if (index >= 0)
347 sValue = pFormField->GetOptionLabel(index);
348 }
349
tsepez4cf55152016-11-02 14:37:54 -0700350 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700351
352 CPDF_AAction aAction = pFormField->GetAdditionalAction();
353 if (aAction.GetDict() && aAction.ActionExist(CPDF_AAction::Format)) {
354 CPDF_Action action = aAction.GetAction(CPDF_AAction::Format);
355 if (action.GetDict()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400356 WideString script = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700357 if (!script.IsEmpty()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400358 WideString Value = sValue;
Tom Sepezc22d6712018-06-05 22:33:31 +0000359 IJS_Runtime::ScopedEventContext pContext(pRuntime);
Tom Sepez3d813152018-06-21 17:28:24 +0000360 pContext->OnField_Format(pFormField, &Value, true);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000361 Optional<IJS_Runtime::JS_Error> err = pContext->RunScript(script);
Dan Sinclairdc5d88b2018-05-17 13:53:52 +0000362 if (!err) {
Tom Sepeze005dc32018-06-19 17:33:32 +0000363 sValue = std::move(Value);
tsepez4cf55152016-11-02 14:37:54 -0700364 bFormatted = true;
jaepark611adb82016-08-17 11:34:36 -0700365 }
366 }
367 }
368 }
jaepark611adb82016-08-17 11:34:36 -0700369 return sValue;
370}
371
372void CPDFSDK_InterForm::ResetFieldAppearance(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400373 const WideString* sValue,
tsepez4cf55152016-11-02 14:37:54 -0700374 bool bValueChanged) {
jaepark611adb82016-08-17 11:34:36 -0700375 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
376 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
377 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700378 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700379 pWidget->ResetAppearance(sValue, bValueChanged);
380 }
381}
382
383void CPDFSDK_InterForm::UpdateField(CPDF_FormField* pFormField) {
Lei Zhang375c2762017-03-10 14:37:14 -0800384 auto* formfiller = m_pFormFillEnv->GetInteractiveFormFiller();
jaepark611adb82016-08-17 11:34:36 -0700385 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
386 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
387 ASSERT(pFormCtrl);
388
Lei Zhang77f9bff2017-08-29 11:34:12 -0700389 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
390 if (!pWidget)
391 continue;
392
Tom Sepez101535f2018-06-12 13:36:05 +0000393 IPDF_Page* pPage = pWidget->GetPage();
Lei Zhang77f9bff2017-08-29 11:34:12 -0700394 FX_RECT rect = formfiller->GetViewBBox(
395 m_pFormFillEnv->GetPageView(pPage, false), pWidget);
396 m_pFormFillEnv->Invalidate(pPage, rect);
jaepark611adb82016-08-17 11:34:36 -0700397 }
398}
399
tsepez4cf55152016-11-02 14:37:54 -0700400bool CPDFSDK_InterForm::OnKeyStrokeCommit(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400401 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700402 CPDF_AAction aAction = pFormField->GetAdditionalAction();
403 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::KeyStroke))
tsepez4cf55152016-11-02 14:37:54 -0700404 return true;
jaepark611adb82016-08-17 11:34:36 -0700405
406 CPDF_Action action = aAction.GetAction(CPDF_AAction::KeyStroke);
407 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700408 return true;
jaepark611adb82016-08-17 11:34:36 -0700409
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700410 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
Dan Sinclair7d125322018-03-28 18:49:34 +0000411 CPDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700412 fa.bModifier = false;
413 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700414 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700415 pActionHandler->DoAction_FieldJavaScript(
Dan Sinclair8cdea722018-01-30 18:56:50 +0000416 action, CPDF_AAction::KeyStroke, m_pFormFillEnv.Get(), pFormField, &fa);
jaepark611adb82016-08-17 11:34:36 -0700417 return fa.bRC;
418}
419
tsepez4cf55152016-11-02 14:37:54 -0700420bool CPDFSDK_InterForm::OnValidate(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400421 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700422 CPDF_AAction aAction = pFormField->GetAdditionalAction();
423 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Validate))
tsepez4cf55152016-11-02 14:37:54 -0700424 return true;
jaepark611adb82016-08-17 11:34:36 -0700425
426 CPDF_Action action = aAction.GetAction(CPDF_AAction::Validate);
427 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700428 return true;
jaepark611adb82016-08-17 11:34:36 -0700429
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700430 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
Dan Sinclair7d125322018-03-28 18:49:34 +0000431 CPDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700432 fa.bModifier = false;
433 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700434 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700435 pActionHandler->DoAction_FieldJavaScript(
Dan Sinclair8cdea722018-01-30 18:56:50 +0000436 action, CPDF_AAction::Validate, m_pFormFillEnv.Get(), pFormField, &fa);
jaepark611adb82016-08-17 11:34:36 -0700437 return fa.bRC;
438}
439
tsepez4cf55152016-11-02 14:37:54 -0700440bool CPDFSDK_InterForm::DoAction_Hide(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700441 ASSERT(action.GetDict());
442
443 CPDF_ActionFields af(&action);
Lei Zhang5cee3f22018-05-25 21:48:49 +0000444 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700445 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
446
447 bool bHide = action.GetHideStatus();
tsepez4cf55152016-11-02 14:37:54 -0700448 bool bChanged = false;
jaepark611adb82016-08-17 11:34:36 -0700449
450 for (CPDF_FormField* pField : fields) {
451 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
452 CPDF_FormControl* pControl = pField->GetControl(i);
453 ASSERT(pControl);
454
dsinclairc5267c52016-11-04 15:35:12 -0700455 if (CPDFSDK_Widget* pWidget = GetWidget(pControl)) {
jaepark611adb82016-08-17 11:34:36 -0700456 uint32_t nFlags = pWidget->GetFlags();
457 nFlags &= ~ANNOTFLAG_INVISIBLE;
458 nFlags &= ~ANNOTFLAG_NOVIEW;
459 if (bHide)
460 nFlags |= ANNOTFLAG_HIDDEN;
461 else
462 nFlags &= ~ANNOTFLAG_HIDDEN;
463 pWidget->SetFlags(nFlags);
464 pWidget->GetPageView()->UpdateView(pWidget);
tsepez4cf55152016-11-02 14:37:54 -0700465 bChanged = true;
jaepark611adb82016-08-17 11:34:36 -0700466 }
467 }
468 }
469
470 return bChanged;
471}
472
tsepez4cf55152016-11-02 14:37:54 -0700473bool CPDFSDK_InterForm::DoAction_SubmitForm(const CPDF_Action& action) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400474 WideString sDestination = action.GetFilePath();
jaepark611adb82016-08-17 11:34:36 -0700475 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700476 return false;
jaepark611adb82016-08-17 11:34:36 -0700477
Lei Zhang5cee3f22018-05-25 21:48:49 +0000478 const CPDF_Dictionary* pActionDict = action.GetDict();
jaepark611adb82016-08-17 11:34:36 -0700479 if (pActionDict->KeyExist("Fields")) {
480 CPDF_ActionFields af(&action);
481 uint32_t dwFlags = action.GetFlags();
Lei Zhang5cee3f22018-05-25 21:48:49 +0000482 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700483 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
484 if (!fields.empty()) {
485 bool bIncludeOrExclude = !(dwFlags & 0x01);
Nicolas Penaa478dc52017-01-23 15:48:51 -0500486 if (!m_pInterForm->CheckRequiredFields(&fields, bIncludeOrExclude))
tsepez4cf55152016-11-02 14:37:54 -0700487 return false;
jaepark611adb82016-08-17 11:34:36 -0700488
489 return SubmitFields(sDestination, fields, bIncludeOrExclude, false);
490 }
491 }
Nicolas Penaa478dc52017-01-23 15:48:51 -0500492 if (!m_pInterForm->CheckRequiredFields(nullptr, true))
tsepez4cf55152016-11-02 14:37:54 -0700493 return false;
jaepark611adb82016-08-17 11:34:36 -0700494
tsepez4cf55152016-11-02 14:37:54 -0700495 return SubmitForm(sDestination, false);
jaepark611adb82016-08-17 11:34:36 -0700496}
497
Ryan Harrison275e2602017-09-18 14:23:18 -0400498bool CPDFSDK_InterForm::SubmitFields(const WideString& csDestination,
tsepez4cf55152016-11-02 14:37:54 -0700499 const std::vector<CPDF_FormField*>& fields,
500 bool bIncludeOrExclude,
501 bool bUrlEncoded) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400502 ByteString textBuf = ExportFieldsToFDFTextBuf(fields, bIncludeOrExclude);
Tom Sepez65be34f2018-09-14 20:58:36 +0000503 if (textBuf.IsEmpty())
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400504 return false;
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400505
Tom Sepez65be34f2018-09-14 20:58:36 +0000506 std::vector<uint8_t> buffer(textBuf.begin(), textBuf.end());
Lei Zhang2ee811f2018-08-13 21:32:50 +0000507 if (bUrlEncoded && !FDFToURLEncodedData(&buffer))
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400508 return false;
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400509
Lei Zhang2ee811f2018-08-13 21:32:50 +0000510 m_pFormFillEnv->JS_docSubmitForm(buffer.data(), buffer.size(), csDestination);
tsepez4cf55152016-11-02 14:37:54 -0700511 return true;
jaepark611adb82016-08-17 11:34:36 -0700512}
513
Ryan Harrison275e2602017-09-18 14:23:18 -0400514ByteString CPDFSDK_InterForm::ExportFieldsToFDFTextBuf(
jaepark611adb82016-08-17 11:34:36 -0700515 const std::vector<CPDF_FormField*>& fields,
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400516 bool bIncludeOrExclude) {
Tom Sepez690d4562017-05-18 11:42:46 -0700517 std::unique_ptr<CFDF_Document> pFDF = m_pInterForm->ExportToFDF(
518 m_pFormFillEnv->JS_docGetFilePath(), fields, bIncludeOrExclude, false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400519
Ryan Harrison275e2602017-09-18 14:23:18 -0400520 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700521}
522
Ryan Harrison275e2602017-09-18 14:23:18 -0400523bool CPDFSDK_InterForm::SubmitForm(const WideString& sDestination,
tsepez4cf55152016-11-02 14:37:54 -0700524 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700525 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700526 return false;
jaepark611adb82016-08-17 11:34:36 -0700527
dsinclair7cbe68e2016-10-12 11:56:23 -0700528 if (!m_pFormFillEnv || !m_pInterForm)
tsepez4cf55152016-11-02 14:37:54 -0700529 return false;
jaepark611adb82016-08-17 11:34:36 -0700530
Tom Sepez690d4562017-05-18 11:42:46 -0700531 std::unique_ptr<CFDF_Document> pFDFDoc =
532 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
jaepark611adb82016-08-17 11:34:36 -0700533 if (!pFDFDoc)
tsepez4cf55152016-11-02 14:37:54 -0700534 return false;
jaepark611adb82016-08-17 11:34:36 -0700535
Ryan Harrison275e2602017-09-18 14:23:18 -0400536 ByteString fdfBuffer = pFDFDoc->WriteToString();
Ryan Harrison875e98c2017-09-27 10:53:11 -0400537 if (fdfBuffer.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700538 return false;
jaepark611adb82016-08-17 11:34:36 -0700539
Tom Sepez65be34f2018-09-14 20:58:36 +0000540 std::vector<uint8_t> buffer(fdfBuffer.begin(), fdfBuffer.end());
Lei Zhang2ee811f2018-08-13 21:32:50 +0000541 if (bUrlEncoded && !FDFToURLEncodedData(&buffer))
tsepez4cf55152016-11-02 14:37:54 -0700542 return false;
jaepark611adb82016-08-17 11:34:36 -0700543
Lei Zhang2ee811f2018-08-13 21:32:50 +0000544 m_pFormFillEnv->JS_docSubmitForm(buffer.data(), buffer.size(), sDestination);
tsepez4cf55152016-11-02 14:37:54 -0700545 return true;
jaepark611adb82016-08-17 11:34:36 -0700546}
547
Ryan Harrison275e2602017-09-18 14:23:18 -0400548ByteString CPDFSDK_InterForm::ExportFormToFDFTextBuf() {
Tom Sepez690d4562017-05-18 11:42:46 -0700549 std::unique_ptr<CFDF_Document> pFDF =
550 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400551
Ryan Harrison275e2602017-09-18 14:23:18 -0400552 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700553}
554
dan sinclair7544a4b2018-03-08 15:19:50 +0000555void CPDFSDK_InterForm::DoAction_ResetForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700556 ASSERT(action.GetDict());
Lei Zhang5cee3f22018-05-25 21:48:49 +0000557 const CPDF_Dictionary* pActionDict = action.GetDict();
dan sinclair7544a4b2018-03-08 15:19:50 +0000558 if (!pActionDict->KeyExist("Fields")) {
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000559 m_pInterForm->ResetForm(NotificationOption::kNotify);
dan sinclair7544a4b2018-03-08 15:19:50 +0000560 return;
561 }
jaepark611adb82016-08-17 11:34:36 -0700562 CPDF_ActionFields af(&action);
563 uint32_t dwFlags = action.GetFlags();
Lei Zhang5cee3f22018-05-25 21:48:49 +0000564 std::vector<const CPDF_Object*> fieldObjects = af.GetAllFields();
jaepark611adb82016-08-17 11:34:36 -0700565 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
Tom Sepezb2e6b4c2018-08-16 20:53:58 +0000566 m_pInterForm->ResetForm(fields, !(dwFlags & 0x01),
567 NotificationOption::kNotify);
jaepark611adb82016-08-17 11:34:36 -0700568}
569
jaepark611adb82016-08-17 11:34:36 -0700570std::vector<CPDF_FormField*> CPDFSDK_InterForm::GetFieldFromObjects(
Lei Zhang5cee3f22018-05-25 21:48:49 +0000571 const std::vector<const CPDF_Object*>& objects) const {
jaepark611adb82016-08-17 11:34:36 -0700572 std::vector<CPDF_FormField*> fields;
Lei Zhang5cee3f22018-05-25 21:48:49 +0000573 for (const CPDF_Object* pObject : objects) {
Dan Sinclairce047a62018-01-30 18:15:02 +0000574 if (!pObject || !pObject->IsString())
575 continue;
576
577 WideString csName = pObject->GetUnicodeText();
578 CPDF_FormField* pField = m_pInterForm->GetField(0, csName);
579 if (pField)
580 fields.push_back(pField);
jaepark611adb82016-08-17 11:34:36 -0700581 }
582 return fields;
583}
584
dan sinclair507fb4e2018-03-08 15:14:09 +0000585bool CPDFSDK_InterForm::BeforeValueChange(CPDF_FormField* pField,
586 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000587 FormFieldType fieldType = pField->GetFieldType();
588 if (!IsFormFieldTypeComboOrText(fieldType))
dan sinclair507fb4e2018-03-08 15:14:09 +0000589 return true;
jaepark611adb82016-08-17 11:34:36 -0700590 if (!OnKeyStrokeCommit(pField, csValue))
dan sinclair507fb4e2018-03-08 15:14:09 +0000591 return false;
592 return OnValidate(pField, csValue);
jaepark611adb82016-08-17 11:34:36 -0700593}
594
595void CPDFSDK_InterForm::AfterValueChange(CPDF_FormField* pField) {
596#ifdef PDF_ENABLE_XFA
Dan Sinclairce047a62018-01-30 18:15:02 +0000597 SynchronizeField(pField);
jaepark611adb82016-08-17 11:34:36 -0700598#endif // PDF_ENABLE_XFA
Dan Sinclairce047a62018-01-30 18:15:02 +0000599
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000600 FormFieldType fieldType = pField->GetFieldType();
Dan Sinclairce047a62018-01-30 18:15:02 +0000601 if (!IsFormFieldTypeComboOrText(fieldType))
602 return;
603
604 OnCalculate(pField);
605 bool bFormatted = false;
606 WideString sValue = OnFormat(pField, bFormatted);
607 ResetFieldAppearance(pField, bFormatted ? &sValue : nullptr, true);
608 UpdateField(pField);
jaepark611adb82016-08-17 11:34:36 -0700609}
610
dan sinclair507fb4e2018-03-08 15:14:09 +0000611bool CPDFSDK_InterForm::BeforeSelectionChange(CPDF_FormField* pField,
612 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000613 if (pField->GetFieldType() != FormFieldType::kListBox)
dan sinclair507fb4e2018-03-08 15:14:09 +0000614 return true;
jaepark611adb82016-08-17 11:34:36 -0700615 if (!OnKeyStrokeCommit(pField, csValue))
dan sinclair507fb4e2018-03-08 15:14:09 +0000616 return false;
617 return OnValidate(pField, csValue);
jaepark611adb82016-08-17 11:34:36 -0700618}
619
620void CPDFSDK_InterForm::AfterSelectionChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000621 if (pField->GetFieldType() != FormFieldType::kListBox)
jaepark611adb82016-08-17 11:34:36 -0700622 return;
623
624 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700625 ResetFieldAppearance(pField, nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700626 UpdateField(pField);
627}
628
629void CPDFSDK_InterForm::AfterCheckedStatusChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000630 FormFieldType fieldType = pField->GetFieldType();
631 if (fieldType != FormFieldType::kCheckBox &&
632 fieldType != FormFieldType::kRadioButton)
jaepark611adb82016-08-17 11:34:36 -0700633 return;
634
635 OnCalculate(pField);
636 UpdateField(pField);
637}
638
jaepark611adb82016-08-17 11:34:36 -0700639void CPDFSDK_InterForm::AfterFormReset(CPDF_InterForm* pForm) {
640 OnCalculate(nullptr);
641}
642
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000643bool CPDFSDK_InterForm::IsNeedHighLight(FormFieldType fieldType) {
644 if (fieldType == FormFieldType::kUnknown)
tsepez4cf55152016-11-02 14:37:54 -0700645 return false;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000646
Ryan Harrison2056fac2018-01-16 16:06:45 +0000647#ifdef PDF_ENABLE_XFA
648 // For the XFA fields, we need to return if the specific field type has
649 // highlight enabled or if the general XFA field type has it enabled.
650 if (IsFormFieldTypeXFA(fieldType)) {
651 if (!m_NeedsHighlight[static_cast<size_t>(fieldType)])
652 return m_NeedsHighlight[static_cast<size_t>(FormFieldType::kXFA)];
653 }
654#endif // PDF_ENABLE_XFA
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000655 return m_NeedsHighlight[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700656}
657
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000658void CPDFSDK_InterForm::RemoveAllHighLights() {
659 std::fill(m_HighlightColor, m_HighlightColor + kFormFieldTypeCount,
Lei Zhang4f7479a2018-03-21 13:37:06 +0000660 kWhiteBGR);
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000661 std::fill(m_NeedsHighlight, m_NeedsHighlight + kFormFieldTypeCount, false);
jaepark611adb82016-08-17 11:34:36 -0700662}
663
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000664void CPDFSDK_InterForm::SetHighlightColor(FX_COLORREF clr,
665 FormFieldType fieldType) {
666 if (fieldType == FormFieldType::kUnknown)
jaepark611adb82016-08-17 11:34:36 -0700667 return;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000668
669 m_HighlightColor[static_cast<size_t>(fieldType)] = clr;
670 m_NeedsHighlight[static_cast<size_t>(fieldType)] = true;
671}
672
673void CPDFSDK_InterForm::SetAllHighlightColors(FX_COLORREF clr) {
674 for (auto type : kFormFieldTypes) {
675 m_HighlightColor[static_cast<size_t>(type)] = clr;
676 m_NeedsHighlight[static_cast<size_t>(type)] = true;
jaepark611adb82016-08-17 11:34:36 -0700677 }
678}
679
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000680FX_COLORREF CPDFSDK_InterForm::GetHighlightColor(FormFieldType fieldType) {
681 if (fieldType == FormFieldType::kUnknown)
Lei Zhang4f7479a2018-03-21 13:37:06 +0000682 return kWhiteBGR;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000683
Ryan Harrison2056fac2018-01-16 16:06:45 +0000684#ifdef PDF_ENABLE_XFA
685 // For the XFA fields, we need to return the specific field type highlight
686 // colour or the general XFA field type colour if present.
687 if (IsFormFieldTypeXFA(fieldType)) {
688 if (!m_NeedsHighlight[static_cast<size_t>(fieldType)] &&
689 m_NeedsHighlight[static_cast<size_t>(FormFieldType::kXFA)]) {
690 return m_HighlightColor[static_cast<size_t>(FormFieldType::kXFA)];
691 }
692 }
693#endif // PDF_ENABLE_XFA
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000694 return m_HighlightColor[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700695}