blob: 4229fcdac54512f902f1631c583d30f1ae397a87 [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"
dsinclair0bb385b2016-09-29 17:03:59 -070031#include "fpdfsdk/fxedit/fxet_edit.h"
dsinclair114e46a2016-09-29 17:18:21 -070032#include "fpdfsdk/ipdfsdk_annothandler.h"
Tom Sepezd6ae2af2017-02-16 11:49:55 -080033#include "fpdfsdk/javascript/ijs_event_context.h"
jaepark611adb82016-08-17 11:34:36 -070034#include "fpdfsdk/javascript/ijs_runtime.h"
35#include "fpdfsdk/pdfwindow/PWL_Utils.h"
36#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"
43#include "xfa/fxfa/xfa_ffdocview.h"
44#include "xfa/fxfa/xfa_ffwidget.h"
45#include "xfa/fxfa/xfa_ffwidgethandler.h"
jaepark611adb82016-08-17 11:34:36 -070046#endif // PDF_ENABLE_XFA
47
dsinclair690c0332016-10-11 09:13:01 -070048CPDFSDK_InterForm::CPDFSDK_InterForm(CPDFSDK_FormFillEnvironment* pFormFillEnv)
49 : m_pFormFillEnv(pFormFillEnv),
dsinclair7cbe68e2016-10-12 11:56:23 -070050 m_pInterForm(new CPDF_InterForm(m_pFormFillEnv->GetPDFDocument())),
jaepark611adb82016-08-17 11:34:36 -070051#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070052 m_bXfaCalculate(true),
53 m_bXfaValidationsEnabled(true),
jaepark611adb82016-08-17 11:34:36 -070054#endif // PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -070055 m_bCalculate(true),
56 m_bBusy(false),
jaepark611adb82016-08-17 11:34:36 -070057 m_iHighlightAlpha(0) {
58 m_pInterForm->SetFormNotify(this);
59 for (int i = 0; i < kNumFieldTypes; ++i)
tsepez4cf55152016-11-02 14:37:54 -070060 m_bNeedHightlight[i] = false;
jaepark611adb82016-08-17 11:34:36 -070061}
62
63CPDFSDK_InterForm::~CPDFSDK_InterForm() {
64 m_Map.clear();
65#ifdef PDF_ENABLE_XFA
66 m_XFAMap.clear();
67#endif // PDF_ENABLE_XFA
68}
69
tsepez4cf55152016-11-02 14:37:54 -070070bool CPDFSDK_InterForm::HighlightWidgets() {
71 return false;
jaepark611adb82016-08-17 11:34:36 -070072}
73
74CPDFSDK_Widget* CPDFSDK_InterForm::GetSibling(CPDFSDK_Widget* pWidget,
tsepez4cf55152016-11-02 14:37:54 -070075 bool bNext) const {
jaepark956553e2016-08-31 06:49:27 -070076 std::unique_ptr<CBA_AnnotIterator> pIterator(new CBA_AnnotIterator(
77 pWidget->GetPageView(), CPDF_Annot::Subtype::WIDGET));
jaepark611adb82016-08-17 11:34:36 -070078
79 if (bNext)
80 return static_cast<CPDFSDK_Widget*>(pIterator->GetNextAnnot(pWidget));
81
82 return static_cast<CPDFSDK_Widget*>(pIterator->GetPrevAnnot(pWidget));
83}
84
dsinclairc5267c52016-11-04 15:35:12 -070085CPDFSDK_Widget* CPDFSDK_InterForm::GetWidget(CPDF_FormControl* pControl) const {
jaepark611adb82016-08-17 11:34:36 -070086 if (!pControl || !m_pInterForm)
87 return nullptr;
88
89 CPDFSDK_Widget* pWidget = nullptr;
90 const auto it = m_Map.find(pControl);
91 if (it != m_Map.end())
92 pWidget = it->second;
93 if (pWidget)
94 return pWidget;
jaepark611adb82016-08-17 11:34:36 -070095
96 CPDF_Dictionary* pControlDict = pControl->GetWidget();
dsinclair7cbe68e2016-10-12 11:56:23 -070097 CPDF_Document* pDocument = m_pFormFillEnv->GetPDFDocument();
jaepark611adb82016-08-17 11:34:36 -070098 CPDFSDK_PageView* pPage = nullptr;
99
dsinclair38fd8442016-09-15 10:15:32 -0700100 if (CPDF_Dictionary* pPageDict = pControlDict->GetDictFor("P")) {
jaepark611adb82016-08-17 11:34:36 -0700101 int nPageIndex = pDocument->GetPageIndex(pPageDict->GetObjNum());
102 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700103 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700104 }
105
106 if (!pPage) {
107 int nPageIndex = GetPageIndexByAnnotDict(pDocument, pControlDict);
108 if (nPageIndex >= 0)
dsinclair7cbe68e2016-10-12 11:56:23 -0700109 pPage = m_pFormFillEnv->GetPageView(nPageIndex);
jaepark611adb82016-08-17 11:34:36 -0700110 }
111
112 if (!pPage)
113 return nullptr;
114
115 return static_cast<CPDFSDK_Widget*>(pPage->GetAnnotByDict(pControlDict));
116}
117
118void CPDFSDK_InterForm::GetWidgets(
119 const CFX_WideString& sFieldName,
tsepez8fa82792017-01-11 09:32:33 -0800120 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700121 for (int i = 0, sz = m_pInterForm->CountFields(sFieldName); i < sz; ++i) {
122 CPDF_FormField* pFormField = m_pInterForm->GetField(i, sFieldName);
123 ASSERT(pFormField);
124 GetWidgets(pFormField, widgets);
125 }
126}
127
128void CPDFSDK_InterForm::GetWidgets(
129 CPDF_FormField* pField,
tsepez8fa82792017-01-11 09:32:33 -0800130 std::vector<CPDFSDK_Annot::ObservedPtr>* widgets) const {
jaepark611adb82016-08-17 11:34:36 -0700131 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
132 CPDF_FormControl* pFormCtrl = pField->GetControl(i);
133 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700134 CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl);
jaepark611adb82016-08-17 11:34:36 -0700135 if (pWidget)
tsepez8fa82792017-01-11 09:32:33 -0800136 widgets->emplace_back(pWidget);
jaepark611adb82016-08-17 11:34:36 -0700137 }
138}
139
140int CPDFSDK_InterForm::GetPageIndexByAnnotDict(
141 CPDF_Document* pDocument,
142 CPDF_Dictionary* pAnnotDict) const {
143 ASSERT(pAnnotDict);
144
145 for (int i = 0, sz = pDocument->GetPageCount(); i < sz; i++) {
146 if (CPDF_Dictionary* pPageDict = pDocument->GetPage(i)) {
dsinclair38fd8442016-09-15 10:15:32 -0700147 if (CPDF_Array* pAnnots = pPageDict->GetArrayFor("Annots")) {
jaepark611adb82016-08-17 11:34:36 -0700148 for (int j = 0, jsz = pAnnots->GetCount(); j < jsz; j++) {
149 CPDF_Object* pDict = pAnnots->GetDirectObjectAt(j);
150 if (pAnnotDict == pDict)
151 return i;
152 }
153 }
154 }
155 }
156
157 return -1;
158}
159
160void CPDFSDK_InterForm::AddMap(CPDF_FormControl* pControl,
161 CPDFSDK_Widget* pWidget) {
162 m_Map[pControl] = pWidget;
163}
164
165void CPDFSDK_InterForm::RemoveMap(CPDF_FormControl* pControl) {
166 m_Map.erase(pControl);
167}
168
tsepez4cf55152016-11-02 14:37:54 -0700169void CPDFSDK_InterForm::EnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700170 m_bCalculate = bEnabled;
171}
172
tsepez4cf55152016-11-02 14:37:54 -0700173bool CPDFSDK_InterForm::IsCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700174 return m_bCalculate;
175}
176
177#ifdef PDF_ENABLE_XFA
178void CPDFSDK_InterForm::AddXFAMap(CXFA_FFWidget* hWidget,
179 CPDFSDK_XFAWidget* pWidget) {
180 ASSERT(hWidget);
181 m_XFAMap[hWidget] = pWidget;
182}
183
184void CPDFSDK_InterForm::RemoveXFAMap(CXFA_FFWidget* hWidget) {
185 ASSERT(hWidget);
186 m_XFAMap.erase(hWidget);
187}
188
189CPDFSDK_XFAWidget* CPDFSDK_InterForm::GetXFAWidget(CXFA_FFWidget* hWidget) {
190 ASSERT(hWidget);
191 auto it = m_XFAMap.find(hWidget);
192 return it != m_XFAMap.end() ? it->second : nullptr;
193}
194
tsepez4cf55152016-11-02 14:37:54 -0700195void CPDFSDK_InterForm::XfaEnableCalculate(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700196 m_bXfaCalculate = bEnabled;
197}
tsepez4cf55152016-11-02 14:37:54 -0700198bool CPDFSDK_InterForm::IsXfaCalculateEnabled() const {
jaepark611adb82016-08-17 11:34:36 -0700199 return m_bXfaCalculate;
200}
201
tsepez4cf55152016-11-02 14:37:54 -0700202bool CPDFSDK_InterForm::IsXfaValidationsEnabled() {
jaepark611adb82016-08-17 11:34:36 -0700203 return m_bXfaValidationsEnabled;
204}
tsepez4cf55152016-11-02 14:37:54 -0700205void CPDFSDK_InterForm::XfaSetValidationsEnabled(bool bEnabled) {
jaepark611adb82016-08-17 11:34:36 -0700206 m_bXfaValidationsEnabled = bEnabled;
207}
208
209void CPDFSDK_InterForm::SynchronizeField(CPDF_FormField* pFormField,
tsepez4cf55152016-11-02 14:37:54 -0700210 bool bSynchronizeElse) {
jaepark611adb82016-08-17 11:34:36 -0700211 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
212 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
dsinclairc5267c52016-11-04 15:35:12 -0700213 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700214 pWidget->Synchronize(bSynchronizeElse);
215 }
216}
217#endif // PDF_ENABLE_XFA
218
219void CPDFSDK_InterForm::OnCalculate(CPDF_FormField* pFormField) {
dsinclair690c0332016-10-11 09:13:01 -0700220 if (!m_pFormFillEnv->IsJSInitiated())
jaepark611adb82016-08-17 11:34:36 -0700221 return;
222
223 if (m_bBusy)
224 return;
225
tsepez4cf55152016-11-02 14:37:54 -0700226 m_bBusy = true;
jaepark611adb82016-08-17 11:34:36 -0700227
228 if (!IsCalculateEnabled()) {
tsepez4cf55152016-11-02 14:37:54 -0700229 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700230 return;
231 }
232
dsinclair690c0332016-10-11 09:13:01 -0700233 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700234 int nSize = m_pInterForm->CountFieldsInCalculationOrder();
235 for (int i = 0; i < nSize; i++) {
236 CPDF_FormField* pField = m_pInterForm->GetFieldInCalculationOrder(i);
237 if (!pField)
238 continue;
239
240 int nType = pField->GetFieldType();
241 if (nType != FIELDTYPE_COMBOBOX && nType != FIELDTYPE_TEXTFIELD)
242 continue;
243
244 CPDF_AAction aAction = pField->GetAdditionalAction();
245 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Calculate))
246 continue;
247
248 CPDF_Action action = aAction.GetAction(CPDF_AAction::Calculate);
249 if (!action.GetDict())
250 continue;
251
252 CFX_WideString csJS = action.GetJavaScript();
253 if (csJS.IsEmpty())
254 continue;
255
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800256 IJS_EventContext* pContext = pRuntime->NewEventContext();
jaepark611adb82016-08-17 11:34:36 -0700257 CFX_WideString sOldValue = pField->GetValue();
258 CFX_WideString sValue = sOldValue;
tsepez4cf55152016-11-02 14:37:54 -0700259 bool bRC = true;
jaepark611adb82016-08-17 11:34:36 -0700260 pContext->OnField_Calculate(pFormField, pField, sValue, bRC);
261
262 CFX_WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700263 bool bRet = pContext->RunScript(csJS, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800264 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700265 if (bRet && bRC && sValue.Compare(sOldValue) != 0)
tsepez4cf55152016-11-02 14:37:54 -0700266 pField->SetValue(sValue, true);
jaepark611adb82016-08-17 11:34:36 -0700267 }
tsepez4cf55152016-11-02 14:37:54 -0700268 m_bBusy = false;
jaepark611adb82016-08-17 11:34:36 -0700269}
270
271CFX_WideString CPDFSDK_InterForm::OnFormat(CPDF_FormField* pFormField,
tsepez4cf55152016-11-02 14:37:54 -0700272 bool& bFormatted) {
jaepark611adb82016-08-17 11:34:36 -0700273 CFX_WideString sValue = pFormField->GetValue();
dsinclair690c0332016-10-11 09:13:01 -0700274 if (!m_pFormFillEnv->IsJSInitiated()) {
tsepez4cf55152016-11-02 14:37:54 -0700275 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700276 return sValue;
277 }
278
dsinclair690c0332016-10-11 09:13:01 -0700279 IJS_Runtime* pRuntime = m_pFormFillEnv->GetJSRuntime();
jaepark611adb82016-08-17 11:34:36 -0700280 if (pFormField->GetFieldType() == FIELDTYPE_COMBOBOX &&
281 pFormField->CountSelectedItems() > 0) {
282 int index = pFormField->GetSelectedIndex(0);
283 if (index >= 0)
284 sValue = pFormField->GetOptionLabel(index);
285 }
286
tsepez4cf55152016-11-02 14:37:54 -0700287 bFormatted = false;
jaepark611adb82016-08-17 11:34:36 -0700288
289 CPDF_AAction aAction = pFormField->GetAdditionalAction();
290 if (aAction.GetDict() && aAction.ActionExist(CPDF_AAction::Format)) {
291 CPDF_Action action = aAction.GetAction(CPDF_AAction::Format);
292 if (action.GetDict()) {
293 CFX_WideString script = action.GetJavaScript();
294 if (!script.IsEmpty()) {
295 CFX_WideString Value = sValue;
296
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800297 IJS_EventContext* pContext = pRuntime->NewEventContext();
tsepez4cf55152016-11-02 14:37:54 -0700298 pContext->OnField_Format(pFormField, Value, true);
jaepark611adb82016-08-17 11:34:36 -0700299 CFX_WideString sInfo;
tsepez4cf55152016-11-02 14:37:54 -0700300 bool bRet = pContext->RunScript(script, &sInfo);
Tom Sepezd6ae2af2017-02-16 11:49:55 -0800301 pRuntime->ReleaseEventContext(pContext);
jaepark611adb82016-08-17 11:34:36 -0700302 if (bRet) {
303 sValue = Value;
tsepez4cf55152016-11-02 14:37:54 -0700304 bFormatted = true;
jaepark611adb82016-08-17 11:34:36 -0700305 }
306 }
307 }
308 }
jaepark611adb82016-08-17 11:34:36 -0700309 return sValue;
310}
311
312void CPDFSDK_InterForm::ResetFieldAppearance(CPDF_FormField* pFormField,
tsepeza31da742016-09-08 11:28:14 -0700313 const CFX_WideString* sValue,
tsepez4cf55152016-11-02 14:37:54 -0700314 bool bValueChanged) {
jaepark611adb82016-08-17 11:34:36 -0700315 for (int i = 0, sz = pFormField->CountControls(); i < sz; i++) {
316 CPDF_FormControl* pFormCtrl = pFormField->GetControl(i);
317 ASSERT(pFormCtrl);
dsinclairc5267c52016-11-04 15:35:12 -0700318 if (CPDFSDK_Widget* pWidget = GetWidget(pFormCtrl))
jaepark611adb82016-08-17 11:34:36 -0700319 pWidget->ResetAppearance(sValue, bValueChanged);
320 }
321}
322
323void CPDFSDK_InterForm::UpdateField(CPDF_FormField* pFormField) {
324 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();
dsinclair7cbe68e2016-10-12 11:56:23 -0700330 CPDFSDK_PageView* pPageView = m_pFormFillEnv->GetPageView(pPage, false);
dsinclair690c0332016-10-11 09:13:01 -0700331 FX_RECT rcBBox = m_pFormFillEnv->GetInteractiveFormFiller()->GetViewBBox(
332 pPageView, pWidget);
jaepark611adb82016-08-17 11:34:36 -0700333
dsinclair690c0332016-10-11 09:13:01 -0700334 m_pFormFillEnv->Invalidate(pPage, rcBBox.left, rcBBox.top, rcBBox.right,
335 rcBBox.bottom);
jaepark611adb82016-08-17 11:34:36 -0700336 }
337 }
338}
339
tsepez4cf55152016-11-02 14:37:54 -0700340bool CPDFSDK_InterForm::OnKeyStrokeCommit(CPDF_FormField* pFormField,
341 const CFX_WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700342 CPDF_AAction aAction = pFormField->GetAdditionalAction();
343 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::KeyStroke))
tsepez4cf55152016-11-02 14:37:54 -0700344 return true;
jaepark611adb82016-08-17 11:34:36 -0700345
346 CPDF_Action action = aAction.GetAction(CPDF_AAction::KeyStroke);
347 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700348 return true;
jaepark611adb82016-08-17 11:34:36 -0700349
dsinclair690c0332016-10-11 09:13:01 -0700350 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHander();
jaepark611adb82016-08-17 11:34:36 -0700351 PDFSDK_FieldAction fa;
dsinclair690c0332016-10-11 09:13:01 -0700352 fa.bModifier = m_pFormFillEnv->IsCTRLKeyDown(0);
353 fa.bShift = m_pFormFillEnv->IsSHIFTKeyDown(0);
jaepark611adb82016-08-17 11:34:36 -0700354 fa.sValue = csValue;
355 pActionHandler->DoAction_FieldJavaScript(action, CPDF_AAction::KeyStroke,
dsinclair19c198b2016-10-11 12:51:37 -0700356 m_pFormFillEnv, pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700357 return fa.bRC;
358}
359
tsepez4cf55152016-11-02 14:37:54 -0700360bool CPDFSDK_InterForm::OnValidate(CPDF_FormField* pFormField,
361 const CFX_WideString& csValue) {
jaepark611adb82016-08-17 11:34:36 -0700362 CPDF_AAction aAction = pFormField->GetAdditionalAction();
363 if (!aAction.GetDict() || !aAction.ActionExist(CPDF_AAction::Validate))
tsepez4cf55152016-11-02 14:37:54 -0700364 return true;
jaepark611adb82016-08-17 11:34:36 -0700365
366 CPDF_Action action = aAction.GetAction(CPDF_AAction::Validate);
367 if (!action.GetDict())
tsepez4cf55152016-11-02 14:37:54 -0700368 return true;
jaepark611adb82016-08-17 11:34:36 -0700369
dsinclair690c0332016-10-11 09:13:01 -0700370 CPDFSDK_ActionHandler* pActionHandler = m_pFormFillEnv->GetActionHander();
jaepark611adb82016-08-17 11:34:36 -0700371 PDFSDK_FieldAction fa;
dsinclair690c0332016-10-11 09:13:01 -0700372 fa.bModifier = m_pFormFillEnv->IsCTRLKeyDown(0);
373 fa.bShift = m_pFormFillEnv->IsSHIFTKeyDown(0);
jaepark611adb82016-08-17 11:34:36 -0700374 fa.sValue = csValue;
375 pActionHandler->DoAction_FieldJavaScript(action, CPDF_AAction::Validate,
dsinclair19c198b2016-10-11 12:51:37 -0700376 m_pFormFillEnv, pFormField, fa);
jaepark611adb82016-08-17 11:34:36 -0700377 return fa.bRC;
378}
379
tsepez4cf55152016-11-02 14:37:54 -0700380bool CPDFSDK_InterForm::DoAction_Hide(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700381 ASSERT(action.GetDict());
382
383 CPDF_ActionFields af(&action);
384 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
385 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
386
387 bool bHide = action.GetHideStatus();
tsepez4cf55152016-11-02 14:37:54 -0700388 bool bChanged = false;
jaepark611adb82016-08-17 11:34:36 -0700389
390 for (CPDF_FormField* pField : fields) {
391 for (int i = 0, sz = pField->CountControls(); i < sz; ++i) {
392 CPDF_FormControl* pControl = pField->GetControl(i);
393 ASSERT(pControl);
394
dsinclairc5267c52016-11-04 15:35:12 -0700395 if (CPDFSDK_Widget* pWidget = GetWidget(pControl)) {
jaepark611adb82016-08-17 11:34:36 -0700396 uint32_t nFlags = pWidget->GetFlags();
397 nFlags &= ~ANNOTFLAG_INVISIBLE;
398 nFlags &= ~ANNOTFLAG_NOVIEW;
399 if (bHide)
400 nFlags |= ANNOTFLAG_HIDDEN;
401 else
402 nFlags &= ~ANNOTFLAG_HIDDEN;
403 pWidget->SetFlags(nFlags);
404 pWidget->GetPageView()->UpdateView(pWidget);
tsepez4cf55152016-11-02 14:37:54 -0700405 bChanged = true;
jaepark611adb82016-08-17 11:34:36 -0700406 }
407 }
408 }
409
410 return bChanged;
411}
412
tsepez4cf55152016-11-02 14:37:54 -0700413bool CPDFSDK_InterForm::DoAction_SubmitForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700414 CFX_WideString sDestination = action.GetFilePath();
415 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700416 return false;
jaepark611adb82016-08-17 11:34:36 -0700417
418 CPDF_Dictionary* pActionDict = action.GetDict();
419 if (pActionDict->KeyExist("Fields")) {
420 CPDF_ActionFields af(&action);
421 uint32_t dwFlags = action.GetFlags();
422 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
423 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
424 if (!fields.empty()) {
425 bool bIncludeOrExclude = !(dwFlags & 0x01);
Nicolas Penaa478dc52017-01-23 15:48:51 -0500426 if (!m_pInterForm->CheckRequiredFields(&fields, bIncludeOrExclude))
tsepez4cf55152016-11-02 14:37:54 -0700427 return false;
jaepark611adb82016-08-17 11:34:36 -0700428
429 return SubmitFields(sDestination, fields, bIncludeOrExclude, false);
430 }
431 }
Nicolas Penaa478dc52017-01-23 15:48:51 -0500432 if (!m_pInterForm->CheckRequiredFields(nullptr, true))
tsepez4cf55152016-11-02 14:37:54 -0700433 return false;
jaepark611adb82016-08-17 11:34:36 -0700434
tsepez4cf55152016-11-02 14:37:54 -0700435 return SubmitForm(sDestination, false);
jaepark611adb82016-08-17 11:34:36 -0700436}
437
tsepez4cf55152016-11-02 14:37:54 -0700438bool CPDFSDK_InterForm::SubmitFields(const CFX_WideString& csDestination,
439 const std::vector<CPDF_FormField*>& fields,
440 bool bIncludeOrExclude,
441 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700442 CFX_ByteTextBuf textBuf;
443 ExportFieldsToFDFTextBuf(fields, bIncludeOrExclude, textBuf);
444
445 uint8_t* pBuffer = textBuf.GetBuffer();
446 FX_STRSIZE nBufSize = textBuf.GetLength();
447
448 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize))
tsepez4cf55152016-11-02 14:37:54 -0700449 return false;
jaepark611adb82016-08-17 11:34:36 -0700450
dsinclair690c0332016-10-11 09:13:01 -0700451 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, csDestination.c_str());
tsepez4cf55152016-11-02 14:37:54 -0700452 return true;
jaepark611adb82016-08-17 11:34:36 -0700453}
454
tsepez4cf55152016-11-02 14:37:54 -0700455bool CPDFSDK_InterForm::FDFToURLEncodedData(CFX_WideString csFDFFile,
456 CFX_WideString csTxtFile) {
457 return true;
jaepark611adb82016-08-17 11:34:36 -0700458}
459
tsepez4cf55152016-11-02 14:37:54 -0700460bool CPDFSDK_InterForm::FDFToURLEncodedData(uint8_t*& pBuf,
461 FX_STRSIZE& nBufSize) {
tsepez05e01692016-11-28 17:30:09 -0800462 std::unique_ptr<CFDF_Document> pFDF =
463 CFDF_Document::ParseMemory(pBuf, nBufSize);
jaepark611adb82016-08-17 11:34:36 -0700464 if (!pFDF)
tsepez4cf55152016-11-02 14:37:54 -0700465 return true;
jaepark611adb82016-08-17 11:34:36 -0700466
dsinclair38fd8442016-09-15 10:15:32 -0700467 CPDF_Dictionary* pMainDict = pFDF->GetRoot()->GetDictFor("FDF");
jaepark611adb82016-08-17 11:34:36 -0700468 if (!pMainDict)
tsepez4cf55152016-11-02 14:37:54 -0700469 return false;
jaepark611adb82016-08-17 11:34:36 -0700470
dsinclair38fd8442016-09-15 10:15:32 -0700471 CPDF_Array* pFields = pMainDict->GetArrayFor("Fields");
jaepark611adb82016-08-17 11:34:36 -0700472 if (!pFields)
tsepez4cf55152016-11-02 14:37:54 -0700473 return false;
jaepark611adb82016-08-17 11:34:36 -0700474
475 CFX_ByteTextBuf fdfEncodedData;
476 for (uint32_t i = 0; i < pFields->GetCount(); i++) {
477 CPDF_Dictionary* pField = pFields->GetDictAt(i);
478 if (!pField)
479 continue;
480 CFX_WideString name;
dsinclair38fd8442016-09-15 10:15:32 -0700481 name = pField->GetUnicodeTextFor("T");
jaepark611adb82016-08-17 11:34:36 -0700482 CFX_ByteString name_b = CFX_ByteString::FromUnicode(name);
dsinclair38fd8442016-09-15 10:15:32 -0700483 CFX_ByteString csBValue = pField->GetStringFor("V");
jaepark611adb82016-08-17 11:34:36 -0700484 CFX_WideString csWValue = PDF_DecodeText(csBValue);
485 CFX_ByteString csValue_b = CFX_ByteString::FromUnicode(csWValue);
486
487 fdfEncodedData << name_b.GetBuffer(name_b.GetLength());
488 name_b.ReleaseBuffer();
489 fdfEncodedData << "=";
490 fdfEncodedData << csValue_b.GetBuffer(csValue_b.GetLength());
491 csValue_b.ReleaseBuffer();
492 if (i != pFields->GetCount() - 1)
493 fdfEncodedData << "&";
494 }
495
496 nBufSize = fdfEncodedData.GetLength();
497 pBuf = FX_Alloc(uint8_t, nBufSize);
498 FXSYS_memcpy(pBuf, fdfEncodedData.GetBuffer(), nBufSize);
tsepez4cf55152016-11-02 14:37:54 -0700499 return true;
jaepark611adb82016-08-17 11:34:36 -0700500}
501
tsepez4cf55152016-11-02 14:37:54 -0700502bool CPDFSDK_InterForm::ExportFieldsToFDFTextBuf(
jaepark611adb82016-08-17 11:34:36 -0700503 const std::vector<CPDF_FormField*>& fields,
504 bool bIncludeOrExclude,
505 CFX_ByteTextBuf& textBuf) {
tsepez05e01692016-11-28 17:30:09 -0800506 std::unique_ptr<CFDF_Document> pFDF =
dsinclair7cbe68e2016-10-12 11:56:23 -0700507 m_pInterForm->ExportToFDF(m_pFormFillEnv->JS_docGetFilePath().AsStringC(),
tsepez05e01692016-11-28 17:30:09 -0800508 fields, bIncludeOrExclude, false);
tsepez4cf55152016-11-02 14:37:54 -0700509 return pFDF ? pFDF->WriteBuf(textBuf) : false;
jaepark611adb82016-08-17 11:34:36 -0700510}
511
512CFX_WideString CPDFSDK_InterForm::GetTemporaryFileName(
513 const CFX_WideString& sFileExt) {
514 return L"";
515}
516
tsepez4cf55152016-11-02 14:37:54 -0700517bool CPDFSDK_InterForm::SubmitForm(const CFX_WideString& sDestination,
518 bool bUrlEncoded) {
jaepark611adb82016-08-17 11:34:36 -0700519 if (sDestination.IsEmpty())
tsepez4cf55152016-11-02 14:37:54 -0700520 return false;
jaepark611adb82016-08-17 11:34:36 -0700521
dsinclair7cbe68e2016-10-12 11:56:23 -0700522 if (!m_pFormFillEnv || !m_pInterForm)
tsepez4cf55152016-11-02 14:37:54 -0700523 return false;
jaepark611adb82016-08-17 11:34:36 -0700524
tsepez05e01692016-11-28 17:30:09 -0800525 std::unique_ptr<CFDF_Document> pFDFDoc = m_pInterForm->ExportToFDF(
526 m_pFormFillEnv->JS_docGetFilePath().AsStringC(), false);
jaepark611adb82016-08-17 11:34:36 -0700527 if (!pFDFDoc)
tsepez4cf55152016-11-02 14:37:54 -0700528 return false;
jaepark611adb82016-08-17 11:34:36 -0700529
530 CFX_ByteTextBuf FdfBuffer;
tsepez05e01692016-11-28 17:30:09 -0800531 if (!pFDFDoc->WriteBuf(FdfBuffer))
tsepez4cf55152016-11-02 14:37:54 -0700532 return false;
jaepark611adb82016-08-17 11:34:36 -0700533
534 uint8_t* pBuffer = FdfBuffer.GetBuffer();
535 FX_STRSIZE nBufSize = FdfBuffer.GetLength();
jaepark611adb82016-08-17 11:34:36 -0700536 if (bUrlEncoded && !FDFToURLEncodedData(pBuffer, nBufSize))
tsepez4cf55152016-11-02 14:37:54 -0700537 return false;
jaepark611adb82016-08-17 11:34:36 -0700538
dsinclair690c0332016-10-11 09:13:01 -0700539 m_pFormFillEnv->JS_docSubmitForm(pBuffer, nBufSize, sDestination.c_str());
jaepark611adb82016-08-17 11:34:36 -0700540 if (bUrlEncoded)
541 FX_Free(pBuffer);
542
tsepez4cf55152016-11-02 14:37:54 -0700543 return true;
jaepark611adb82016-08-17 11:34:36 -0700544}
545
tsepez4cf55152016-11-02 14:37:54 -0700546bool CPDFSDK_InterForm::ExportFormToFDFTextBuf(CFX_ByteTextBuf& textBuf) {
tsepez05e01692016-11-28 17:30:09 -0800547 std::unique_ptr<CFDF_Document> pFDF = m_pInterForm->ExportToFDF(
dsinclair7cbe68e2016-10-12 11:56:23 -0700548 m_pFormFillEnv->JS_docGetFilePath().AsStringC(), false);
tsepez05e01692016-11-28 17:30:09 -0800549 return pFDF && pFDF->WriteBuf(textBuf);
jaepark611adb82016-08-17 11:34:36 -0700550}
551
tsepez4cf55152016-11-02 14:37:54 -0700552bool CPDFSDK_InterForm::DoAction_ResetForm(const CPDF_Action& action) {
jaepark611adb82016-08-17 11:34:36 -0700553 ASSERT(action.GetDict());
554
555 CPDF_Dictionary* pActionDict = action.GetDict();
556 if (!pActionDict->KeyExist("Fields"))
557 return m_pInterForm->ResetForm(true);
558
559 CPDF_ActionFields af(&action);
560 uint32_t dwFlags = action.GetFlags();
561
562 std::vector<CPDF_Object*> fieldObjects = af.GetAllFields();
563 std::vector<CPDF_FormField*> fields = GetFieldFromObjects(fieldObjects);
564 return m_pInterForm->ResetForm(fields, !(dwFlags & 0x01), true);
565}
566
tsepez4cf55152016-11-02 14:37:54 -0700567bool CPDFSDK_InterForm::DoAction_ImportData(const CPDF_Action& action) {
568 return false;
jaepark611adb82016-08-17 11:34:36 -0700569}
570
571std::vector<CPDF_FormField*> CPDFSDK_InterForm::GetFieldFromObjects(
572 const std::vector<CPDF_Object*>& objects) const {
573 std::vector<CPDF_FormField*> fields;
574 for (CPDF_Object* pObject : objects) {
575 if (pObject && pObject->IsString()) {
576 CFX_WideString csName = pObject->GetUnicodeText();
577 CPDF_FormField* pField = m_pInterForm->GetField(0, csName);
578 if (pField)
579 fields.push_back(pField);
580 }
581 }
582 return fields;
583}
584
585int CPDFSDK_InterForm::BeforeValueChange(CPDF_FormField* pField,
586 const CFX_WideString& csValue) {
587 int nType = pField->GetFieldType();
588 if (nType != FIELDTYPE_COMBOBOX && nType != FIELDTYPE_TEXTFIELD)
589 return 0;
590
591 if (!OnKeyStrokeCommit(pField, csValue))
592 return -1;
593
594 if (!OnValidate(pField, csValue))
595 return -1;
596
597 return 1;
598}
599
600void CPDFSDK_InterForm::AfterValueChange(CPDF_FormField* pField) {
601#ifdef PDF_ENABLE_XFA
tsepez4cf55152016-11-02 14:37:54 -0700602 SynchronizeField(pField, false);
jaepark611adb82016-08-17 11:34:36 -0700603#endif // PDF_ENABLE_XFA
604 int nType = pField->GetFieldType();
605 if (nType == FIELDTYPE_COMBOBOX || nType == FIELDTYPE_TEXTFIELD) {
606 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700607 bool bFormatted = false;
tsepez8c2a8cd2016-09-07 15:29:11 -0700608 CFX_WideString sValue = OnFormat(pField, bFormatted);
tsepez4cf55152016-11-02 14:37:54 -0700609 ResetFieldAppearance(pField, bFormatted ? &sValue : nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700610 UpdateField(pField);
611 }
612}
613
614int CPDFSDK_InterForm::BeforeSelectionChange(CPDF_FormField* pField,
615 const CFX_WideString& csValue) {
616 if (pField->GetFieldType() != FIELDTYPE_LISTBOX)
617 return 0;
618
619 if (!OnKeyStrokeCommit(pField, csValue))
620 return -1;
621
622 if (!OnValidate(pField, csValue))
623 return -1;
624
625 return 1;
626}
627
628void CPDFSDK_InterForm::AfterSelectionChange(CPDF_FormField* pField) {
629 if (pField->GetFieldType() != FIELDTYPE_LISTBOX)
630 return;
631
632 OnCalculate(pField);
tsepez4cf55152016-11-02 14:37:54 -0700633 ResetFieldAppearance(pField, nullptr, true);
jaepark611adb82016-08-17 11:34:36 -0700634 UpdateField(pField);
635}
636
637void CPDFSDK_InterForm::AfterCheckedStatusChange(CPDF_FormField* pField) {
638 int nType = pField->GetFieldType();
639 if (nType != FIELDTYPE_CHECKBOX && nType != FIELDTYPE_RADIOBUTTON)
640 return;
641
642 OnCalculate(pField);
643 UpdateField(pField);
644}
645
646int CPDFSDK_InterForm::BeforeFormReset(CPDF_InterForm* pForm) {
647 return 0;
648}
649
650void CPDFSDK_InterForm::AfterFormReset(CPDF_InterForm* pForm) {
651 OnCalculate(nullptr);
652}
653
654int CPDFSDK_InterForm::BeforeFormImportData(CPDF_InterForm* pForm) {
655 return 0;
656}
657
658void CPDFSDK_InterForm::AfterFormImportData(CPDF_InterForm* pForm) {
659 OnCalculate(nullptr);
660}
661
tsepez4cf55152016-11-02 14:37:54 -0700662bool CPDFSDK_InterForm::IsNeedHighLight(int nFieldType) {
jaepark611adb82016-08-17 11:34:36 -0700663 if (nFieldType < 1 || nFieldType > kNumFieldTypes)
tsepez4cf55152016-11-02 14:37:54 -0700664 return false;
jaepark611adb82016-08-17 11:34:36 -0700665 return m_bNeedHightlight[nFieldType - 1];
666}
667
668void CPDFSDK_InterForm::RemoveAllHighLight() {
669 for (int i = 0; i < kNumFieldTypes; ++i)
tsepez4cf55152016-11-02 14:37:54 -0700670 m_bNeedHightlight[i] = false;
jaepark611adb82016-08-17 11:34:36 -0700671}
672
673void CPDFSDK_InterForm::SetHighlightColor(FX_COLORREF clr, int nFieldType) {
674 if (nFieldType < 0 || nFieldType > kNumFieldTypes)
675 return;
676 switch (nFieldType) {
677 case 0: {
678 for (int i = 0; i < kNumFieldTypes; ++i) {
679 m_aHighlightColor[i] = clr;
tsepez4cf55152016-11-02 14:37:54 -0700680 m_bNeedHightlight[i] = true;
jaepark611adb82016-08-17 11:34:36 -0700681 }
682 break;
683 }
684 default: {
685 m_aHighlightColor[nFieldType - 1] = clr;
tsepez4cf55152016-11-02 14:37:54 -0700686 m_bNeedHightlight[nFieldType - 1] = true;
jaepark611adb82016-08-17 11:34:36 -0700687 break;
688 }
689 }
690}
691
692FX_COLORREF CPDFSDK_InterForm::GetHighlightColor(int nFieldType) {
693 if (nFieldType < 0 || nFieldType > kNumFieldTypes)
694 return FXSYS_RGB(255, 255, 255);
695 if (nFieldType == 0)
696 return m_aHighlightColor[0];
697 return m_aHighlightColor[nFieldType - 1];
698}