blob: 552bb31af21d11d98c7075c1d1b79b6eca8667bf [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>
thestig7c292e02016-09-28 14:14:26 -070011#include <vector>
jaepark611adb82016-08-17 11:34:36 -070012
dsinclair41872fa2016-10-04 11:29:35 -070013#include "core/fpdfapi/page/cpdf_page.h"
dsinclair488b7ad2016-10-04 11:55:50 -070014#include "core/fpdfapi/parser/cfdf_document.h"
15#include "core/fpdfapi/parser/cpdf_array.h"
16#include "core/fpdfapi/parser/cpdf_document.h"
17#include "core/fpdfapi/parser/cpdf_stream.h"
dsinclair1727aee2016-09-29 13:12:56 -070018#include "core/fpdfdoc/cpdf_actionfields.h"
19#include "core/fpdfdoc/cpdf_interform.h"
dsinclair74a34fc2016-09-29 16:41:42 -070020#include "core/fxge/cfx_graphstatedata.h"
21#include "core/fxge/cfx_pathdata.h"
22#include "core/fxge/cfx_renderdevice.h"
dsinclair114e46a2016-09-29 17:18:21 -070023#include "fpdfsdk/cba_annotiterator.h"
24#include "fpdfsdk/cpdfsdk_annot.h"
dsinclair735606d2016-10-05 15:47:02 -070025#include "fpdfsdk/cpdfsdk_formfillenvironment.h"
dsinclair114e46a2016-09-29 17:18:21 -070026#include "fpdfsdk/cpdfsdk_pageview.h"
27#include "fpdfsdk/cpdfsdk_widget.h"
jaepark611adb82016-08-17 11:34:36 -070028#include "fpdfsdk/formfiller/cffl_formfiller.h"
dsinclair114e46a2016-09-29 17:18:21 -070029#include "fpdfsdk/fsdk_actionhandler.h"
30#include "fpdfsdk/fsdk_define.h"
dsinclair114e46a2016-09-29 17:18:21 -070031#include "fpdfsdk/ipdfsdk_annothandler.h"
Tom Sepezd6ae2af2017-02-16 11:49:55 -080032#include "fpdfsdk/javascript/ijs_event_context.h"
jaepark611adb82016-08-17 11:34:36 -070033#include "fpdfsdk/javascript/ijs_runtime.h"
jaepark611adb82016-08-17 11:34:36 -070034#include "third_party/base/stl_util.h"
35
36#ifdef PDF_ENABLE_XFA
dsinclair114e46a2016-09-29 17:18:21 -070037#include "fpdfsdk/cpdfsdk_xfawidget.h"
dsinclair521b7502016-11-02 13:02:28 -070038#include "fpdfsdk/fpdfxfa/cpdfxfa_context.h"
dsinclair4d29e782016-10-04 14:02:47 -070039#include "fpdfsdk/fpdfxfa/cxfa_fwladaptertimermgr.h"
dsinclair5b493092016-09-29 20:20:24 -070040#include "xfa/fxfa/cxfa_eventparam.h"
Dan Sinclair80c48782017-03-23 12:11:20 -040041#include "xfa/fxfa/cxfa_ffdocview.h"
42#include "xfa/fxfa/cxfa_ffwidget.h"
43#include "xfa/fxfa/cxfa_ffwidgethandler.h"
jaepark611adb82016-08-17 11:34:36 -070044#endif // PDF_ENABLE_XFA
45
dsinclair690c0332016-10-11 09:13:01 -070046CPDFSDK_InterForm::CPDFSDK_InterForm(CPDFSDK_FormFillEnvironment* pFormFillEnv)
47 : m_pFormFillEnv(pFormFillEnv),
Dan Sinclair0bb13332017-03-30 16:12:02 -040048 m_pInterForm(
49 pdfium::MakeUnique<CPDF_InterForm>(m_pFormFillEnv->GetPDFDocument())),
jaepark611adb82016-08-17 11:34:36 -070050#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070051 m_bXfaCalculate(true),
52 m_bXfaValidationsEnabled(true),
jaepark611adb82016-08-17 11:34:36 -070053#endif // PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070054 m_bCalculate(true),
55 m_bBusy(false),
jaepark611adb82016-08-17 11:34:36 -070056 m_iHighlightAlpha(0) {
57 m_pInterForm->SetFormNotify(this);
58 for (int i = 0; i < kNumFieldTypes; ++i)
tsepez4cf55152016-11-02 14:37:54 -070059 m_bNeedHightlight[i] = false;
jaepark611adb82016-08-17 11:34:36 -070060}
61
62CPDFSDK_InterForm::~CPDFSDK_InterForm() {
63 m_Map.clear();
64#ifdef PDF_ENABLE_XFA
65 m_XFAMap.clear();
66#endif // PDF_ENABLE_XFA
67}
68
tsepez4cf55152016-11-02 14:37:54 -070069bool CPDFSDK_InterForm::HighlightWidgets() {
70 return false;
jaepark611adb82016-08-17 11:34:36 -070071}
72
73CPDFSDK_Widget* CPDFSDK_InterForm::GetSibling(CPDFSDK_Widget* pWidget,
tsepez4cf55152016-11-02 14:37:54 -070074 bool bNext) const {
Dan Sinclair0bb13332017-03-30 16:12:02 -040075 auto pIterator = pdfium::MakeUnique<CBA_AnnotIterator>(
76 pWidget->GetPageView(), CPDF_Annot::Subtype::WIDGET);
jaepark611adb82016-08-17 11:34:36 -070077
78 if (bNext)
79 return static_cast<CPDFSDK_Widget*>(pIterator->GetNextAnnot(pWidget));
80
81 return static_cast<CPDFSDK_Widget*>(pIterator->GetPrevAnnot(pWidget));
82}
83
dsinclairc5267c52016-11-04 15:35:12 -070084CPDFSDK_Widget* CPDFSDK_InterForm::GetWidget(CPDF_FormControl* pControl) const {
jaepark611adb82016-08-17 11:34:36 -070085 if (!pControl || !m_pInterForm)
86 return nullptr;
87
88 CPDFSDK_Widget* pWidget = nullptr;
89 const auto it = m_Map.find(pControl);
90 if (it != m_Map.end())
91 pWidget = it->second;
92 if (pWidget)
93 return pWidget;
jaepark611adb82016-08-17 11:34:36 -070094
95 CPDF_Dictionary* pControlDict = pControl->GetWidget();
dsinclair7cbe68e2016-10-12 11:56:23 -070096 CPDF_Document* pDocument = m_pFormFillEnv->GetPDFDocument();
jaepark611adb82016-08-17 11:34:36 -070097 CPDFSDK_PageView* pPage = nullptr;
98
dsinclair38fd8442016-09-15 10:15:32 -070099 if (CPDF_Dictionary* pPageDict = pControlDict->GetDictFor("P")) {
jaepark611adb82016-08-17 11:34:36 -0700100 int nPageIndex = pDocument->GetPageIndex(pPageDict->GetObjNum());
101 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700102 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700103 }
104
105 if (!pPage) {
106 int nPageIndex = GetPageIndexByAnnotDict(pDocument, pControlDict);
107 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700108 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700109 }
110
111 if (!pPage)
112 return nullptr;
113
114 return static_cast<CPDFSDK_Widget*>(pPage->GetAnnotByDict(pControlDict));
115}
116
117void CPDFSDK_InterForm::GetWidgets(
118 const CFX_WideString& sFieldName,
tsepez8fa82792017-01-11 09:32:33 -0800119 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700120 for (int i = 0, sz = m_pInterForm->CountFields(sFieldName); i < sz; ++i) {
121 CPDF_FormField* pFormField = m_pInterForm->GetField(i, sFieldName);
122 ASSERT(pFormField);
123 GetWidgets(pFormField, widgets);
124 }
125}
126
127void CPDFSDK_InterForm::GetWidgets(
128 CPDF_FormField* pField,
tsepez8fa82792017-01-11 09:32:33 -0800129 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700130 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
131 CPDF_FormControl* pFormCtrl = pField->GetControl(i);
132 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700133 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
jaepark611adb82016-08-17 11:34:36 -0700134 if (pWidget)
tsepez8fa82792017-01-11 09:32:33 -0800135 widgets->emplace_back(pWidget);
jaepark611adb82016-08-17 11:34:36 -0700136 }
137}
138
139int CPDFSDK_InterForm::GetPageIndexByAnnotDict(
140 CPDF_Document* pDocument,
141 CPDF_Dictionary* pAnnotDict) const {
142 ASSERT(pAnnotDict);
143
144 for (int i = 0, sz = pDocument->GetPageCount(); i < sz; i++) {
145 if (CPDF_Dictionary* pPageDict = pDocument->GetPage(i)) {
dsinclair38fd8442016-09-15 10:15:32 -0700146 if (CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots")) {
jaepark611adb82016-08-17 11:34:36 -0700147 for (int j = 0, jsz = pAnnots->GetCount(); j < jsz; j++) {
148 CPDF_Object* pDict = pAnnots->GetDirectObjectAt(j);
149 if (pAnnotDict == pDict)
150 return i;
151 }
152 }
153 }
154 }
155
156 return -1;
157}
158
159void CPDFSDK_InterForm::AddMap(CPDF_FormControl* pControl,
160 CPDFSDK_Widget* pWidget) {
161 m_Map[pControl] = pWidget;
162}
163
164void CPDFSDK_InterForm::RemoveMap(CPDF_FormControl* pControl) {
165 m_Map.erase(pControl);
166}
167
tsepez4cf55152016-11-02 14:37:54 -0700168void CPDFSDK_InterForm::EnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700169 m_bCalculate = bEnabled;
170}
171
tsepez4cf55152016-11-02 14:37:54 -0700172bool CPDFSDK_InterForm::IsCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700173 return m_bCalculate;
174}
175
176#ifdef PDF_ENABLE_XFA
177void CPDFSDK_InterForm::AddXFAMap(CXFA_FFWidget* hWidget,
178 CPDFSDK_XFAWidget* pWidget) {
179 ASSERT(hWidget);
180 m_XFAMap[hWidget] = pWidget;
181}
182
183void CPDFSDK_InterForm::RemoveXFAMap(CXFA_FFWidget* hWidget) {
184 ASSERT(hWidget);
185 m_XFAMap.erase(hWidget);
186}
187
188CPDFSDK_XFAWidget* CPDFSDK_InterForm::GetXFAWidget(CXFA_FFWidget* hWidget) {
189 ASSERT(hWidget);
190 auto it = m_XFAMap.find(hWidget);
191 return it != m_XFAMap.end() ? it->second : nullptr;
192}
193
tsepez4cf55152016-11-02 14:37:54 -0700194void CPDFSDK_InterForm::XfaEnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700195 m_bXfaCalculate = bEnabled;
196}
tsepez4cf55152016-11-02 14:37:54 -0700197bool CPDFSDK_InterForm::IsXfaCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700198 return m_bXfaCalculate;
199}
200
tsepez4cf55152016-11-02 14:37:54 -0700201bool CPDFSDK_InterForm::IsXfaValidationsEnabled() {
jaepark611adb82016-08-17 11:34:36 -0700202 return m_bXfaValidationsEnabled;
203}
tsepez4cf55152016-11-02 14:37:54 -0700204void CPDFSDK_InterForm::XfaSetValidationsEnabled(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700205 m_bXfaValidationsEnabled = bEnabled;
206}
207
208void CPDFSDK_InterForm::SynchronizeField(CPDF_FormField* pFormField,
tsepez4cf55152016-11-02 14:37:54 -0700209 bool bSynchronizeElse) {
jaepark611adb82016-08-17 11:34:36 -0700210 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
211 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
dsinclairc5267c52016-11-04 15:35:12 -0700212 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700213 pWidget->Synchronize(bSynchronizeElse);
214 }
215}
216#endif // PDF_ENABLE_XFA
217
218void CPDFSDK_InterForm::OnCalculate(CPDF_FormField* pFormField) {
dsinclair690c0332016-10-11 09:13:01 -0700219 if (!m_pFormFillEnv->IsJSInitiated())
jaepark611adb82016-08-17 11:34:36 -0700220 return;
221
222 if (m_bBusy)
223 return;
224
tsepez4cf55152016-11-02 14:37:54 -0700225 m_bBusy = true;
jaepark611adb82016-08-17 11:34:36 -0700226
227 if (!IsCalculateEnabled()) {
tsepez4cf55152016-11-02 14:37:54 -0700228 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700229 return;
230 }
231
dsinclair690c0332016-10-11 09:13:01 -0700232 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700233 int nSize = m_pInterForm->CountFieldsInCalculationOrder();
234 for (int i = 0; i < nSize; i++) {
235 CPDF_FormField* pField = m_pInterForm->GetFieldInCalculationOrder(i);
236 if (!pField)
237 continue;
238
239 int nType = pField->GetFieldType();
240 if (nType != FIELDTYPE_COMBOBOX && nType != FIELDTYPE_TEXTFIELD)
241 continue;
242
243 CPDF_AAction aAction = pField->GetAdditionalAction();
244 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Calculate))
245 continue;
246
247 CPDF_Action action = aAction.GetAction(CPDF_AAction::Calculate);
248 if (!action.GetDict())
249 continue;
250
251 CFX_WideString csJS = action.GetJavaScript();
252 if (csJS.IsEmpty())
253 continue;
254
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800255 IJS_EventContext* pContext = pRuntime->NewEventContext();
jaepark611adb82016-08-17 11:34:36 -0700256 CFX_WideString sOldValue = pField->GetValue();
257 CFX_WideString sValue = sOldValue;
tsepez4cf55152016-11-02 14:37:54 -0700258 bool bRC = true;
jaepark611adb82016-08-17 11:34:36 -0700259 pContext->OnField_Calculate(pFormField, pField, sValue, bRC);
260
261 CFX_WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700262 bool bRet = pContext->RunScript(csJS, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800263 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700264 if (bRet && bRC && sValue.Compare(sOldValue) != 0)
tsepez4cf55152016-11-02 14:37:54 -0700265 pField->SetValue(sValue, true);
jaepark611adb82016-08-17 11:34:36 -0700266 }
tsepez4cf55152016-11-02 14:37:54 -0700267 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700268}
269
270CFX_WideString CPDFSDK_InterForm::OnFormat(CPDF_FormField* pFormField,
tsepez4cf55152016-11-02 14:37:54 -0700271 bool& bFormatted) {
jaepark611adb82016-08-17 11:34:36 -0700272 CFX_WideString sValue = pFormField->GetValue();
dsinclair690c0332016-10-11 09:13:01 -0700273 if (!m_pFormFillEnv->IsJSInitiated()) {
tsepez4cf55152016-11-02 14:37:54 -0700274 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700275 return sValue;
276 }
277
dsinclair690c0332016-10-11 09:13:01 -0700278 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700279 if (pFormField->GetFieldType() == FIELDTYPE_COMBOBOX &&
280 pFormField->CountSelectedItems() > 0) {
281 int index = pFormField->GetSelectedIndex(0);
282 if (index >= 0)
283 sValue = pFormField->GetOptionLabel(index);
284 }
285
tsepez4cf55152016-11-02 14:37:54 -0700286 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700287
288 CPDF_AAction aAction = pFormField->GetAdditionalAction();
289 if (aAction.GetDict() && aAction.ActionExist(CPDF_AAction::Format)) {
290 CPDF_Action action = aAction.GetAction(CPDF_AAction::Format);
291 if (action.GetDict()) {
292 CFX_WideString script = action.GetJavaScript();
293 if (!script.IsEmpty()) {
294 CFX_WideString Value = sValue;
295
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800296 IJS_EventContext* pContext = pRuntime->NewEventContext();
tsepez4cf55152016-11-02 14:37:54 -0700297 pContext->OnField_Format(pFormField, Value, true);
jaepark611adb82016-08-17 11:34:36 -0700298 CFX_WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700299 bool bRet = pContext->RunScript(script, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800300 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700301 if (bRet) {
302 sValue = Value;
tsepez4cf55152016-11-02 14:37:54 -0700303 bFormatted = true;
jaepark611adb82016-08-17 11:34:36 -0700304 }
305 }
306 }
307 }
jaepark611adb82016-08-17 11:34:36 -0700308 return sValue;
309}
310
311void CPDFSDK_InterForm::ResetFieldAppearance(CPDF_FormField* pFormField,
tsepeza31da742016-09-08 11:28:14 -0700312 const CFX_WideString* sValue,
tsepez4cf55152016-11-02 14:37:54 -0700313 bool bValueChanged) {
jaepark611adb82016-08-17 11:34:36 -0700314 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
315 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
316 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700317 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700318 pWidget->ResetAppearance(sValue, bValueChanged);
319 }
320}
321
322void CPDFSDK_InterForm::UpdateField(CPDF_FormField* pFormField) {
Lei Zhang375c2762017-03-10 14:37:14 -0800323 auto* formfiller = m_pFormFillEnv->GetInteractiveFormFiller();
jaepark611adb82016-08-17 11:34:36 -0700324 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
325 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
326 ASSERT(pFormCtrl);
327
dsinclairc5267c52016-11-04 15:35:12 -0700328 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl)) {
jaepark611adb82016-08-17 11:34:36 -0700329 UnderlyingPageType* pPage = pWidget->GetUnderlyingPage();
Dan Sinclair6eec1c42017-02-21 17:20:43 -0500330 m_pFormFillEnv->Invalidate(
331 pPage, formfiller->GetViewBBox(
332 m_pFormFillEnv->GetPageView(pPage, false), pWidget));
jaepark611adb82016-08-17 11:34:36 -0700333 }
334 }
335}
336
tsepez4cf55152016-11-02 14:37:54 -0700337bool CPDFSDK_InterForm::OnKeyStrokeCommit(CPDF_FormField* pFormField,
338 const CFX_WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700339 CPDF_AAction aAction = pFormField->GetAdditionalAction();
340 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::KeyStroke))
tsepez4cf55152016-11-02 14:37:54 -0700341 return true;
jaepark611adb82016-08-17 11:34:36 -0700342
343 CPDF_Action action = aAction.GetAction(CPDF_AAction::KeyStroke);
344 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700345 return true;
jaepark611adb82016-08-17 11:34:36 -0700346
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700347 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
jaepark611adb82016-08-17 11:34:36 -0700348 PDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700349 fa.bModifier = false;
350 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700351 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700352 pActionHandler->DoAction_FieldJavaScript(
353 action, CPDF_AAction::KeyStroke, m_pFormFillEnv.Get(), pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700354 return fa.bRC;
355}
356
tsepez4cf55152016-11-02 14:37:54 -0700357bool CPDFSDK_InterForm::OnValidate(CPDF_FormField* pFormField,
358 const CFX_WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700359 CPDF_AAction aAction = pFormField->GetAdditionalAction();
360 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Validate))
tsepez4cf55152016-11-02 14:37:54 -0700361 return true;
jaepark611adb82016-08-17 11:34:36 -0700362
363 CPDF_Action action = aAction.GetAction(CPDF_AAction::Validate);
364 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700365 return true;
jaepark611adb82016-08-17 11:34:36 -0700366
Lei Zhangcddc8ed2017-06-20 17:26:44 -0700367 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHandler();
jaepark611adb82016-08-17 11:34:36 -0700368 PDFSDK_FieldAction fa;
Lei Zhang60fa2fc2017-07-21 17:42:19 -0700369 fa.bModifier = false;
370 fa.bShift = false;
jaepark611adb82016-08-17 11:34:36 -0700371 fa.sValue = csValue;
Tom Sepezcc205132017-05-16 14:01:47 -0700372 pActionHandler->DoAction_FieldJavaScript(
373 action, CPDF_AAction::Validate, m_pFormFillEnv.Get(), pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700374 return fa.bRC;
375}
376
tsepez4cf55152016-11-02 14:37:54 -0700377bool CPDFSDK_InterForm::DoAction_Hide(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700378 ASSERT(action.GetDict());
379
380 CPDF_ActionFields af(&action);
381 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
382 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
383
384 bool bHide = action.GetHideStatus();
tsepez4cf55152016-11-02 14:37:54 -0700385 bool bChanged = false;
jaepark611adb82016-08-17 11:34:36 -0700386
387 for (CPDF_FormField* pField : fields) {
388 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
389 CPDF_FormControl* pControl = pField->GetControl(i);
390 ASSERT(pControl);
391
dsinclairc5267c52016-11-04 15:35:12 -0700392 if (CPDFSDK_Widget* pWidget = GetWidget(pControl)) {
jaepark611adb82016-08-17 11:34:36 -0700393 uint32_t nFlags = pWidget->GetFlags();
394 nFlags &= ~ANNOTFLAG_INVISIBLE;
395 nFlags &= ~ANNOTFLAG_NOVIEW;
396 if (bHide)
397 nFlags |= ANNOTFLAG_HIDDEN;
398 else
399 nFlags &= ~ANNOTFLAG_HIDDEN;
400 pWidget->SetFlags(nFlags);
401 pWidget->GetPageView()->UpdateView(pWidget);
tsepez4cf55152016-11-02 14:37:54 -0700402 bChanged = true;
jaepark611adb82016-08-17 11:34:36 -0700403 }
404 }
405 }
406
407 return bChanged;
408}
409
tsepez4cf55152016-11-02 14:37:54 -0700410bool CPDFSDK_InterForm::DoAction_SubmitForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700411 CFX_WideString sDestination = action.GetFilePath();
412 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700413 return false;
jaepark611adb82016-08-17 11:34:36 -0700414
415 CPDF_Dictionary* pActionDict = action.GetDict();
416 if (pActionDict->KeyExist("Fields")) {
417 CPDF_ActionFields af(&action);
418 uint32_t dwFlags = action.GetFlags();
419 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
420 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
421 if (!fields.empty()) {
422 bool bIncludeOrExclude = !(dwFlags & 0x01);
Nicolas Penaa478dc52017-01-23 15:48:51 -0500423 if (!m_pInterForm->CheckRequiredFields(&fields, bIncludeOrExclude))
tsepez4cf55152016-11-02 14:37:54 -0700424 return false;
jaepark611adb82016-08-17 11:34:36 -0700425
426 return SubmitFields(sDestination, fields, bIncludeOrExclude, false);
427 }
428 }
Nicolas Penaa478dc52017-01-23 15:48:51 -0500429 if (!m_pInterForm->CheckRequiredFields(nullptr, true))
tsepez4cf55152016-11-02 14:37:54 -0700430 return false;
jaepark611adb82016-08-17 11:34:36 -0700431
tsepez4cf55152016-11-02 14:37:54 -0700432 return SubmitForm(sDestination, false);
jaepark611adb82016-08-17 11:34:36 -0700433}
434
tsepez4cf55152016-11-02 14:37:54 -0700435bool CPDFSDK_InterForm::SubmitFields(const CFX_WideString& csDestination,
436 const std::vector<CPDF_FormField*>& fields,
437 bool bIncludeOrExclude,
438 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700439 CFX_ByteTextBuf textBuf;
440 ExportFieldsToFDFTextBuf(fields, bIncludeOrExclude, textBuf);
441
442 uint8_t* pBuffer = textBuf.GetBuffer();
443 FX_STRSIZE nBufSize = textBuf.GetLength();
444
445 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize))
tsepez4cf55152016-11-02 14:37:54 -0700446 return false;
jaepark611adb82016-08-17 11:34:36 -0700447
dsinclair690c0332016-10-11 09:13:01 -0700448 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, csDestination.c_str());
tsepez4cf55152016-11-02 14:37:54 -0700449 return true;
jaepark611adb82016-08-17 11:34:36 -0700450}
451
tsepez4cf55152016-11-02 14:37:54 -0700452bool CPDFSDK_InterForm::FDFToURLEncodedData(CFX_WideString csFDFFile,
453 CFX_WideString csTxtFile) {
454 return true;
jaepark611adb82016-08-17 11:34:36 -0700455}
456
tsepez4cf55152016-11-02 14:37:54 -0700457bool CPDFSDK_InterForm::FDFToURLEncodedData(uint8_t*& pBuf,
458 FX_STRSIZE& nBufSize) {
tsepez05e01692016-11-28 17:30:09 -0800459 std::unique_ptr<CFDF_Document> pFDF =
460 CFDF_Document::ParseMemory(pBuf, nBufSize);
jaepark611adb82016-08-17 11:34:36 -0700461 if (!pFDF)
tsepez4cf55152016-11-02 14:37:54 -0700462 return true;
jaepark611adb82016-08-17 11:34:36 -0700463
dsinclair38fd8442016-09-15 10:15:32 -0700464 CPDF_Dictionary* pMainDict = pFDF->GetRoot()->GetDictFor("FDF");
jaepark611adb82016-08-17 11:34:36 -0700465 if (!pMainDict)
tsepez4cf55152016-11-02 14:37:54 -0700466 return false;
jaepark611adb82016-08-17 11:34:36 -0700467
dsinclair38fd8442016-09-15 10:15:32 -0700468 CPDF_Array* pFields = pMainDict->GetArrayFor("Fields");
jaepark611adb82016-08-17 11:34:36 -0700469 if (!pFields)
tsepez4cf55152016-11-02 14:37:54 -0700470 return false;
jaepark611adb82016-08-17 11:34:36 -0700471
472 CFX_ByteTextBuf fdfEncodedData;
473 for (uint32_t i = 0; i < pFields->GetCount(); i++) {
474 CPDF_Dictionary* pField = pFields->GetDictAt(i);
475 if (!pField)
476 continue;
477 CFX_WideString name;
dsinclair38fd8442016-09-15 10:15:32 -0700478 name = pField->GetUnicodeTextFor("T");
jaepark611adb82016-08-17 11:34:36 -0700479 CFX_ByteString name_b = CFX_ByteString::FromUnicode(name);
dsinclair38fd8442016-09-15 10:15:32 -0700480 CFX_ByteString csBValue = pField->GetStringFor("V");
jaepark611adb82016-08-17 11:34:36 -0700481 CFX_WideString csWValue = PDF_DecodeText(csBValue);
482 CFX_ByteString csValue_b = CFX_ByteString::FromUnicode(csWValue);
483
484 fdfEncodedData << name_b.GetBuffer(name_b.GetLength());
Ryan Harrison0186c182017-08-01 16:20:40 -0400485 name_b.ReleaseBuffer(name_b.GetStringLength());
jaepark611adb82016-08-17 11:34:36 -0700486 fdfEncodedData << "=";
487 fdfEncodedData << csValue_b.GetBuffer(csValue_b.GetLength());
Ryan Harrison0186c182017-08-01 16:20:40 -0400488 csValue_b.ReleaseBuffer(csValue_b.GetStringLength());
jaepark611adb82016-08-17 11:34:36 -0700489 if (i != pFields->GetCount() - 1)
490 fdfEncodedData << "&";
491 }
492
493 nBufSize = fdfEncodedData.GetLength();
494 pBuf = FX_Alloc(uint8_t, nBufSize);
Dan Sinclair1c5d0b42017-04-03 15:05:11 -0400495 memcpy(pBuf, fdfEncodedData.GetBuffer(), nBufSize);
tsepez4cf55152016-11-02 14:37:54 -0700496 return true;
jaepark611adb82016-08-17 11:34:36 -0700497}
498
tsepez4cf55152016-11-02 14:37:54 -0700499bool CPDFSDK_InterForm::ExportFieldsToFDFTextBuf(
jaepark611adb82016-08-17 11:34:36 -0700500 const std::vector<CPDF_FormField*>& fields,
501 bool bIncludeOrExclude,
502 CFX_ByteTextBuf& textBuf) {
Tom Sepez690d4562017-05-18 11:42:46 -0700503 std::unique_ptr<CFDF_Document> pFDF = m_pInterForm->ExportToFDF(
504 m_pFormFillEnv->JS_docGetFilePath(), fields, bIncludeOrExclude, false);
tsepez4cf55152016-11-02 14:37:54 -0700505 return pFDF ? pFDF->WriteBuf(textBuf) : false;
jaepark611adb82016-08-17 11:34:36 -0700506}
507
508CFX_WideString CPDFSDK_InterForm::GetTemporaryFileName(
509 const CFX_WideString& sFileExt) {
510 return L"";
511}
512
tsepez4cf55152016-11-02 14:37:54 -0700513bool CPDFSDK_InterForm::SubmitForm(const CFX_WideString& sDestination,
514 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700515 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700516 return false;
jaepark611adb82016-08-17 11:34:36 -0700517
dsinclair7cbe68e2016-10-12 11:56:23 -0700518 if (!m_pFormFillEnv || !m_pInterForm)
tsepez4cf55152016-11-02 14:37:54 -0700519 return false;
jaepark611adb82016-08-17 11:34:36 -0700520
Tom Sepez690d4562017-05-18 11:42:46 -0700521 std::unique_ptr<CFDF_Document> pFDFDoc =
522 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
jaepark611adb82016-08-17 11:34:36 -0700523 if (!pFDFDoc)
tsepez4cf55152016-11-02 14:37:54 -0700524 return false;
jaepark611adb82016-08-17 11:34:36 -0700525
526 CFX_ByteTextBuf FdfBuffer;
tsepez05e01692016-11-28 17:30:09 -0800527 if (!pFDFDoc->WriteBuf(FdfBuffer))
tsepez4cf55152016-11-02 14:37:54 -0700528 return false;
jaepark611adb82016-08-17 11:34:36 -0700529
530 uint8_t* pBuffer = FdfBuffer.GetBuffer();
531 FX_STRSIZE nBufSize = FdfBuffer.GetLength();
jaepark611adb82016-08-17 11:34:36 -0700532 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize))
tsepez4cf55152016-11-02 14:37:54 -0700533 return false;
jaepark611adb82016-08-17 11:34:36 -0700534
dsinclair690c0332016-10-11 09:13:01 -0700535 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, sDestination.c_str());
jaepark611adb82016-08-17 11:34:36 -0700536 if (bUrlEncoded)
537 FX_Free(pBuffer);
538
tsepez4cf55152016-11-02 14:37:54 -0700539 return true;
jaepark611adb82016-08-17 11:34:36 -0700540}
541
tsepez4cf55152016-11-02 14:37:54 -0700542bool CPDFSDK_InterForm::ExportFormToFDFTextBuf(CFX_ByteTextBuf& textBuf) {
Tom Sepez690d4562017-05-18 11:42:46 -0700543 std::unique_ptr<CFDF_Document> pFDF =
544 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath(), false);
tsepez05e01692016-11-28 17:30:09 -0800545 return pFDF && pFDF->WriteBuf(textBuf);
jaepark611adb82016-08-17 11:34:36 -0700546}
547
tsepez4cf55152016-11-02 14:37:54 -0700548bool CPDFSDK_InterForm::DoAction_ResetForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700549 ASSERT(action.GetDict());
550
551 CPDF_Dictionary* pActionDict = action.GetDict();
552 if (!pActionDict->KeyExist("Fields"))
553 return m_pInterForm->ResetForm(true);
554
555 CPDF_ActionFields af(&action);
556 uint32_t dwFlags = action.GetFlags();
557
558 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
559 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
560 return m_pInterForm->ResetForm(fields, !(dwFlags & 0x01), true);
561}
562
tsepez4cf55152016-11-02 14:37:54 -0700563bool CPDFSDK_InterForm::DoAction_ImportData(const CPDF_Action& action) {
564 return false;
jaepark611adb82016-08-17 11:34:36 -0700565}
566
567std::vector<CPDF_FormField*> CPDFSDK_InterForm::GetFieldFromObjects(
568 const std::vector<CPDF_Object*>& objects) const {
569 std::vector<CPDF_FormField*> fields;
570 for (CPDF_Object* pObject : objects) {
571 if (pObject && pObject->IsString()) {
572 CFX_WideString csName = pObject->GetUnicodeText();
573 CPDF_FormField* pField = m_pInterForm->GetField(0, csName);
574 if (pField)
575 fields.push_back(pField);
576 }
577 }
578 return fields;
579}
580
581int CPDFSDK_InterForm::BeforeValueChange(CPDF_FormField* pField,
582 const CFX_WideString& csValue) {
583 int nType = pField->GetFieldType();
584 if (nType != FIELDTYPE_COMBOBOX && nType != FIELDTYPE_TEXTFIELD)
585 return 0;
586
587 if (!OnKeyStrokeCommit(pField, csValue))
588 return -1;
589
590 if (!OnValidate(pField, csValue))
591 return -1;
592
593 return 1;
594}
595
596void CPDFSDK_InterForm::AfterValueChange(CPDF_FormField* pField) {
597#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700598 SynchronizeField(pField, false);
jaepark611adb82016-08-17 11:34:36 -0700599#endif // PDF_ENABLE_XFA
600 int nType = pField->GetFieldType();
601 if (nType == FIELDTYPE_COMBOBOX || nType == FIELDTYPE_TEXTFIELD) {
602 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700603 bool bFormatted = false;
tsepez8c2a8cd2016-09-07 15:29:11 -0700604 CFX_WideString sValue = OnFormat(pField, bFormatted);
tsepez4cf55152016-11-02 14:37:54 -0700605 ResetFieldAppearance(pField, bFormatted ? &sValue : nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700606 UpdateField(pField);
607 }
608}
609
610int CPDFSDK_InterForm::BeforeSelectionChange(CPDF_FormField* pField,
611 const CFX_WideString& csValue) {
612 if (pField->GetFieldType() != FIELDTYPE_LISTBOX)
613 return 0;
614
615 if (!OnKeyStrokeCommit(pField, csValue))
616 return -1;
617
618 if (!OnValidate(pField, csValue))
619 return -1;
620
621 return 1;
622}
623
624void CPDFSDK_InterForm::AfterSelectionChange(CPDF_FormField* pField) {
625 if (pField->GetFieldType() != FIELDTYPE_LISTBOX)
626 return;
627
628 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700629 ResetFieldAppearance(pField, nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700630 UpdateField(pField);
631}
632
633void CPDFSDK_InterForm::AfterCheckedStatusChange(CPDF_FormField* pField) {
634 int nType = pField->GetFieldType();
635 if (nType != FIELDTYPE_CHECKBOX && nType != FIELDTYPE_RADIOBUTTON)
636 return;
637
638 OnCalculate(pField);
639 UpdateField(pField);
640}
641
642int CPDFSDK_InterForm::BeforeFormReset(CPDF_InterForm* pForm) {
643 return 0;
644}
645
646void CPDFSDK_InterForm::AfterFormReset(CPDF_InterForm* pForm) {
647 OnCalculate(nullptr);
648}
649
650int CPDFSDK_InterForm::BeforeFormImportData(CPDF_InterForm* pForm) {
651 return 0;
652}
653
654void CPDFSDK_InterForm::AfterFormImportData(CPDF_InterForm* pForm) {
655 OnCalculate(nullptr);
656}
657
tsepez4cf55152016-11-02 14:37:54 -0700658bool CPDFSDK_InterForm::IsNeedHighLight(int nFieldType) {
jaepark611adb82016-08-17 11:34:36 -0700659 if (nFieldType < 1 || nFieldType > kNumFieldTypes)
tsepez4cf55152016-11-02 14:37:54 -0700660 return false;
jaepark611adb82016-08-17 11:34:36 -0700661 return m_bNeedHightlight[nFieldType - 1];
662}
663
664void CPDFSDK_InterForm::RemoveAllHighLight() {
665 for (int i = 0; i < kNumFieldTypes; ++i)
tsepez4cf55152016-11-02 14:37:54 -0700666 m_bNeedHightlight[i] = false;
jaepark611adb82016-08-17 11:34:36 -0700667}
668
669void CPDFSDK_InterForm::SetHighlightColor(FX_COLORREF clr, int nFieldType) {
670 if (nFieldType < 0 || nFieldType > kNumFieldTypes)
671 return;
672 switch (nFieldType) {
673 case 0: {
674 for (int i = 0; i < kNumFieldTypes; ++i) {
675 m_aHighlightColor[i] = clr;
tsepez4cf55152016-11-02 14:37:54 -0700676 m_bNeedHightlight[i] = true;
jaepark611adb82016-08-17 11:34:36 -0700677 }
678 break;
679 }
680 default: {
681 m_aHighlightColor[nFieldType - 1] = clr;
tsepez4cf55152016-11-02 14:37:54 -0700682 m_bNeedHightlight[nFieldType - 1] = true;
jaepark611adb82016-08-17 11:34:36 -0700683 break;
684 }
685 }
686}
687
688FX_COLORREF CPDFSDK_InterForm::GetHighlightColor(int nFieldType) {
689 if (nFieldType < 0 || nFieldType > kNumFieldTypes)
690 return FXSYS_RGB(255, 255, 255);
691 if (nFieldType == 0)
692 return m_aHighlightColor[0];
693 return m_aHighlightColor[nFieldType - 1];
694}