blob: 04ccc35d569161901e0c6c6ba4db41f60ac935a7 [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>
thestig7c292e02016-09-28 14:14:26 -070013#include <vector>
jaepark611adb82016-08-17 11:34:36 -070014
dsinclair41872fa2016-10-04 11:29:35 -070015#include "core/fpdfapi/page/cpdf_page.h"
dsinclair488b7ad2016-10-04 11:55:50 -070016#include "core/fpdfapi/parser/cfdf_document.h"
17#include "core/fpdfapi/parser/cpdf_array.h"
18#include "core/fpdfapi/parser/cpdf_document.h"
19#include "core/fpdfapi/parser/cpdf_stream.h"
dsinclair1727aee2016-09-29 13:12:56 -070020#include "core/fpdfdoc/cpdf_actionfields.h"
21#include "core/fpdfdoc/cpdf_interform.h"
dsinclair74a34fc2016-09-29 16:41:42 -070022#include "core/fxge/cfx_graphstatedata.h"
23#include "core/fxge/cfx_pathdata.h"
24#include "core/fxge/cfx_renderdevice.h"
dsinclair114e46a2016-09-29 17:18:21 -070025#include "fpdfsdk/cba_annotiterator.h"
26#include "fpdfsdk/cpdfsdk_annot.h"
dsinclair735606d2016-10-05 15:47:02 -070027#include "fpdfsdk/cpdfsdk_formfillenvironment.h"
dsinclair114e46a2016-09-29 17:18:21 -070028#include "fpdfsdk/cpdfsdk_pageview.h"
29#include "fpdfsdk/cpdfsdk_widget.h"
jaepark611adb82016-08-17 11:34:36 -070030#include "fpdfsdk/formfiller/cffl_formfiller.h"
dsinclair114e46a2016-09-29 17:18:21 -070031#include "fpdfsdk/fsdk_actionhandler.h"
32#include "fpdfsdk/fsdk_define.h"
dsinclair114e46a2016-09-29 17:18:21 -070033#include "fpdfsdk/ipdfsdk_annothandler.h"
Dan Sinclaire0345a42017-10-30 20:20:42 +000034#include "fxjs/ijs_event_context.h"
35#include "fxjs/ijs_runtime.h"
jaepark611adb82016-08-17 11:34:36 -070036#include "third_party/base/stl_util.h"
37
38#ifdef PDF_ENABLE_XFA
dsinclair114e46a2016-09-29 17:18:21 -070039#include "fpdfsdk/cpdfsdk_xfawidget.h"
dsinclair521b7502016-11-02 13:02:28 -070040#include "fpdfsdk/fpdfxfa/cpdfxfa_context.h"
dsinclair4d29e782016-10-04 14:02:47 -070041#include "fpdfsdk/fpdfxfa/cxfa_fwladaptertimermgr.h"
dsinclair5b493092016-09-29 20:20:24 -070042#include "xfa/fxfa/cxfa_eventparam.h"
Dan Sinclair80c48782017-03-23 12:11:20 -040043#include "xfa/fxfa/cxfa_ffdocview.h"
44#include "xfa/fxfa/cxfa_ffwidget.h"
45#include "xfa/fxfa/cxfa_ffwidgethandler.h"
jaepark611adb82016-08-17 11:34:36 -070046#endif // PDF_ENABLE_XFA
47
Ryan Harrison9baf31f2018-01-12 18:36:30 +000048namespace {
49
50bool IsFormFieldTypeComboOrText(FormFieldType fieldType) {
51 switch (fieldType) {
52 case FormFieldType::kComboBox:
53 case FormFieldType::kTextField:
54 return true;
55 default:
56 return false;
57 }
58}
59
60} // namespace
61
dsinclair690c0332016-10-11 09:13:01 -070062CPDFSDK_InterForm::CPDFSDK_InterForm(CPDFSDK_FormFillEnvironment* pFormFillEnv)
63 : m_pFormFillEnv(pFormFillEnv),
Dan Sinclair0bb13332017-03-30 16:12:02 -040064 m_pInterForm(
65 pdfium::MakeUnique<CPDF_InterForm>(m_pFormFillEnv->GetPDFDocument())),
jaepark611adb82016-08-17 11:34:36 -070066#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070067 m_bXfaCalculate(true),
68 m_bXfaValidationsEnabled(true),
jaepark611adb82016-08-17 11:34:36 -070069#endif // PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070070 m_bCalculate(true),
71 m_bBusy(false),
Ryan Harrison9baf31f2018-01-12 18:36:30 +000072 m_HighlightAlpha(0) {
jaepark611adb82016-08-17 11:34:36 -070073 m_pInterForm->SetFormNotify(this);
Ryan Harrison9baf31f2018-01-12 18:36:30 +000074 RemoveAllHighLights();
jaepark611adb82016-08-17 11:34:36 -070075}
76
77CPDFSDK_InterForm::~CPDFSDK_InterForm() {
78 m_Map.clear();
79#ifdef PDF_ENABLE_XFA
80 m_XFAMap.clear();
81#endif // PDF_ENABLE_XFA
82}
83
tsepez4cf55152016-11-02 14:37:54 -070084bool CPDFSDK_InterForm::HighlightWidgets() {
85 return false;
jaepark611adb82016-08-17 11:34:36 -070086}
87
88CPDFSDK_Widget* CPDFSDK_InterForm::GetSibling(CPDFSDK_Widget* pWidget,
tsepez4cf55152016-11-02 14:37:54 -070089 bool bNext) const {
Dan Sinclair0bb13332017-03-30 16:12:02 -040090 auto pIterator = pdfium::MakeUnique<CBA_AnnotIterator>(
91 pWidget->GetPageView(), CPDF_Annot::Subtype::WIDGET);
jaepark611adb82016-08-17 11:34:36 -070092
93 if (bNext)
94 return static_cast<CPDFSDK_Widget*>(pIterator->GetNextAnnot(pWidget));
95
96 return static_cast<CPDFSDK_Widget*>(pIterator->GetPrevAnnot(pWidget));
97}
98
dsinclairc5267c52016-11-04 15:35:12 -070099CPDFSDK_Widget* CPDFSDK_InterForm::GetWidget(CPDF_FormControl* pControl) const {
jaepark611adb82016-08-17 11:34:36 -0700100 if (!pControl || !m_pInterForm)
101 return nullptr;
102
103 CPDFSDK_Widget* pWidget = nullptr;
104 const auto it = m_Map.find(pControl);
105 if (it != m_Map.end())
106 pWidget = it->second;
107 if (pWidget)
108 return pWidget;
jaepark611adb82016-08-17 11:34:36 -0700109
110 CPDF_Dictionary* pControlDict = pControl->GetWidget();
dsinclair7cbe68e2016-10-12 11:56:23 -0700111 CPDF_Document* pDocument = m_pFormFillEnv->GetPDFDocument();
jaepark611adb82016-08-17 11:34:36 -0700112 CPDFSDK_PageView* pPage = nullptr;
113
dsinclair38fd8442016-09-15 10:15:32 -0700114 if (CPDF_Dictionary* pPageDict = pControlDict->GetDictFor("P")) {
jaepark611adb82016-08-17 11:34:36 -0700115 int nPageIndex = pDocument->GetPageIndex(pPageDict->GetObjNum());
116 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700117 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700118 }
119
120 if (!pPage) {
121 int nPageIndex = GetPageIndexByAnnotDict(pDocument, pControlDict);
122 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700123 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700124 }
125
126 if (!pPage)
127 return nullptr;
128
129 return static_cast<CPDFSDK_Widget*>(pPage->GetAnnotByDict(pControlDict));
130}
131
132void CPDFSDK_InterForm::GetWidgets(
Ryan Harrison275e2602017-09-18 14:23:18 -0400133 const WideString& sFieldName,
tsepez8fa82792017-01-11 09:32:33 -0800134 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700135 for (int i = 0, sz = m_pInterForm->CountFields(sFieldName); i < sz; ++i) {
136 CPDF_FormField* pFormField = m_pInterForm->GetField(i, sFieldName);
137 ASSERT(pFormField);
138 GetWidgets(pFormField, widgets);
139 }
140}
141
142void CPDFSDK_InterForm::GetWidgets(
143 CPDF_FormField* pField,
tsepez8fa82792017-01-11 09:32:33 -0800144 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700145 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
146 CPDF_FormControl* pFormCtrl = pField->GetControl(i);
147 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700148 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
jaepark611adb82016-08-17 11:34:36 -0700149 if (pWidget)
tsepez8fa82792017-01-11 09:32:33 -0800150 widgets->emplace_back(pWidget);
jaepark611adb82016-08-17 11:34:36 -0700151 }
152}
153
154int CPDFSDK_InterForm::GetPageIndexByAnnotDict(
155 CPDF_Document* pDocument,
156 CPDF_Dictionary* pAnnotDict) const {
157 ASSERT(pAnnotDict);
158
159 for (int i = 0, sz = pDocument->GetPageCount(); i < sz; i++) {
160 if (CPDF_Dictionary* pPageDict = pDocument->GetPage(i)) {
dsinclair38fd8442016-09-15 10:15:32 -0700161 if (CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots")) {
jaepark611adb82016-08-17 11:34:36 -0700162 for (int j = 0, jsz = pAnnots->GetCount(); j < jsz; j++) {
163 CPDF_Object* pDict = pAnnots->GetDirectObjectAt(j);
164 if (pAnnotDict == pDict)
165 return i;
166 }
167 }
168 }
169 }
170
171 return -1;
172}
173
174void CPDFSDK_InterForm::AddMap(CPDF_FormControl* pControl,
175 CPDFSDK_Widget* pWidget) {
176 m_Map[pControl] = pWidget;
177}
178
179void CPDFSDK_InterForm::RemoveMap(CPDF_FormControl* pControl) {
180 m_Map.erase(pControl);
181}
182
tsepez4cf55152016-11-02 14:37:54 -0700183void CPDFSDK_InterForm::EnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700184 m_bCalculate = bEnabled;
185}
186
tsepez4cf55152016-11-02 14:37:54 -0700187bool CPDFSDK_InterForm::IsCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700188 return m_bCalculate;
189}
190
191#ifdef PDF_ENABLE_XFA
192void CPDFSDK_InterForm::AddXFAMap(CXFA_FFWidget* hWidget,
193 CPDFSDK_XFAWidget* pWidget) {
194 ASSERT(hWidget);
195 m_XFAMap[hWidget] = pWidget;
196}
197
198void CPDFSDK_InterForm::RemoveXFAMap(CXFA_FFWidget* hWidget) {
199 ASSERT(hWidget);
200 m_XFAMap.erase(hWidget);
201}
202
203CPDFSDK_XFAWidget* CPDFSDK_InterForm::GetXFAWidget(CXFA_FFWidget* hWidget) {
204 ASSERT(hWidget);
205 auto it = m_XFAMap.find(hWidget);
206 return it != m_XFAMap.end() ? it->second : nullptr;
207}
208
tsepez4cf55152016-11-02 14:37:54 -0700209void CPDFSDK_InterForm::XfaEnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700210 m_bXfaCalculate = bEnabled;
211}
tsepez4cf55152016-11-02 14:37:54 -0700212bool CPDFSDK_InterForm::IsXfaCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700213 return m_bXfaCalculate;
214}
215
tsepez4cf55152016-11-02 14:37:54 -0700216bool CPDFSDK_InterForm::IsXfaValidationsEnabled() {
jaepark611adb82016-08-17 11:34:36 -0700217 return m_bXfaValidationsEnabled;
218}
tsepez4cf55152016-11-02 14:37:54 -0700219void CPDFSDK_InterForm::XfaSetValidationsEnabled(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700220 m_bXfaValidationsEnabled = bEnabled;
221}
222
223void CPDFSDK_InterForm::SynchronizeField(CPDF_FormField* pFormField,
tsepez4cf55152016-11-02 14:37:54 -0700224 bool bSynchronizeElse) {
jaepark611adb82016-08-17 11:34:36 -0700225 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
226 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
dsinclairc5267c52016-11-04 15:35:12 -0700227 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700228 pWidget->Synchronize(bSynchronizeElse);
229 }
230}
231#endif // PDF_ENABLE_XFA
232
233void CPDFSDK_InterForm::OnCalculate(CPDF_FormField* pFormField) {
dsinclair690c0332016-10-11 09:13:01 -0700234 if (!m_pFormFillEnv->IsJSInitiated())
jaepark611adb82016-08-17 11:34:36 -0700235 return;
236
237 if (m_bBusy)
238 return;
239
tsepez4cf55152016-11-02 14:37:54 -0700240 m_bBusy = true;
jaepark611adb82016-08-17 11:34:36 -0700241
242 if (!IsCalculateEnabled()) {
tsepez4cf55152016-11-02 14:37:54 -0700243 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700244 return;
245 }
246
dsinclair690c0332016-10-11 09:13:01 -0700247 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700248 int nSize = m_pInterForm->CountFieldsInCalculationOrder();
249 for (int i = 0; i < nSize; i++) {
250 CPDF_FormField* pField = m_pInterForm->GetFieldInCalculationOrder(i);
251 if (!pField)
252 continue;
253
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000254 FormFieldType fieldType = pField->GetFieldType();
255 if (!IsFormFieldTypeComboOrText(fieldType))
jaepark611adb82016-08-17 11:34:36 -0700256 continue;
257
258 CPDF_AAction aAction = pField->GetAdditionalAction();
259 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Calculate))
260 continue;
261
262 CPDF_Action action = aAction.GetAction(CPDF_AAction::Calculate);
263 if (!action.GetDict())
264 continue;
265
Ryan Harrison275e2602017-09-18 14:23:18 -0400266 WideString csJS = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700267 if (csJS.IsEmpty())
268 continue;
269
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800270 IJS_EventContext* pContext = pRuntime->NewEventContext();
Ryan Harrison275e2602017-09-18 14:23:18 -0400271 WideString sOldValue = pField->GetValue();
272 WideString sValue = sOldValue;
tsepez4cf55152016-11-02 14:37:54 -0700273 bool bRC = true;
jaepark611adb82016-08-17 11:34:36 -0700274 pContext->OnField_Calculate(pFormField, pField, sValue, bRC);
275
Ryan Harrison275e2602017-09-18 14:23:18 -0400276 WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700277 bool bRet = pContext->RunScript(csJS, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800278 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700279 if (bRet && bRC && sValue.Compare(sOldValue) != 0)
tsepez4cf55152016-11-02 14:37:54 -0700280 pField->SetValue(sValue, true);
jaepark611adb82016-08-17 11:34:36 -0700281 }
tsepez4cf55152016-11-02 14:37:54 -0700282 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700283}
284
Ryan Harrison275e2602017-09-18 14:23:18 -0400285WideString CPDFSDK_InterForm::OnFormat(CPDF_FormField* pFormField,
286 bool& bFormatted) {
287 WideString sValue = pFormField->GetValue();
dsinclair690c0332016-10-11 09:13:01 -0700288 if (!m_pFormFillEnv->IsJSInitiated()) {
tsepez4cf55152016-11-02 14:37:54 -0700289 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700290 return sValue;
291 }
292
dsinclair690c0332016-10-11 09:13:01 -0700293 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000294 if (pFormField->GetFieldType() == FormFieldType::kComboBox &&
jaepark611adb82016-08-17 11:34:36 -0700295 pFormField->CountSelectedItems() > 0) {
296 int index = pFormField->GetSelectedIndex(0);
297 if (index >= 0)
298 sValue = pFormField->GetOptionLabel(index);
299 }
300
tsepez4cf55152016-11-02 14:37:54 -0700301 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700302
303 CPDF_AAction aAction = pFormField->GetAdditionalAction();
304 if (aAction.GetDict() && aAction.ActionExist(CPDF_AAction::Format)) {
305 CPDF_Action action = aAction.GetAction(CPDF_AAction::Format);
306 if (action.GetDict()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400307 WideString script = action.GetJavaScript();
jaepark611adb82016-08-17 11:34:36 -0700308 if (!script.IsEmpty()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400309 WideString Value = sValue;
jaepark611adb82016-08-17 11:34:36 -0700310
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800311 IJS_EventContext* pContext = pRuntime->NewEventContext();
tsepez4cf55152016-11-02 14:37:54 -0700312 pContext->OnField_Format(pFormField, Value, true);
Ryan Harrison275e2602017-09-18 14:23:18 -0400313 WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700314 bool bRet = pContext->RunScript(script, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800315 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700316 if (bRet) {
317 sValue = Value;
tsepez4cf55152016-11-02 14:37:54 -0700318 bFormatted = true;
jaepark611adb82016-08-17 11:34:36 -0700319 }
320 }
321 }
322 }
jaepark611adb82016-08-17 11:34:36 -0700323 return sValue;
324}
325
326void CPDFSDK_InterForm::ResetFieldAppearance(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400327 const WideString* sValue,
tsepez4cf55152016-11-02 14:37:54 -0700328 bool bValueChanged) {
jaepark611adb82016-08-17 11:34:36 -0700329 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
330 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
331 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700332 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700333 pWidget->ResetAppearance(sValue, bValueChanged);
334 }
335}
336
337void CPDFSDK_InterForm::UpdateField(CPDF_FormField* pFormField) {
Lei Zhang375c2762017-03-10 14:37:14 -0800338 auto* formfiller = m_pFormFillEnv->GetInteractiveFormFiller();
jaepark611adb82016-08-17 11:34:36 -0700339 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
340 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
341 ASSERT(pFormCtrl);
342
Lei Zhang77f9bff2017-08-29 11:34:12 -0700343 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
344 if (!pWidget)
345 continue;
346
347 UnderlyingPageType* pPage = pWidget->GetUnderlyingPage();
348 FX_RECT rect = formfiller->GetViewBBox(
349 m_pFormFillEnv->GetPageView(pPage, false), pWidget);
350 m_pFormFillEnv->Invalidate(pPage, rect);
jaepark611adb82016-08-17 11:34:36 -0700351 }
352}
353
tsepez4cf55152016-11-02 14:37:54 -0700354bool CPDFSDK_InterForm::OnKeyStrokeCommit(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400355 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700356 CPDF_AAction aAction = pFormField->GetAdditionalAction();
357 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::KeyStroke))
tsepez4cf55152016-11-02 14:37:54 -0700358 return true;
jaepark611adb82016-08-17 11:34:36 -0700359
360 CPDF_Action action = aAction.GetAction(CPDF_AAction::KeyStroke);
361 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700362 return true;
jaepark611adb82016-08-17 11:34:36 -0700363
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700364 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
jaepark611adb82016-08-17 11:34:36 -0700365 PDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700366 fa.bModifier = false;
367 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700368 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700369 pActionHandler->DoAction_FieldJavaScript(
370 action, CPDF_AAction::KeyStroke, m_pFormFillEnv.Get(), pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700371 return fa.bRC;
372}
373
tsepez4cf55152016-11-02 14:37:54 -0700374bool CPDFSDK_InterForm::OnValidate(CPDF_FormField* pFormField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400375 const WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700376 CPDF_AAction aAction = pFormField->GetAdditionalAction();
377 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Validate))
tsepez4cf55152016-11-02 14:37:54 -0700378 return true;
jaepark611adb82016-08-17 11:34:36 -0700379
380 CPDF_Action action = aAction.GetAction(CPDF_AAction::Validate);
381 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700382 return true;
jaepark611adb82016-08-17 11:34:36 -0700383
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700384 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
jaepark611adb82016-08-17 11:34:36 -0700385 PDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700386 fa.bModifier = false;
387 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700388 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700389 pActionHandler->DoAction_FieldJavaScript(
390 action, CPDF_AAction::Validate, m_pFormFillEnv.Get(), pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700391 return fa.bRC;
392}
393
tsepez4cf55152016-11-02 14:37:54 -0700394bool CPDFSDK_InterForm::DoAction_Hide(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700395 ASSERT(action.GetDict());
396
397 CPDF_ActionFields af(&action);
398 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
399 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
400
401 bool bHide = action.GetHideStatus();
tsepez4cf55152016-11-02 14:37:54 -0700402 bool bChanged = false;
jaepark611adb82016-08-17 11:34:36 -0700403
404 for (CPDF_FormField* pField : fields) {
405 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
406 CPDF_FormControl* pControl = pField->GetControl(i);
407 ASSERT(pControl);
408
dsinclairc5267c52016-11-04 15:35:12 -0700409 if (CPDFSDK_Widget* pWidget = GetWidget(pControl)) {
jaepark611adb82016-08-17 11:34:36 -0700410 uint32_t nFlags = pWidget->GetFlags();
411 nFlags &= ~ANNOTFLAG_INVISIBLE;
412 nFlags &= ~ANNOTFLAG_NOVIEW;
413 if (bHide)
414 nFlags |= ANNOTFLAG_HIDDEN;
415 else
416 nFlags &= ~ANNOTFLAG_HIDDEN;
417 pWidget->SetFlags(nFlags);
418 pWidget->GetPageView()->UpdateView(pWidget);
tsepez4cf55152016-11-02 14:37:54 -0700419 bChanged = true;
jaepark611adb82016-08-17 11:34:36 -0700420 }
421 }
422 }
423
424 return bChanged;
425}
426
tsepez4cf55152016-11-02 14:37:54 -0700427bool CPDFSDK_InterForm::DoAction_SubmitForm(const CPDF_Action& action) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400428 WideString sDestination = action.GetFilePath();
jaepark611adb82016-08-17 11:34:36 -0700429 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700430 return false;
jaepark611adb82016-08-17 11:34:36 -0700431
432 CPDF_Dictionary* pActionDict = action.GetDict();
433 if (pActionDict->KeyExist("Fields")) {
434 CPDF_ActionFields af(&action);
435 uint32_t dwFlags = action.GetFlags();
436 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
437 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
438 if (!fields.empty()) {
439 bool bIncludeOrExclude = !(dwFlags & 0x01);
Nicolas Penaa478dc52017-01-23 15:48:51 -0500440 if (!m_pInterForm->CheckRequiredFields(&fields, bIncludeOrExclude))
tsepez4cf55152016-11-02 14:37:54 -0700441 return false;
jaepark611adb82016-08-17 11:34:36 -0700442
443 return SubmitFields(sDestination, fields, bIncludeOrExclude, false);
444 }
445 }
Nicolas Penaa478dc52017-01-23 15:48:51 -0500446 if (!m_pInterForm->CheckRequiredFields(nullptr, true))
tsepez4cf55152016-11-02 14:37:54 -0700447 return false;
jaepark611adb82016-08-17 11:34:36 -0700448
tsepez4cf55152016-11-02 14:37:54 -0700449 return SubmitForm(sDestination, false);
jaepark611adb82016-08-17 11:34:36 -0700450}
451
Ryan Harrison275e2602017-09-18 14:23:18 -0400452bool CPDFSDK_InterForm::SubmitFields(const WideString& csDestination,
tsepez4cf55152016-11-02 14:37:54 -0700453 const std::vector<CPDF_FormField*>& fields,
454 bool bIncludeOrExclude,
455 bool bUrlEncoded) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400456 ByteString textBuf = ExportFieldsToFDFTextBuf(fields, bIncludeOrExclude);
jaepark611adb82016-08-17 11:34:36 -0700457
Ryan Harrison875e98c2017-09-27 10:53:11 -0400458 size_t nBufSize = textBuf.GetLength();
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400459 if (nBufSize == 0)
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400460 return false;
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400461
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400462 uint8_t* pLocalBuffer = FX_Alloc(uint8_t, nBufSize);
463 memcpy(pLocalBuffer, textBuf.c_str(), nBufSize);
464 uint8_t* pBuffer = pLocalBuffer;
Henrique Nakashimaaea80dc2017-08-01 19:47:24 -0400465
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400466 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize)) {
467 FX_Free(pLocalBuffer);
468 return false;
469 }
470
471 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, csDestination.c_str());
472
473 if (pBuffer != pLocalBuffer)
474 FX_Free(pBuffer);
475
476 FX_Free(pLocalBuffer);
477
tsepez4cf55152016-11-02 14:37:54 -0700478 return true;
jaepark611adb82016-08-17 11:34:36 -0700479}
480
Ryan Harrison875e98c2017-09-27 10:53:11 -0400481bool CPDFSDK_InterForm::FDFToURLEncodedData(uint8_t*& pBuf, size_t& nBufSize) {
tsepez05e01692016-11-28 17:30:09 -0800482 std::unique_ptr<CFDF_Document> pFDF =
483 CFDF_Document::ParseMemory(pBuf, nBufSize);
jaepark611adb82016-08-17 11:34:36 -0700484 if (!pFDF)
tsepez4cf55152016-11-02 14:37:54 -0700485 return true;
jaepark611adb82016-08-17 11:34:36 -0700486
dsinclair38fd8442016-09-15 10:15:32 -0700487 CPDF_Dictionary* pMainDict = pFDF->GetRoot()->GetDictFor("FDF");
jaepark611adb82016-08-17 11:34:36 -0700488 if (!pMainDict)
tsepez4cf55152016-11-02 14:37:54 -0700489 return false;
jaepark611adb82016-08-17 11:34:36 -0700490
dsinclair38fd8442016-09-15 10:15:32 -0700491 CPDF_Array* pFields = pMainDict->GetArrayFor("Fields");
jaepark611adb82016-08-17 11:34:36 -0700492 if (!pFields)
tsepez4cf55152016-11-02 14:37:54 -0700493 return false;
jaepark611adb82016-08-17 11:34:36 -0700494
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400495 std::ostringstream fdfEncodedData;
jaepark611adb82016-08-17 11:34:36 -0700496 for (uint32_t i = 0; i < pFields->GetCount(); i++) {
497 CPDF_Dictionary* pField = pFields->GetDictAt(i);
498 if (!pField)
499 continue;
Ryan Harrison275e2602017-09-18 14:23:18 -0400500 WideString name;
dsinclair38fd8442016-09-15 10:15:32 -0700501 name = pField->GetUnicodeTextFor("T");
Ryan Harrison275e2602017-09-18 14:23:18 -0400502 ByteString name_b = ByteString::FromUnicode(name);
503 ByteString csBValue = pField->GetStringFor("V");
504 WideString csWValue = PDF_DecodeText(csBValue);
505 ByteString csValue_b = ByteString::FromUnicode(csWValue);
jaepark611adb82016-08-17 11:34:36 -0700506
507 fdfEncodedData << name_b.GetBuffer(name_b.GetLength());
Ryan Harrison0186c182017-08-01 16:20:40 -0400508 name_b.ReleaseBuffer(name_b.GetStringLength());
jaepark611adb82016-08-17 11:34:36 -0700509 fdfEncodedData << "=";
510 fdfEncodedData << csValue_b.GetBuffer(csValue_b.GetLength());
Ryan Harrison0186c182017-08-01 16:20:40 -0400511 csValue_b.ReleaseBuffer(csValue_b.GetStringLength());
jaepark611adb82016-08-17 11:34:36 -0700512 if (i != pFields->GetCount() - 1)
513 fdfEncodedData << "&";
514 }
515
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400516 nBufSize = fdfEncodedData.tellp();
517 if (nBufSize == 0)
518 return false;
519
jaepark611adb82016-08-17 11:34:36 -0700520 pBuf = FX_Alloc(uint8_t, nBufSize);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400521 memcpy(pBuf, fdfEncodedData.str().c_str(), nBufSize);
tsepez4cf55152016-11-02 14:37:54 -0700522 return true;
jaepark611adb82016-08-17 11:34:36 -0700523}
524
Ryan Harrison275e2602017-09-18 14:23:18 -0400525ByteString CPDFSDK_InterForm::ExportFieldsToFDFTextBuf(
jaepark611adb82016-08-17 11:34:36 -0700526 const std::vector<CPDF_FormField*>& fields,
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400527 bool bIncludeOrExclude) {
Tom Sepez690d4562017-05-18 11:42:46 -0700528 std::unique_ptr<CFDF_Document> pFDF = m_pInterForm->ExportToFDF(
529 m_pFormFillEnv->JS_docGetFilePath(), fields, bIncludeOrExclude, false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400530
Ryan Harrison275e2602017-09-18 14:23:18 -0400531 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700532}
533
Ryan Harrison275e2602017-09-18 14:23:18 -0400534bool CPDFSDK_InterForm::SubmitForm(const WideString& sDestination,
tsepez4cf55152016-11-02 14:37:54 -0700535 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700536 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700537 return false;
jaepark611adb82016-08-17 11:34:36 -0700538
dsinclair7cbe68e2016-10-12 11:56:23 -0700539 if (!m_pFormFillEnv || !m_pInterForm)
tsepez4cf55152016-11-02 14:37:54 -0700540 return false;
jaepark611adb82016-08-17 11:34:36 -0700541
Tom Sepez690d4562017-05-18 11:42:46 -0700542 std::unique_ptr<CFDF_Document> pFDFDoc =
543 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
jaepark611adb82016-08-17 11:34:36 -0700544 if (!pFDFDoc)
tsepez4cf55152016-11-02 14:37:54 -0700545 return false;
jaepark611adb82016-08-17 11:34:36 -0700546
Ryan Harrison275e2602017-09-18 14:23:18 -0400547 ByteString fdfBuffer = pFDFDoc->WriteToString();
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400548
Ryan Harrison875e98c2017-09-27 10:53:11 -0400549 if (fdfBuffer.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700550 return false;
jaepark611adb82016-08-17 11:34:36 -0700551
Ryan Harrison875e98c2017-09-27 10:53:11 -0400552 uint8_t* pLocalBuffer = FX_Alloc(uint8_t, fdfBuffer.GetLength());
553 memcpy(pLocalBuffer, fdfBuffer.c_str(), fdfBuffer.GetLength());
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400554 uint8_t* pBuffer = pLocalBuffer;
555
Ryan Harrison875e98c2017-09-27 10:53:11 -0400556 size_t nBufSize = fdfBuffer.GetLength();
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400557 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize)) {
558 FX_Free(pLocalBuffer);
tsepez4cf55152016-11-02 14:37:54 -0700559 return false;
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400560 }
jaepark611adb82016-08-17 11:34:36 -0700561
dsinclair690c0332016-10-11 09:13:01 -0700562 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, sDestination.c_str());
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400563
564 if (pBuffer != pLocalBuffer)
jaepark611adb82016-08-17 11:34:36 -0700565 FX_Free(pBuffer);
566
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400567 FX_Free(pLocalBuffer);
568
tsepez4cf55152016-11-02 14:37:54 -0700569 return true;
jaepark611adb82016-08-17 11:34:36 -0700570}
571
Ryan Harrison275e2602017-09-18 14:23:18 -0400572ByteString CPDFSDK_InterForm::ExportFormToFDFTextBuf() {
Tom Sepez690d4562017-05-18 11:42:46 -0700573 std::unique_ptr<CFDF_Document> pFDF =
574 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
Henrique Nakashima5c09f4c2017-08-04 12:28:52 -0400575
Ryan Harrison275e2602017-09-18 14:23:18 -0400576 return pFDF ? pFDF->WriteToString() : ByteString();
jaepark611adb82016-08-17 11:34:36 -0700577}
578
tsepez4cf55152016-11-02 14:37:54 -0700579bool CPDFSDK_InterForm::DoAction_ResetForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700580 ASSERT(action.GetDict());
581
582 CPDF_Dictionary* pActionDict = action.GetDict();
583 if (!pActionDict->KeyExist("Fields"))
584 return m_pInterForm->ResetForm(true);
585
586 CPDF_ActionFields af(&action);
587 uint32_t dwFlags = action.GetFlags();
588
589 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
590 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
591 return m_pInterForm->ResetForm(fields, !(dwFlags & 0x01), true);
592}
593
tsepez4cf55152016-11-02 14:37:54 -0700594bool CPDFSDK_InterForm::DoAction_ImportData(const CPDF_Action& action) {
595 return false;
jaepark611adb82016-08-17 11:34:36 -0700596}
597
598std::vector<CPDF_FormField*> CPDFSDK_InterForm::GetFieldFromObjects(
599 const std::vector<CPDF_Object*>& objects) const {
600 std::vector<CPDF_FormField*> fields;
601 for (CPDF_Object* pObject : objects) {
602 if (pObject && pObject->IsString()) {
Ryan Harrison275e2602017-09-18 14:23:18 -0400603 WideString csName = pObject->GetUnicodeText();
jaepark611adb82016-08-17 11:34:36 -0700604 CPDF_FormField* pField = m_pInterForm->GetField(0, csName);
605 if (pField)
606 fields.push_back(pField);
607 }
608 }
609 return fields;
610}
611
612int CPDFSDK_InterForm::BeforeValueChange(CPDF_FormField* pField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400613 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000614 FormFieldType fieldType = pField->GetFieldType();
615 if (!IsFormFieldTypeComboOrText(fieldType))
jaepark611adb82016-08-17 11:34:36 -0700616 return 0;
617
618 if (!OnKeyStrokeCommit(pField, csValue))
619 return -1;
620
621 if (!OnValidate(pField, csValue))
622 return -1;
623
624 return 1;
625}
626
627void CPDFSDK_InterForm::AfterValueChange(CPDF_FormField* pField) {
628#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700629 SynchronizeField(pField, false);
jaepark611adb82016-08-17 11:34:36 -0700630#endif // PDF_ENABLE_XFA
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000631 FormFieldType fieldType = pField->GetFieldType();
632 if (IsFormFieldTypeComboOrText(fieldType)) {
jaepark611adb82016-08-17 11:34:36 -0700633 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700634 bool bFormatted = false;
Ryan Harrison275e2602017-09-18 14:23:18 -0400635 WideString sValue = OnFormat(pField, bFormatted);
tsepez4cf55152016-11-02 14:37:54 -0700636 ResetFieldAppearance(pField, bFormatted ? &sValue : nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700637 UpdateField(pField);
638 }
639}
640
641int CPDFSDK_InterForm::BeforeSelectionChange(CPDF_FormField* pField,
Ryan Harrison275e2602017-09-18 14:23:18 -0400642 const WideString& csValue) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000643 if (pField->GetFieldType() != FormFieldType::kListBox)
jaepark611adb82016-08-17 11:34:36 -0700644 return 0;
645
646 if (!OnKeyStrokeCommit(pField, csValue))
647 return -1;
648
649 if (!OnValidate(pField, csValue))
650 return -1;
651
652 return 1;
653}
654
655void CPDFSDK_InterForm::AfterSelectionChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000656 if (pField->GetFieldType() != FormFieldType::kListBox)
jaepark611adb82016-08-17 11:34:36 -0700657 return;
658
659 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700660 ResetFieldAppearance(pField, nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700661 UpdateField(pField);
662}
663
664void CPDFSDK_InterForm::AfterCheckedStatusChange(CPDF_FormField* pField) {
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000665 FormFieldType fieldType = pField->GetFieldType();
666 if (fieldType != FormFieldType::kCheckBox &&
667 fieldType != FormFieldType::kRadioButton)
jaepark611adb82016-08-17 11:34:36 -0700668 return;
669
670 OnCalculate(pField);
671 UpdateField(pField);
672}
673
674int CPDFSDK_InterForm::BeforeFormReset(CPDF_InterForm* pForm) {
675 return 0;
676}
677
678void CPDFSDK_InterForm::AfterFormReset(CPDF_InterForm* pForm) {
679 OnCalculate(nullptr);
680}
681
682int CPDFSDK_InterForm::BeforeFormImportData(CPDF_InterForm* pForm) {
683 return 0;
684}
685
686void CPDFSDK_InterForm::AfterFormImportData(CPDF_InterForm* pForm) {
687 OnCalculate(nullptr);
688}
689
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000690bool CPDFSDK_InterForm::IsNeedHighLight(FormFieldType fieldType) {
691 if (fieldType == FormFieldType::kUnknown)
tsepez4cf55152016-11-02 14:37:54 -0700692 return false;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000693
694 return m_NeedsHighlight[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700695}
696
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000697void CPDFSDK_InterForm::RemoveAllHighLights() {
698 std::fill(m_HighlightColor, m_HighlightColor + kFormFieldTypeCount,
699 FXSYS_RGB(255, 255, 255));
700 std::fill(m_NeedsHighlight, m_NeedsHighlight + kFormFieldTypeCount, false);
jaepark611adb82016-08-17 11:34:36 -0700701}
702
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000703void CPDFSDK_InterForm::SetHighlightColor(FX_COLORREF clr,
704 FormFieldType fieldType) {
705 if (fieldType == FormFieldType::kUnknown)
jaepark611adb82016-08-17 11:34:36 -0700706 return;
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000707
708 m_HighlightColor[static_cast<size_t>(fieldType)] = clr;
709 m_NeedsHighlight[static_cast<size_t>(fieldType)] = true;
710}
711
712void CPDFSDK_InterForm::SetAllHighlightColors(FX_COLORREF clr) {
713 for (auto type : kFormFieldTypes) {
714 m_HighlightColor[static_cast<size_t>(type)] = clr;
715 m_NeedsHighlight[static_cast<size_t>(type)] = true;
jaepark611adb82016-08-17 11:34:36 -0700716 }
717}
718
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000719FX_COLORREF CPDFSDK_InterForm::GetHighlightColor(FormFieldType fieldType) {
720 if (fieldType == FormFieldType::kUnknown)
jaepark611adb82016-08-17 11:34:36 -0700721 return FXSYS_RGB(255, 255, 255);
Ryan Harrison9baf31f2018-01-12 18:36:30 +0000722
723 return m_HighlightColor[static_cast<size_t>(fieldType)];
jaepark611adb82016-08-17 11:34:36 -0700724}