1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2017 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker
5*3ac0a46fSAndroid Build Coastguard Worker // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6*3ac0a46fSAndroid Build Coastguard Worker
7*3ac0a46fSAndroid Build Coastguard Worker #include "fxjs/xfa/cjx_instancemanager.h"
8*3ac0a46fSAndroid Build Coastguard Worker
9*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
10*3ac0a46fSAndroid Build Coastguard Worker #include <vector>
11*3ac0a46fSAndroid Build Coastguard Worker
12*3ac0a46fSAndroid Build Coastguard Worker #include "fxjs/fxv8.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "fxjs/js_resources.h"
14*3ac0a46fSAndroid Build Coastguard Worker #include "fxjs/xfa/cfxjse_engine.h"
15*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/notreached.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "v8/include/v8-object.h"
17*3ac0a46fSAndroid Build Coastguard Worker #include "v8/include/v8-primitive.h"
18*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/cxfa_ffdoc.h"
19*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/cxfa_ffnotify.h"
20*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/parser/cxfa_document.h"
21*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/parser/cxfa_instancemanager.h"
22*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/parser/cxfa_occur.h"
23*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fxfa/parser/cxfa_subform.h"
24*3ac0a46fSAndroid Build Coastguard Worker
25*3ac0a46fSAndroid Build Coastguard Worker const CJX_MethodSpec CJX_InstanceManager::MethodSpecs[] = {
26*3ac0a46fSAndroid Build Coastguard Worker {"addInstance", addInstance_static},
27*3ac0a46fSAndroid Build Coastguard Worker {"insertInstance", insertInstance_static},
28*3ac0a46fSAndroid Build Coastguard Worker {"moveInstance", moveInstance_static},
29*3ac0a46fSAndroid Build Coastguard Worker {"removeInstance", removeInstance_static},
30*3ac0a46fSAndroid Build Coastguard Worker {"setInstances", setInstances_static}};
31*3ac0a46fSAndroid Build Coastguard Worker
CJX_InstanceManager(CXFA_InstanceManager * mgr)32*3ac0a46fSAndroid Build Coastguard Worker CJX_InstanceManager::CJX_InstanceManager(CXFA_InstanceManager* mgr)
33*3ac0a46fSAndroid Build Coastguard Worker : CJX_Node(mgr) {
34*3ac0a46fSAndroid Build Coastguard Worker DefineMethods(MethodSpecs);
35*3ac0a46fSAndroid Build Coastguard Worker }
36*3ac0a46fSAndroid Build Coastguard Worker
37*3ac0a46fSAndroid Build Coastguard Worker CJX_InstanceManager::~CJX_InstanceManager() = default;
38*3ac0a46fSAndroid Build Coastguard Worker
DynamicTypeIs(TypeTag eType) const39*3ac0a46fSAndroid Build Coastguard Worker bool CJX_InstanceManager::DynamicTypeIs(TypeTag eType) const {
40*3ac0a46fSAndroid Build Coastguard Worker return eType == static_type__ || ParentType__::DynamicTypeIs(eType);
41*3ac0a46fSAndroid Build Coastguard Worker }
42*3ac0a46fSAndroid Build Coastguard Worker
SetInstances(v8::Isolate * pIsolate,int32_t iDesired)43*3ac0a46fSAndroid Build Coastguard Worker int32_t CJX_InstanceManager::SetInstances(v8::Isolate* pIsolate,
44*3ac0a46fSAndroid Build Coastguard Worker int32_t iDesired) {
45*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
46*3ac0a46fSAndroid Build Coastguard Worker int32_t iMin = occur ? occur->GetMin() : CXFA_Occur::kDefaultMin;
47*3ac0a46fSAndroid Build Coastguard Worker if (iDesired < iMin) {
48*3ac0a46fSAndroid Build Coastguard Worker ThrowTooManyOccurrencesException(pIsolate, L"min");
49*3ac0a46fSAndroid Build Coastguard Worker return 1;
50*3ac0a46fSAndroid Build Coastguard Worker }
51*3ac0a46fSAndroid Build Coastguard Worker
52*3ac0a46fSAndroid Build Coastguard Worker int32_t iMax = occur ? occur->GetMax() : CXFA_Occur::kDefaultMax;
53*3ac0a46fSAndroid Build Coastguard Worker if (iMax >= 0 && iDesired > iMax) {
54*3ac0a46fSAndroid Build Coastguard Worker ThrowTooManyOccurrencesException(pIsolate, L"max");
55*3ac0a46fSAndroid Build Coastguard Worker return 2;
56*3ac0a46fSAndroid Build Coastguard Worker }
57*3ac0a46fSAndroid Build Coastguard Worker
58*3ac0a46fSAndroid Build Coastguard Worker int32_t iCount = GetXFANode()->GetCount();
59*3ac0a46fSAndroid Build Coastguard Worker if (iDesired == iCount)
60*3ac0a46fSAndroid Build Coastguard Worker return 0;
61*3ac0a46fSAndroid Build Coastguard Worker
62*3ac0a46fSAndroid Build Coastguard Worker if (iDesired < iCount) {
63*3ac0a46fSAndroid Build Coastguard Worker WideString wsInstManagerName = GetCData(XFA_Attribute::Name);
64*3ac0a46fSAndroid Build Coastguard Worker WideString wsInstanceName = WideString(
65*3ac0a46fSAndroid Build Coastguard Worker wsInstManagerName.IsEmpty()
66*3ac0a46fSAndroid Build Coastguard Worker ? wsInstManagerName
67*3ac0a46fSAndroid Build Coastguard Worker : wsInstManagerName.Last(wsInstManagerName.GetLength() - 1));
68*3ac0a46fSAndroid Build Coastguard Worker uint32_t dInstanceNameHash =
69*3ac0a46fSAndroid Build Coastguard Worker FX_HashCode_GetW(wsInstanceName.AsStringView());
70*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pPrevSibling = iDesired == 0
71*3ac0a46fSAndroid Build Coastguard Worker ? GetXFANode()
72*3ac0a46fSAndroid Build Coastguard Worker : GetXFANode()->GetItemIfExists(iDesired - 1);
73*3ac0a46fSAndroid Build Coastguard Worker if (!pPrevSibling) {
74*3ac0a46fSAndroid Build Coastguard Worker // TODO(dsinclair): Better error?
75*3ac0a46fSAndroid Build Coastguard Worker ThrowIndexOutOfBoundsException(pIsolate);
76*3ac0a46fSAndroid Build Coastguard Worker return 0;
77*3ac0a46fSAndroid Build Coastguard Worker }
78*3ac0a46fSAndroid Build Coastguard Worker
79*3ac0a46fSAndroid Build Coastguard Worker while (iCount > iDesired) {
80*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pRemoveInstance = pPrevSibling->GetNextSibling();
81*3ac0a46fSAndroid Build Coastguard Worker if (pRemoveInstance->GetElementType() != XFA_Element::Subform &&
82*3ac0a46fSAndroid Build Coastguard Worker pRemoveInstance->GetElementType() != XFA_Element::SubformSet) {
83*3ac0a46fSAndroid Build Coastguard Worker continue;
84*3ac0a46fSAndroid Build Coastguard Worker }
85*3ac0a46fSAndroid Build Coastguard Worker if (pRemoveInstance->GetElementType() == XFA_Element::InstanceManager) {
86*3ac0a46fSAndroid Build Coastguard Worker NOTREACHED();
87*3ac0a46fSAndroid Build Coastguard Worker break;
88*3ac0a46fSAndroid Build Coastguard Worker }
89*3ac0a46fSAndroid Build Coastguard Worker if (pRemoveInstance->GetNameHash() == dInstanceNameHash) {
90*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->RemoveItem(pRemoveInstance, true);
91*3ac0a46fSAndroid Build Coastguard Worker iCount--;
92*3ac0a46fSAndroid Build Coastguard Worker }
93*3ac0a46fSAndroid Build Coastguard Worker }
94*3ac0a46fSAndroid Build Coastguard Worker } else {
95*3ac0a46fSAndroid Build Coastguard Worker while (iCount < iDesired) {
96*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pNewInstance = GetXFANode()->CreateInstanceIfPossible(true);
97*3ac0a46fSAndroid Build Coastguard Worker if (!pNewInstance)
98*3ac0a46fSAndroid Build Coastguard Worker return 0;
99*3ac0a46fSAndroid Build Coastguard Worker
100*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->InsertItem(pNewInstance, iCount, iCount, false);
101*3ac0a46fSAndroid Build Coastguard Worker ++iCount;
102*3ac0a46fSAndroid Build Coastguard Worker
103*3ac0a46fSAndroid Build Coastguard Worker CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
104*3ac0a46fSAndroid Build Coastguard Worker if (!pNotify)
105*3ac0a46fSAndroid Build Coastguard Worker return 0;
106*3ac0a46fSAndroid Build Coastguard Worker
107*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunNodeInitialize(pNewInstance);
108*3ac0a46fSAndroid Build Coastguard Worker }
109*3ac0a46fSAndroid Build Coastguard Worker }
110*3ac0a46fSAndroid Build Coastguard Worker GetDocument()->GetLayoutProcessor()->SetHasChangedContainer();
111*3ac0a46fSAndroid Build Coastguard Worker return 0;
112*3ac0a46fSAndroid Build Coastguard Worker }
113*3ac0a46fSAndroid Build Coastguard Worker
MoveInstance(v8::Isolate * pIsolate,int32_t iTo,int32_t iFrom)114*3ac0a46fSAndroid Build Coastguard Worker int32_t CJX_InstanceManager::MoveInstance(v8::Isolate* pIsolate,
115*3ac0a46fSAndroid Build Coastguard Worker int32_t iTo,
116*3ac0a46fSAndroid Build Coastguard Worker int32_t iFrom) {
117*3ac0a46fSAndroid Build Coastguard Worker int32_t iCount = GetXFANode()->GetCount();
118*3ac0a46fSAndroid Build Coastguard Worker if (iFrom > iCount || iTo > iCount - 1) {
119*3ac0a46fSAndroid Build Coastguard Worker ThrowIndexOutOfBoundsException(pIsolate);
120*3ac0a46fSAndroid Build Coastguard Worker return 1;
121*3ac0a46fSAndroid Build Coastguard Worker }
122*3ac0a46fSAndroid Build Coastguard Worker if (iFrom < 0 || iTo < 0 || iFrom == iTo)
123*3ac0a46fSAndroid Build Coastguard Worker return 0;
124*3ac0a46fSAndroid Build Coastguard Worker
125*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pMoveInstance = GetXFANode()->GetItemIfExists(iFrom);
126*3ac0a46fSAndroid Build Coastguard Worker if (!pMoveInstance) {
127*3ac0a46fSAndroid Build Coastguard Worker ThrowIndexOutOfBoundsException(pIsolate);
128*3ac0a46fSAndroid Build Coastguard Worker return 1;
129*3ac0a46fSAndroid Build Coastguard Worker }
130*3ac0a46fSAndroid Build Coastguard Worker
131*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->RemoveItem(pMoveInstance, false);
132*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->InsertItem(pMoveInstance, iTo, iCount - 1, true);
133*3ac0a46fSAndroid Build Coastguard Worker GetDocument()->GetLayoutProcessor()->SetHasChangedContainer();
134*3ac0a46fSAndroid Build Coastguard Worker return 0;
135*3ac0a46fSAndroid Build Coastguard Worker }
136*3ac0a46fSAndroid Build Coastguard Worker
moveInstance(CFXJSE_Engine * runtime,const std::vector<v8::Local<v8::Value>> & params)137*3ac0a46fSAndroid Build Coastguard Worker CJS_Result CJX_InstanceManager::moveInstance(
138*3ac0a46fSAndroid Build Coastguard Worker CFXJSE_Engine* runtime,
139*3ac0a46fSAndroid Build Coastguard Worker const std::vector<v8::Local<v8::Value>>& params) {
140*3ac0a46fSAndroid Build Coastguard Worker CXFA_Document* doc = runtime->GetDocument();
141*3ac0a46fSAndroid Build Coastguard Worker if (doc->GetFormType() != FormType::kXFAFull)
142*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kNotSupportedError);
143*3ac0a46fSAndroid Build Coastguard Worker
144*3ac0a46fSAndroid Build Coastguard Worker if (params.size() != 2)
145*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
146*3ac0a46fSAndroid Build Coastguard Worker
147*3ac0a46fSAndroid Build Coastguard Worker int32_t iFrom = runtime->ToInt32(params[0]);
148*3ac0a46fSAndroid Build Coastguard Worker int32_t iTo = runtime->ToInt32(params[1]);
149*3ac0a46fSAndroid Build Coastguard Worker MoveInstance(runtime->GetIsolate(), iTo, iFrom);
150*3ac0a46fSAndroid Build Coastguard Worker
151*3ac0a46fSAndroid Build Coastguard Worker CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
152*3ac0a46fSAndroid Build Coastguard Worker if (!pNotify)
153*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success();
154*3ac0a46fSAndroid Build Coastguard Worker
155*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pXFA = GetXFANode();
156*3ac0a46fSAndroid Build Coastguard Worker auto* pToInstance = CXFA_Subform::FromNode(pXFA->GetItemIfExists(iTo));
157*3ac0a46fSAndroid Build Coastguard Worker if (pToInstance)
158*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunSubformIndexChange(pToInstance);
159*3ac0a46fSAndroid Build Coastguard Worker
160*3ac0a46fSAndroid Build Coastguard Worker auto* pFromInstance = CXFA_Subform::FromNode(pXFA->GetItemIfExists(iFrom));
161*3ac0a46fSAndroid Build Coastguard Worker if (pFromInstance)
162*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunSubformIndexChange(pFromInstance);
163*3ac0a46fSAndroid Build Coastguard Worker
164*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success();
165*3ac0a46fSAndroid Build Coastguard Worker }
166*3ac0a46fSAndroid Build Coastguard Worker
removeInstance(CFXJSE_Engine * runtime,const std::vector<v8::Local<v8::Value>> & params)167*3ac0a46fSAndroid Build Coastguard Worker CJS_Result CJX_InstanceManager::removeInstance(
168*3ac0a46fSAndroid Build Coastguard Worker CFXJSE_Engine* runtime,
169*3ac0a46fSAndroid Build Coastguard Worker const std::vector<v8::Local<v8::Value>>& params) {
170*3ac0a46fSAndroid Build Coastguard Worker CXFA_Document* doc = runtime->GetDocument();
171*3ac0a46fSAndroid Build Coastguard Worker if (doc->GetFormType() != FormType::kXFAFull)
172*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kNotSupportedError);
173*3ac0a46fSAndroid Build Coastguard Worker
174*3ac0a46fSAndroid Build Coastguard Worker if (params.size() != 1)
175*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
176*3ac0a46fSAndroid Build Coastguard Worker
177*3ac0a46fSAndroid Build Coastguard Worker int32_t iIndex = runtime->ToInt32(params[0]);
178*3ac0a46fSAndroid Build Coastguard Worker int32_t iCount = GetXFANode()->GetCount();
179*3ac0a46fSAndroid Build Coastguard Worker if (iIndex < 0 || iIndex >= iCount)
180*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kInvalidInputError);
181*3ac0a46fSAndroid Build Coastguard Worker
182*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
183*3ac0a46fSAndroid Build Coastguard Worker int32_t iMin = occur ? occur->GetMin() : CXFA_Occur::kDefaultMin;
184*3ac0a46fSAndroid Build Coastguard Worker if (iCount - 1 < iMin)
185*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kTooManyOccurrences);
186*3ac0a46fSAndroid Build Coastguard Worker
187*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pRemoveInstance = GetXFANode()->GetItemIfExists(iIndex);
188*3ac0a46fSAndroid Build Coastguard Worker if (!pRemoveInstance)
189*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
190*3ac0a46fSAndroid Build Coastguard Worker
191*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->RemoveItem(pRemoveInstance, true);
192*3ac0a46fSAndroid Build Coastguard Worker
193*3ac0a46fSAndroid Build Coastguard Worker CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
194*3ac0a46fSAndroid Build Coastguard Worker if (pNotify) {
195*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pXFA = GetXFANode();
196*3ac0a46fSAndroid Build Coastguard Worker for (int32_t i = iIndex; i < iCount - 1; i++) {
197*3ac0a46fSAndroid Build Coastguard Worker auto* pSubformInstance = CXFA_Subform::FromNode(pXFA->GetItemIfExists(i));
198*3ac0a46fSAndroid Build Coastguard Worker if (pSubformInstance)
199*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunSubformIndexChange(pSubformInstance);
200*3ac0a46fSAndroid Build Coastguard Worker }
201*3ac0a46fSAndroid Build Coastguard Worker }
202*3ac0a46fSAndroid Build Coastguard Worker GetDocument()->GetLayoutProcessor()->SetHasChangedContainer();
203*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success();
204*3ac0a46fSAndroid Build Coastguard Worker }
205*3ac0a46fSAndroid Build Coastguard Worker
setInstances(CFXJSE_Engine * runtime,const std::vector<v8::Local<v8::Value>> & params)206*3ac0a46fSAndroid Build Coastguard Worker CJS_Result CJX_InstanceManager::setInstances(
207*3ac0a46fSAndroid Build Coastguard Worker CFXJSE_Engine* runtime,
208*3ac0a46fSAndroid Build Coastguard Worker const std::vector<v8::Local<v8::Value>>& params) {
209*3ac0a46fSAndroid Build Coastguard Worker CXFA_Document* doc = runtime->GetDocument();
210*3ac0a46fSAndroid Build Coastguard Worker if (doc->GetFormType() != FormType::kXFAFull)
211*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kNotSupportedError);
212*3ac0a46fSAndroid Build Coastguard Worker
213*3ac0a46fSAndroid Build Coastguard Worker if (params.size() != 1)
214*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
215*3ac0a46fSAndroid Build Coastguard Worker
216*3ac0a46fSAndroid Build Coastguard Worker SetInstances(runtime->GetIsolate(), runtime->ToInt32(params[0]));
217*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success();
218*3ac0a46fSAndroid Build Coastguard Worker }
219*3ac0a46fSAndroid Build Coastguard Worker
addInstance(CFXJSE_Engine * runtime,const std::vector<v8::Local<v8::Value>> & params)220*3ac0a46fSAndroid Build Coastguard Worker CJS_Result CJX_InstanceManager::addInstance(
221*3ac0a46fSAndroid Build Coastguard Worker CFXJSE_Engine* runtime,
222*3ac0a46fSAndroid Build Coastguard Worker const std::vector<v8::Local<v8::Value>>& params) {
223*3ac0a46fSAndroid Build Coastguard Worker CXFA_Document* doc = runtime->GetDocument();
224*3ac0a46fSAndroid Build Coastguard Worker if (doc->GetFormType() != FormType::kXFAFull)
225*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kNotSupportedError);
226*3ac0a46fSAndroid Build Coastguard Worker
227*3ac0a46fSAndroid Build Coastguard Worker if (!params.empty() && params.size() != 1)
228*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
229*3ac0a46fSAndroid Build Coastguard Worker
230*3ac0a46fSAndroid Build Coastguard Worker bool fFlags = true;
231*3ac0a46fSAndroid Build Coastguard Worker if (params.size() == 1)
232*3ac0a46fSAndroid Build Coastguard Worker fFlags = runtime->ToBoolean(params[0]);
233*3ac0a46fSAndroid Build Coastguard Worker
234*3ac0a46fSAndroid Build Coastguard Worker int32_t iCount = GetXFANode()->GetCount();
235*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
236*3ac0a46fSAndroid Build Coastguard Worker int32_t iMax = occur ? occur->GetMax() : CXFA_Occur::kDefaultMax;
237*3ac0a46fSAndroid Build Coastguard Worker if (iMax >= 0 && iCount >= iMax)
238*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kTooManyOccurrences);
239*3ac0a46fSAndroid Build Coastguard Worker
240*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pNewInstance = GetXFANode()->CreateInstanceIfPossible(fFlags);
241*3ac0a46fSAndroid Build Coastguard Worker if (!pNewInstance)
242*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success(runtime->NewNull());
243*3ac0a46fSAndroid Build Coastguard Worker
244*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->InsertItem(pNewInstance, iCount, iCount, false);
245*3ac0a46fSAndroid Build Coastguard Worker
246*3ac0a46fSAndroid Build Coastguard Worker CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
247*3ac0a46fSAndroid Build Coastguard Worker if (pNotify) {
248*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunNodeInitialize(pNewInstance);
249*3ac0a46fSAndroid Build Coastguard Worker GetDocument()->GetLayoutProcessor()->SetHasChangedContainer();
250*3ac0a46fSAndroid Build Coastguard Worker }
251*3ac0a46fSAndroid Build Coastguard Worker
252*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success(
253*3ac0a46fSAndroid Build Coastguard Worker runtime->GetOrCreateJSBindingFromMap(pNewInstance));
254*3ac0a46fSAndroid Build Coastguard Worker }
255*3ac0a46fSAndroid Build Coastguard Worker
insertInstance(CFXJSE_Engine * runtime,const std::vector<v8::Local<v8::Value>> & params)256*3ac0a46fSAndroid Build Coastguard Worker CJS_Result CJX_InstanceManager::insertInstance(
257*3ac0a46fSAndroid Build Coastguard Worker CFXJSE_Engine* runtime,
258*3ac0a46fSAndroid Build Coastguard Worker const std::vector<v8::Local<v8::Value>>& params) {
259*3ac0a46fSAndroid Build Coastguard Worker CXFA_Document* doc = runtime->GetDocument();
260*3ac0a46fSAndroid Build Coastguard Worker if (doc->GetFormType() != FormType::kXFAFull)
261*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kNotSupportedError);
262*3ac0a46fSAndroid Build Coastguard Worker
263*3ac0a46fSAndroid Build Coastguard Worker if (params.size() != 1 && params.size() != 2)
264*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kParamError);
265*3ac0a46fSAndroid Build Coastguard Worker
266*3ac0a46fSAndroid Build Coastguard Worker int32_t iIndex = runtime->ToInt32(params[0]);
267*3ac0a46fSAndroid Build Coastguard Worker bool bBind = false;
268*3ac0a46fSAndroid Build Coastguard Worker if (params.size() == 2)
269*3ac0a46fSAndroid Build Coastguard Worker bBind = runtime->ToBoolean(params[1]);
270*3ac0a46fSAndroid Build Coastguard Worker
271*3ac0a46fSAndroid Build Coastguard Worker int32_t iCount = GetXFANode()->GetCount();
272*3ac0a46fSAndroid Build Coastguard Worker if (iIndex < 0 || iIndex > iCount)
273*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kInvalidInputError);
274*3ac0a46fSAndroid Build Coastguard Worker
275*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
276*3ac0a46fSAndroid Build Coastguard Worker int32_t iMax = occur ? occur->GetMax() : CXFA_Occur::kDefaultMax;
277*3ac0a46fSAndroid Build Coastguard Worker if (iMax >= 0 && iCount >= iMax)
278*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Failure(JSMessage::kInvalidInputError);
279*3ac0a46fSAndroid Build Coastguard Worker
280*3ac0a46fSAndroid Build Coastguard Worker CXFA_Node* pNewInstance = GetXFANode()->CreateInstanceIfPossible(bBind);
281*3ac0a46fSAndroid Build Coastguard Worker if (!pNewInstance)
282*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success(runtime->NewNull());
283*3ac0a46fSAndroid Build Coastguard Worker
284*3ac0a46fSAndroid Build Coastguard Worker GetXFANode()->InsertItem(pNewInstance, iIndex, iCount, true);
285*3ac0a46fSAndroid Build Coastguard Worker
286*3ac0a46fSAndroid Build Coastguard Worker CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
287*3ac0a46fSAndroid Build Coastguard Worker if (pNotify) {
288*3ac0a46fSAndroid Build Coastguard Worker pNotify->RunNodeInitialize(pNewInstance);
289*3ac0a46fSAndroid Build Coastguard Worker GetDocument()->GetLayoutProcessor()->SetHasChangedContainer();
290*3ac0a46fSAndroid Build Coastguard Worker }
291*3ac0a46fSAndroid Build Coastguard Worker
292*3ac0a46fSAndroid Build Coastguard Worker return CJS_Result::Success(
293*3ac0a46fSAndroid Build Coastguard Worker runtime->GetOrCreateJSBindingFromMap(pNewInstance));
294*3ac0a46fSAndroid Build Coastguard Worker }
295*3ac0a46fSAndroid Build Coastguard Worker
max(v8::Isolate * pIsolate,v8::Local<v8::Value> * pValue,bool bSetting,XFA_Attribute eAttribute)296*3ac0a46fSAndroid Build Coastguard Worker void CJX_InstanceManager::max(v8::Isolate* pIsolate,
297*3ac0a46fSAndroid Build Coastguard Worker v8::Local<v8::Value>* pValue,
298*3ac0a46fSAndroid Build Coastguard Worker bool bSetting,
299*3ac0a46fSAndroid Build Coastguard Worker XFA_Attribute eAttribute) {
300*3ac0a46fSAndroid Build Coastguard Worker if (bSetting) {
301*3ac0a46fSAndroid Build Coastguard Worker ThrowInvalidPropertyException(pIsolate);
302*3ac0a46fSAndroid Build Coastguard Worker return;
303*3ac0a46fSAndroid Build Coastguard Worker }
304*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
305*3ac0a46fSAndroid Build Coastguard Worker *pValue = fxv8::NewNumberHelper(
306*3ac0a46fSAndroid Build Coastguard Worker pIsolate, occur ? occur->GetMax() : CXFA_Occur::kDefaultMax);
307*3ac0a46fSAndroid Build Coastguard Worker }
308*3ac0a46fSAndroid Build Coastguard Worker
min(v8::Isolate * pIsolate,v8::Local<v8::Value> * pValue,bool bSetting,XFA_Attribute eAttribute)309*3ac0a46fSAndroid Build Coastguard Worker void CJX_InstanceManager::min(v8::Isolate* pIsolate,
310*3ac0a46fSAndroid Build Coastguard Worker v8::Local<v8::Value>* pValue,
311*3ac0a46fSAndroid Build Coastguard Worker bool bSetting,
312*3ac0a46fSAndroid Build Coastguard Worker XFA_Attribute eAttribute) {
313*3ac0a46fSAndroid Build Coastguard Worker if (bSetting) {
314*3ac0a46fSAndroid Build Coastguard Worker ThrowInvalidPropertyException(pIsolate);
315*3ac0a46fSAndroid Build Coastguard Worker return;
316*3ac0a46fSAndroid Build Coastguard Worker }
317*3ac0a46fSAndroid Build Coastguard Worker CXFA_Occur* occur = GetXFANode()->GetOccurIfExists();
318*3ac0a46fSAndroid Build Coastguard Worker *pValue = fxv8::NewNumberHelper(
319*3ac0a46fSAndroid Build Coastguard Worker pIsolate, occur ? occur->GetMin() : CXFA_Occur::kDefaultMin);
320*3ac0a46fSAndroid Build Coastguard Worker }
321*3ac0a46fSAndroid Build Coastguard Worker
count(v8::Isolate * pIsolate,v8::Local<v8::Value> * pValue,bool bSetting,XFA_Attribute eAttribute)322*3ac0a46fSAndroid Build Coastguard Worker void CJX_InstanceManager::count(v8::Isolate* pIsolate,
323*3ac0a46fSAndroid Build Coastguard Worker v8::Local<v8::Value>* pValue,
324*3ac0a46fSAndroid Build Coastguard Worker bool bSetting,
325*3ac0a46fSAndroid Build Coastguard Worker XFA_Attribute eAttribute) {
326*3ac0a46fSAndroid Build Coastguard Worker if (bSetting) {
327*3ac0a46fSAndroid Build Coastguard Worker SetInstances(pIsolate, fxv8::ReentrantToInt32Helper(pIsolate, *pValue));
328*3ac0a46fSAndroid Build Coastguard Worker return;
329*3ac0a46fSAndroid Build Coastguard Worker }
330*3ac0a46fSAndroid Build Coastguard Worker *pValue = fxv8::NewNumberHelper(pIsolate, GetXFANode()->GetCount());
331*3ac0a46fSAndroid Build Coastguard Worker }
332