| // Copyright 2014 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "fxjs/xfa/cfxjse_engine.h" |
| |
| #include <utility> |
| |
| #include "core/fxcrt/autorestorer.h" |
| #include "core/fxcrt/cfx_widetextbuf.h" |
| #include "core/fxcrt/fx_extension.h" |
| #include "fxjs/cjs_runtime.h" |
| #include "fxjs/xfa/cfxjse_class.h" |
| #include "fxjs/xfa/cfxjse_context.h" |
| #include "fxjs/xfa/cfxjse_formcalc_context.h" |
| #include "fxjs/xfa/cfxjse_resolveprocessor.h" |
| #include "fxjs/xfa/cfxjse_value.h" |
| #include "fxjs/xfa/cjx_object.h" |
| #include "third_party/base/ptr_util.h" |
| #include "third_party/base/stl_util.h" |
| #include "xfa/fxfa/cxfa_eventparam.h" |
| #include "xfa/fxfa/cxfa_ffdoc.h" |
| #include "xfa/fxfa/cxfa_ffnotify.h" |
| #include "xfa/fxfa/parser/cxfa_document.h" |
| #include "xfa/fxfa/parser/cxfa_localemgr.h" |
| #include "xfa/fxfa/parser/cxfa_node.h" |
| #include "xfa/fxfa/parser/cxfa_nodehelper.h" |
| #include "xfa/fxfa/parser/cxfa_object.h" |
| #include "xfa/fxfa/parser/cxfa_thisproxy.h" |
| #include "xfa/fxfa/parser/cxfa_treelist.h" |
| #include "xfa/fxfa/parser/xfa_basic_data.h" |
| #include "xfa/fxfa/parser/xfa_resolvenode_rs.h" |
| #include "xfa/fxfa/parser/xfa_utils.h" |
| |
| using pdfium::fxjse::kClassTag; |
| |
| const FXJSE_CLASS_DESCRIPTOR GlobalClassDescriptor = { |
| kClassTag, // tag |
| "Root", // name |
| nullptr, // methods |
| 0, // method count |
| CFXJSE_Engine::GlobalPropTypeGetter, |
| CFXJSE_Engine::GlobalPropertyGetter, |
| CFXJSE_Engine::GlobalPropertySetter, |
| CFXJSE_Engine::NormalMethodCall, |
| }; |
| |
| const FXJSE_CLASS_DESCRIPTOR NormalClassDescriptor = { |
| kClassTag, // tag |
| "XFAObject", // name |
| nullptr, // methods |
| 0, // method count |
| CFXJSE_Engine::NormalPropTypeGetter, |
| CFXJSE_Engine::NormalPropertyGetter, |
| CFXJSE_Engine::NormalPropertySetter, |
| CFXJSE_Engine::NormalMethodCall, |
| }; |
| |
| const FXJSE_CLASS_DESCRIPTOR VariablesClassDescriptor = { |
| kClassTag, // tag |
| "XFAScriptObject", // name |
| nullptr, // methods |
| 0, // method count |
| CFXJSE_Engine::NormalPropTypeGetter, |
| CFXJSE_Engine::GlobalPropertyGetter, |
| CFXJSE_Engine::GlobalPropertySetter, |
| CFXJSE_Engine::NormalMethodCall, |
| }; |
| |
| namespace { |
| |
| const char kFormCalcRuntime[] = "pfm_rt"; |
| |
| CXFA_ThisProxy* ToThisProxy(CFXJSE_Value* pValue) { |
| CFXJSE_HostObject* pHostObject = pValue->ToHostObject(); |
| return pHostObject ? ToThisProxy(pHostObject->AsCXFAObject()) : nullptr; |
| } |
| |
| } // namespace |
| |
| // static |
| CXFA_Object* CFXJSE_Engine::ToObject( |
| const v8::FunctionCallbackInfo<v8::Value>& info) { |
| return ToObject(info.Holder()); |
| } |
| |
| // static |
| CXFA_Object* CFXJSE_Engine::ToObject(v8::Local<v8::Value> value) { |
| if (!value->IsObject()) |
| return nullptr; |
| |
| return ToObject(FXJSE_RetrieveObjectBinding(value.As<v8::Object>())); |
| } |
| |
| // static. |
| CXFA_Object* CFXJSE_Engine::ToObject(CFXJSE_Value* pValue) { |
| return ToObject(pValue->ToHostObject()); |
| } |
| |
| // static |
| CXFA_Object* CFXJSE_Engine::ToObject(CFXJSE_HostObject* pHostObj) { |
| return pHostObj ? pHostObj->AsCXFAObject() : nullptr; |
| } |
| |
| CFXJSE_Engine::CFXJSE_Engine(CXFA_Document* pDocument, |
| CJS_Runtime* fxjs_runtime) |
| : CFX_V8(fxjs_runtime->GetIsolate()), |
| m_pSubordinateRuntime(fxjs_runtime), |
| m_pDocument(pDocument), |
| m_JsContext(CFXJSE_Context::Create(fxjs_runtime->GetIsolate(), |
| &GlobalClassDescriptor, |
| pDocument->GetRoot())), |
| m_ResolveProcessor(pdfium::MakeUnique<CFXJSE_ResolveProcessor>()) { |
| RemoveBuiltInObjs(m_JsContext.get()); |
| m_JsContext->EnableCompatibleMode(); |
| |
| // Don't know if this can happen before we remove the builtin objs and set |
| // compatibility mode. |
| m_pJsClass = |
| CFXJSE_Class::Create(m_JsContext.get(), &NormalClassDescriptor, false); |
| } |
| |
| CFXJSE_Engine::~CFXJSE_Engine() { |
| for (const auto& pair : m_mapVariableToContext) |
| delete ToThisProxy(pair.second->GetGlobalObject().get()); |
| |
| for (const auto& pair : m_mapObjectToValue) |
| pair.second->ClearHostObject(); |
| } |
| |
| bool CFXJSE_Engine::RunScript(CXFA_Script::Type eScriptType, |
| WideStringView wsScript, |
| CFXJSE_Value* hRetValue, |
| CXFA_Object* pThisObject) { |
| ByteString btScript; |
| AutoRestorer<CXFA_Script::Type> typeRestorer(&m_eScriptType); |
| m_eScriptType = eScriptType; |
| if (eScriptType == CXFA_Script::Type::Formcalc) { |
| if (!m_FM2JSContext) { |
| m_FM2JSContext = pdfium::MakeUnique<CFXJSE_FormCalcContext>( |
| GetIsolate(), m_JsContext.get(), m_pDocument.Get()); |
| } |
| CFX_WideTextBuf wsJavaScript; |
| if (!CFXJSE_FormCalcContext::Translate(wsScript, &wsJavaScript)) { |
| hRetValue->SetUndefined(); |
| return false; |
| } |
| btScript = FX_UTF8Encode(wsJavaScript.AsStringView()); |
| } else { |
| btScript = FX_UTF8Encode(wsScript); |
| } |
| AutoRestorer<UnownedPtr<CXFA_Object>> nodeRestorer(&m_pThisObject); |
| m_pThisObject = pThisObject; |
| |
| CFXJSE_Value* pValue = |
| pThisObject ? GetOrCreateJSBindingFromMap(pThisObject) : nullptr; |
| IJS_Runtime::ScopedEventContext ctx(m_pSubordinateRuntime.Get()); |
| return m_JsContext->ExecuteScript(btScript.c_str(), hRetValue, pValue); |
| } |
| |
| bool CFXJSE_Engine::QueryNodeByFlag(CXFA_Node* refNode, |
| WideStringView propname, |
| CFXJSE_Value* pValue, |
| uint32_t dwFlag, |
| bool bSetting) { |
| if (!refNode) |
| return false; |
| |
| XFA_RESOLVENODE_RS resolveRs; |
| if (!ResolveObjects(refNode, propname, &resolveRs, dwFlag, nullptr)) |
| return false; |
| if (resolveRs.dwFlags == XFA_ResolveNode_RSType_Nodes) { |
| pValue->Assign( |
| GetOrCreateJSBindingFromMap(resolveRs.objects.front().Get())); |
| return true; |
| } |
| if (resolveRs.dwFlags == XFA_ResolveNode_RSType_Attribute && |
| resolveRs.script_attribute.callback) { |
| CJX_Object* jsObject = resolveRs.objects.front()->JSObject(); |
| (*resolveRs.script_attribute.callback)( |
| jsObject, pValue, bSetting, resolveRs.script_attribute.attribute); |
| } |
| return true; |
| } |
| |
| // static |
| void CFXJSE_Engine::GlobalPropertySetter(CFXJSE_Value* pObject, |
| ByteStringView szPropName, |
| CFXJSE_Value* pValue) { |
| CXFA_Object* lpOrginalNode = ToObject(pObject); |
| CXFA_Document* pDoc = lpOrginalNode->GetDocument(); |
| CFXJSE_Engine* lpScriptContext = pDoc->GetScriptContext(); |
| CXFA_Node* pRefNode = ToNode(lpScriptContext->GetThisObject()); |
| if (lpOrginalNode->IsThisProxy()) |
| pRefNode = ToNode(lpScriptContext->GetVariablesThis(lpOrginalNode, false)); |
| |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| if (lpScriptContext->QueryNodeByFlag( |
| pRefNode, wsPropName.AsStringView(), pValue, |
| XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings | |
| XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Properties | |
| XFA_RESOLVENODE_Attributes, |
| true)) { |
| return; |
| } |
| if (lpOrginalNode->IsThisProxy() && pValue && pValue->IsUndefined()) { |
| pObject->DeleteObjectProperty(szPropName); |
| return; |
| } |
| CXFA_FFNotify* pNotify = pDoc->GetNotify(); |
| if (!pNotify) |
| return; |
| |
| CXFA_FFDoc* hDoc = pNotify->GetHDOC(); |
| auto* pCJSRuntime = |
| static_cast<CJS_Runtime*>(hDoc->GetDocEnvironment()->GetIJSRuntime(hDoc)); |
| if (!pCJSRuntime) |
| return; |
| |
| v8::HandleScope handle_scope(lpScriptContext->GetIsolate()); |
| IJS_Runtime::ScopedEventContext pContext(pCJSRuntime); |
| pCJSRuntime->SetValueByNameInGlobalObject( |
| szPropName, v8::Local<v8::Value>::New(lpScriptContext->GetIsolate(), |
| pValue->DirectGetValue())); |
| } |
| |
| // static |
| void CFXJSE_Engine::GlobalPropertyGetter(CFXJSE_Value* pObject, |
| ByteStringView szPropName, |
| CFXJSE_Value* pValue) { |
| CXFA_Object* pOriginalObject = ToObject(pObject); |
| CXFA_Document* pDoc = pOriginalObject->GetDocument(); |
| CFXJSE_Engine* lpScriptContext = pDoc->GetScriptContext(); |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| |
| pValue->SetUndefined(); // Assume failure. |
| if (lpScriptContext->GetType() == CXFA_Script::Type::Formcalc) { |
| if (szPropName == kFormCalcRuntime) { |
| lpScriptContext->m_FM2JSContext->GlobalPropertyGetter(pValue); |
| return; |
| } |
| XFA_HashCode uHashCode = static_cast<XFA_HashCode>( |
| FX_HashCode_GetW(wsPropName.AsStringView(), false)); |
| if (uHashCode != XFA_HASHCODE_Layout) { |
| CXFA_Object* pObj = |
| lpScriptContext->GetDocument()->GetXFAObject(uHashCode); |
| if (pObj) { |
| pValue->Assign(lpScriptContext->GetOrCreateJSBindingFromMap(pObj)); |
| return; |
| } |
| } |
| } |
| |
| CXFA_Node* pRefNode = ToNode(lpScriptContext->GetThisObject()); |
| if (pOriginalObject->IsThisProxy()) { |
| pRefNode = |
| ToNode(lpScriptContext->GetVariablesThis(pOriginalObject, false)); |
| } |
| if (lpScriptContext->QueryNodeByFlag( |
| pRefNode, wsPropName.AsStringView(), pValue, |
| XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Properties | |
| XFA_RESOLVENODE_Attributes, |
| false)) { |
| return; |
| } |
| if (lpScriptContext->QueryNodeByFlag( |
| pRefNode, wsPropName.AsStringView(), pValue, |
| XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings, false)) { |
| return; |
| } |
| |
| CXFA_Object* pScriptObject = |
| lpScriptContext->GetVariablesThis(pOriginalObject, true); |
| if (pScriptObject && lpScriptContext->QueryVariableValue( |
| pScriptObject->AsNode(), szPropName, pValue, true)) { |
| return; |
| } |
| |
| CXFA_FFNotify* pNotify = pDoc->GetNotify(); |
| if (!pNotify) |
| return; |
| |
| CXFA_FFDoc* hDoc = pNotify->GetHDOC(); |
| auto* pCJSRuntime = |
| static_cast<CJS_Runtime*>(hDoc->GetDocEnvironment()->GetIJSRuntime(hDoc)); |
| if (!pCJSRuntime) |
| return; |
| |
| v8::HandleScope handle_scope(lpScriptContext->GetIsolate()); |
| IJS_Runtime::ScopedEventContext pContext(pCJSRuntime); |
| v8::Local<v8::Value> temp_value; |
| if (!pCJSRuntime->GetValueByNameFromGlobalObject(szPropName, &temp_value)) |
| return; |
| |
| if (temp_value.IsEmpty()) |
| return; |
| |
| pValue->ForceSetValue(temp_value); |
| } |
| |
| int32_t CFXJSE_Engine::GlobalPropTypeGetter(CFXJSE_Value* pOriginalValue, |
| ByteStringView szPropName, |
| bool bQueryIn) { |
| CXFA_Object* pObject = ToObject(pOriginalValue); |
| if (!pObject) |
| return FXJSE_ClassPropType_None; |
| |
| CFXJSE_Engine* lpScriptContext = pObject->GetDocument()->GetScriptContext(); |
| pObject = lpScriptContext->GetVariablesThis(pObject, false); |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| if (pObject->JSObject()->HasMethod(wsPropName)) |
| return FXJSE_ClassPropType_Method; |
| |
| return FXJSE_ClassPropType_Property; |
| } |
| |
| // static |
| void CFXJSE_Engine::NormalPropertyGetter(CFXJSE_Value* pOriginalValue, |
| ByteStringView szPropName, |
| CFXJSE_Value* pReturnValue) { |
| pReturnValue->SetUndefined(); // Assume failure. |
| CXFA_Object* pOriginalObject = ToObject(pOriginalValue); |
| if (!pOriginalObject) |
| return; |
| |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| CFXJSE_Engine* lpScriptContext = |
| pOriginalObject->GetDocument()->GetScriptContext(); |
| CXFA_Object* pObject = |
| lpScriptContext->GetVariablesThis(pOriginalObject, false); |
| if (wsPropName.EqualsASCII("xfa")) { |
| CFXJSE_Value* pValue = lpScriptContext->GetOrCreateJSBindingFromMap( |
| lpScriptContext->GetDocument()->GetRoot()); |
| pReturnValue->Assign(pValue); |
| return; |
| } |
| |
| bool bRet = lpScriptContext->QueryNodeByFlag( |
| ToNode(pObject), wsPropName.AsStringView(), pReturnValue, |
| XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Properties | |
| XFA_RESOLVENODE_Attributes, |
| false); |
| if (bRet) |
| return; |
| |
| if (pObject == lpScriptContext->GetThisObject() || |
| (lpScriptContext->GetType() == CXFA_Script::Type::Javascript && |
| !lpScriptContext->IsStrictScopeInJavaScript())) { |
| bRet = lpScriptContext->QueryNodeByFlag( |
| ToNode(pObject), wsPropName.AsStringView(), pReturnValue, |
| XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings, false); |
| } |
| if (bRet) |
| return; |
| |
| CXFA_Object* pScriptObject = |
| lpScriptContext->GetVariablesThis(pOriginalObject, true); |
| if (!pScriptObject) |
| return; |
| |
| bRet = lpScriptContext->QueryVariableValue(ToNode(pScriptObject), szPropName, |
| pReturnValue, true); |
| if (bRet) |
| return; |
| |
| Optional<XFA_SCRIPTATTRIBUTEINFO> info = XFA_GetScriptAttributeByName( |
| pObject->GetElementType(), wsPropName.AsStringView()); |
| if (info.has_value()) { |
| CJX_Object* jsObject = pObject->JSObject(); |
| (*info.value().callback)(jsObject, pReturnValue, false, |
| info.value().attribute); |
| return; |
| } |
| |
| CXFA_FFNotify* pNotify = pObject->GetDocument()->GetNotify(); |
| if (!pNotify) |
| return; |
| |
| CXFA_FFDoc* hDoc = pNotify->GetHDOC(); |
| auto* pCJSRuntime = |
| static_cast<CJS_Runtime*>(hDoc->GetDocEnvironment()->GetIJSRuntime(hDoc)); |
| if (!pCJSRuntime) |
| return; |
| |
| v8::HandleScope handle_scope(lpScriptContext->GetIsolate()); |
| IJS_Runtime::ScopedEventContext pContext(pCJSRuntime); |
| v8::Local<v8::Value> temp_local; |
| if (!pCJSRuntime->GetValueByNameFromGlobalObject(szPropName, &temp_local)) |
| return; |
| |
| if (temp_local.IsEmpty()) |
| return; |
| |
| pReturnValue->ForceSetValue(temp_local); |
| } |
| |
| // static |
| void CFXJSE_Engine::NormalPropertySetter(CFXJSE_Value* pOriginalValue, |
| ByteStringView szPropName, |
| CFXJSE_Value* pReturnValue) { |
| CXFA_Object* pOriginalObject = ToObject(pOriginalValue); |
| if (!pOriginalObject) |
| return; |
| |
| CFXJSE_Engine* lpScriptContext = |
| pOriginalObject->GetDocument()->GetScriptContext(); |
| CXFA_Object* pObject = |
| lpScriptContext->GetVariablesThis(pOriginalObject, false); |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| WideStringView wsPropNameView = wsPropName.AsStringView(); |
| Optional<XFA_SCRIPTATTRIBUTEINFO> info = |
| XFA_GetScriptAttributeByName(pObject->GetElementType(), wsPropNameView); |
| if (info.has_value()) { |
| CJX_Object* jsObject = pObject->JSObject(); |
| (*info.value().callback)(jsObject, pReturnValue, true, |
| info.value().attribute); |
| return; |
| } |
| |
| if (pObject->IsNode()) { |
| if (wsPropNameView[0] == '#') |
| wsPropNameView = wsPropNameView.Last(wsPropNameView.GetLength() - 1); |
| |
| CXFA_Node* pNode = ToNode(pObject); |
| CXFA_Node* pPropOrChild = nullptr; |
| XFA_Element eType = XFA_GetElementByName(wsPropNameView); |
| if (eType != XFA_Element::Unknown) { |
| pPropOrChild = |
| pNode->JSObject()->GetOrCreateProperty<CXFA_Node>(0, eType); |
| } else { |
| pPropOrChild = pNode->GetFirstChildByName(wsPropNameView); |
| } |
| |
| if (pPropOrChild) { |
| info = XFA_GetScriptAttributeByName(pPropOrChild->GetElementType(), |
| L"{default}"); |
| if (info.has_value()) { |
| pPropOrChild->JSObject()->ScriptSomDefaultValue(pReturnValue, true, |
| XFA_Attribute::Unknown); |
| return; |
| } |
| } |
| } |
| |
| CXFA_Object* pScriptObject = |
| lpScriptContext->GetVariablesThis(pOriginalObject, true); |
| if (pScriptObject) { |
| lpScriptContext->QueryVariableValue(ToNode(pScriptObject), szPropName, |
| pReturnValue, false); |
| } |
| } |
| |
| int32_t CFXJSE_Engine::NormalPropTypeGetter(CFXJSE_Value* pOriginalValue, |
| ByteStringView szPropName, |
| bool bQueryIn) { |
| CXFA_Object* pObject = ToObject(pOriginalValue); |
| if (!pObject) |
| return FXJSE_ClassPropType_None; |
| |
| CFXJSE_Engine* lpScriptContext = pObject->GetDocument()->GetScriptContext(); |
| pObject = lpScriptContext->GetVariablesThis(pObject, false); |
| XFA_Element eType = pObject->GetElementType(); |
| WideString wsPropName = WideString::FromUTF8(szPropName); |
| if (pObject->JSObject()->HasMethod(wsPropName)) |
| return FXJSE_ClassPropType_Method; |
| |
| if (bQueryIn && |
| !XFA_GetScriptAttributeByName(eType, wsPropName.AsStringView())) { |
| return FXJSE_ClassPropType_None; |
| } |
| return FXJSE_ClassPropType_Property; |
| } |
| |
| CJS_Result CFXJSE_Engine::NormalMethodCall( |
| const v8::FunctionCallbackInfo<v8::Value>& info, |
| const WideString& functionName) { |
| CXFA_Object* pObject = ToObject(info); |
| if (!pObject) |
| return CJS_Result::Failure(L"no Holder() present."); |
| |
| CFXJSE_Engine* lpScriptContext = pObject->GetDocument()->GetScriptContext(); |
| pObject = lpScriptContext->GetVariablesThis(pObject, false); |
| |
| std::vector<v8::Local<v8::Value>> parameters; |
| for (int i = 0; i < info.Length(); i++) |
| parameters.push_back(info[i]); |
| |
| return pObject->JSObject()->RunMethod(functionName, parameters); |
| } |
| |
| bool CFXJSE_Engine::IsStrictScopeInJavaScript() { |
| return m_pDocument->is_strict_scoping(); |
| } |
| |
| CXFA_Script::Type CFXJSE_Engine::GetType() { |
| return m_eScriptType; |
| } |
| |
| CFXJSE_Context* CFXJSE_Engine::CreateVariablesContext(CXFA_Node* pScriptNode, |
| CXFA_Node* pSubform) { |
| if (!pScriptNode || !pSubform) |
| return nullptr; |
| |
| auto pNewContext = |
| CFXJSE_Context::Create(GetIsolate(), &VariablesClassDescriptor, |
| new CXFA_ThisProxy(pSubform, pScriptNode)); |
| RemoveBuiltInObjs(pNewContext.get()); |
| pNewContext->EnableCompatibleMode(); |
| CFXJSE_Context* pResult = pNewContext.get(); |
| m_mapVariableToContext[pScriptNode] = std::move(pNewContext); |
| return pResult; |
| } |
| |
| CXFA_Object* CFXJSE_Engine::GetVariablesThis(CXFA_Object* pObject, |
| bool bScriptNode) { |
| CXFA_ThisProxy* pProxy = ToThisProxy(pObject); |
| if (!pProxy) |
| return pObject; |
| |
| return bScriptNode ? pProxy->GetScriptNode() : pProxy->GetThisNode(); |
| } |
| |
| bool CFXJSE_Engine::RunVariablesScript(CXFA_Node* pScriptNode) { |
| if (!pScriptNode) |
| return false; |
| |
| if (pScriptNode->GetElementType() != XFA_Element::Script) |
| return true; |
| |
| CXFA_Node* pParent = pScriptNode->GetParent(); |
| if (!pParent || pParent->GetElementType() != XFA_Element::Variables) |
| return false; |
| |
| auto it = m_mapVariableToContext.find(pScriptNode); |
| if (it != m_mapVariableToContext.end() && it->second) |
| return true; |
| |
| CXFA_Node* pTextNode = pScriptNode->GetFirstChild(); |
| if (!pTextNode) |
| return false; |
| |
| Optional<WideString> wsScript = |
| pTextNode->JSObject()->TryCData(XFA_Attribute::Value, true); |
| if (!wsScript) |
| return false; |
| |
| ByteString btScript = wsScript->ToUTF8(); |
| auto hRetValue = pdfium::MakeUnique<CFXJSE_Value>(GetIsolate()); |
| CXFA_Node* pThisObject = pParent->GetParent(); |
| CFXJSE_Context* pVariablesContext = |
| CreateVariablesContext(pScriptNode, pThisObject); |
| AutoRestorer<UnownedPtr<CXFA_Object>> nodeRestorer(&m_pThisObject); |
| m_pThisObject = pThisObject; |
| return pVariablesContext->ExecuteScript(btScript.c_str(), hRetValue.get(), |
| nullptr); |
| } |
| |
| bool CFXJSE_Engine::QueryVariableValue(CXFA_Node* pScriptNode, |
| ByteStringView szPropName, |
| CFXJSE_Value* pValue, |
| bool bGetter) { |
| if (!pScriptNode || pScriptNode->GetElementType() != XFA_Element::Script) |
| return false; |
| |
| CXFA_Node* variablesNode = pScriptNode->GetParent(); |
| if (!variablesNode || |
| variablesNode->GetElementType() != XFA_Element::Variables) |
| return false; |
| |
| auto it = m_mapVariableToContext.find(pScriptNode); |
| if (it == m_mapVariableToContext.end() || !it->second) |
| return false; |
| |
| CFXJSE_Context* pVariableContext = it->second.get(); |
| std::unique_ptr<CFXJSE_Value> pObject = pVariableContext->GetGlobalObject(); |
| auto hVariableValue = pdfium::MakeUnique<CFXJSE_Value>(GetIsolate()); |
| if (!bGetter) { |
| pObject->SetObjectOwnProperty(szPropName, pValue); |
| return true; |
| } |
| |
| if (!pObject->HasObjectOwnProperty(szPropName, false)) |
| return false; |
| |
| pObject->GetObjectProperty(szPropName, hVariableValue.get()); |
| if (hVariableValue->IsFunction()) |
| pValue->SetFunctionBind(hVariableValue.get(), pObject.get()); |
| else if (bGetter) |
| pValue->Assign(hVariableValue.get()); |
| else |
| hVariableValue.get()->Assign(pValue); |
| return true; |
| } |
| |
| void CFXJSE_Engine::RemoveBuiltInObjs(CFXJSE_Context* pContext) const { |
| const ByteStringView kObjNames[2] = {"Number", "Date"}; |
| std::unique_ptr<CFXJSE_Value> pObject = pContext->GetGlobalObject(); |
| auto hProp = pdfium::MakeUnique<CFXJSE_Value>(GetIsolate()); |
| for (const auto& obj : kObjNames) { |
| if (pObject->GetObjectProperty(obj, hProp.get())) |
| pObject->DeleteObjectProperty(obj); |
| } |
| } |
| |
| bool CFXJSE_Engine::ResolveObjects(CXFA_Object* refObject, |
| WideStringView wsExpression, |
| XFA_RESOLVENODE_RS* resolveNodeRS, |
| uint32_t dwStyles, |
| CXFA_Node* bindNode) { |
| if (wsExpression.IsEmpty()) |
| return false; |
| |
| if (m_eScriptType != CXFA_Script::Type::Formcalc || |
| (dwStyles & (XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings))) { |
| m_upObjectArray.clear(); |
| } |
| if (refObject && refObject->IsNode() && |
| (dwStyles & (XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings))) { |
| m_upObjectArray.push_back(refObject->AsNode()); |
| } |
| |
| bool bNextCreate = false; |
| CXFA_NodeHelper* pNodeHelper = m_ResolveProcessor->GetNodeHelper(); |
| if (dwStyles & XFA_RESOLVENODE_CreateNode) |
| pNodeHelper->SetCreateNodeType(bindNode); |
| |
| pNodeHelper->m_pCreateParent = nullptr; |
| pNodeHelper->m_iCurAllStart = -1; |
| |
| CFXJSE_ResolveNodeData rndFind(this); |
| int32_t nStart = 0; |
| int32_t nLevel = 0; |
| |
| std::vector<UnownedPtr<CXFA_Object>> findObjects; |
| findObjects.emplace_back(refObject ? refObject : m_pDocument->GetRoot()); |
| int32_t nNodes = 0; |
| while (true) { |
| nNodes = pdfium::CollectionSize<int32_t>(findObjects); |
| int32_t i = 0; |
| rndFind.m_dwStyles = dwStyles; |
| m_ResolveProcessor->SetCurStart(nStart); |
| nStart = m_ResolveProcessor->GetFilter(wsExpression, nStart, rndFind); |
| if (nStart < 1) { |
| if ((dwStyles & XFA_RESOLVENODE_CreateNode) && !bNextCreate) { |
| CXFA_Node* pDataNode = nullptr; |
| nStart = pNodeHelper->m_iCurAllStart; |
| if (nStart != -1) { |
| pDataNode = m_pDocument->GetNotBindNode(findObjects); |
| if (pDataNode) { |
| findObjects.clear(); |
| findObjects.emplace_back(pDataNode); |
| break; |
| } |
| } else { |
| pDataNode = findObjects.front()->AsNode(); |
| findObjects.clear(); |
| findObjects.emplace_back(pDataNode); |
| break; |
| } |
| dwStyles |= XFA_RESOLVENODE_Bind; |
| findObjects.clear(); |
| findObjects.emplace_back(pNodeHelper->m_pAllStartParent.Get()); |
| continue; |
| } |
| break; |
| } |
| if (bNextCreate) { |
| int32_t checked_length = |
| pdfium::base::checked_cast<int32_t>(wsExpression.GetLength()); |
| if (pNodeHelper->CreateNode(rndFind.m_wsName, rndFind.m_wsCondition, |
| nStart == checked_length, this)) { |
| continue; |
| } |
| break; |
| } |
| std::vector<UnownedPtr<CXFA_Object>> retObjects; |
| while (i < nNodes) { |
| bool bDataBind = false; |
| if (((dwStyles & XFA_RESOLVENODE_Bind) || |
| (dwStyles & XFA_RESOLVENODE_CreateNode)) && |
| nNodes > 1) { |
| CFXJSE_ResolveNodeData rndBind(nullptr); |
| m_ResolveProcessor->GetFilter(wsExpression, nStart, rndBind); |
| m_ResolveProcessor->SetIndexDataBind(rndBind.m_wsCondition, i, nNodes); |
| bDataBind = true; |
| } |
| rndFind.m_CurObject = findObjects[i++].Get(); |
| rndFind.m_nLevel = nLevel; |
| rndFind.m_dwFlag = XFA_ResolveNode_RSType_Nodes; |
| if (!m_ResolveProcessor->Resolve(rndFind)) |
| continue; |
| |
| if (rndFind.m_dwFlag == XFA_ResolveNode_RSType_Attribute && |
| rndFind.m_ScriptAttribute.callback && |
| nStart < |
| pdfium::base::checked_cast<int32_t>(wsExpression.GetLength())) { |
| auto pValue = pdfium::MakeUnique<CFXJSE_Value>(GetIsolate()); |
| CJX_Object* jsObject = rndFind.m_Objects.front()->JSObject(); |
| (*rndFind.m_ScriptAttribute.callback)( |
| jsObject, pValue.get(), false, rndFind.m_ScriptAttribute.attribute); |
| if (!pValue->IsEmpty()) |
| rndFind.m_Objects.front() = ToObject(pValue.get()); |
| } |
| if (!m_upObjectArray.empty()) |
| m_upObjectArray.pop_back(); |
| retObjects.insert(retObjects.end(), rndFind.m_Objects.begin(), |
| rndFind.m_Objects.end()); |
| rndFind.m_Objects.clear(); |
| if (bDataBind) |
| break; |
| } |
| findObjects.clear(); |
| |
| nNodes = pdfium::CollectionSize<int32_t>(retObjects); |
| if (nNodes < 1) { |
| if (dwStyles & XFA_RESOLVENODE_CreateNode) { |
| bNextCreate = true; |
| if (!pNodeHelper->m_pCreateParent) { |
| pNodeHelper->m_pCreateParent = ToNode(rndFind.m_CurObject.Get()); |
| pNodeHelper->m_iCreateCount = 1; |
| } |
| int32_t checked_length = |
| pdfium::base::checked_cast<int32_t>(wsExpression.GetLength()); |
| if (pNodeHelper->CreateNode(rndFind.m_wsName, rndFind.m_wsCondition, |
| nStart == checked_length, this)) { |
| continue; |
| } |
| } |
| break; |
| } |
| |
| findObjects = std::move(retObjects); |
| rndFind.m_Objects.clear(); |
| if (nLevel == 0) |
| dwStyles &= ~(XFA_RESOLVENODE_Parent | XFA_RESOLVENODE_Siblings); |
| |
| nLevel++; |
| } |
| |
| if (!bNextCreate) { |
| resolveNodeRS->dwFlags = rndFind.m_dwFlag; |
| if (nNodes > 0) { |
| resolveNodeRS->objects.insert(resolveNodeRS->objects.end(), |
| findObjects.begin(), findObjects.end()); |
| } |
| if (rndFind.m_dwFlag == XFA_ResolveNode_RSType_Attribute) { |
| resolveNodeRS->script_attribute = rndFind.m_ScriptAttribute; |
| return true; |
| } |
| } |
| if (dwStyles & (XFA_RESOLVENODE_CreateNode | XFA_RESOLVENODE_Bind | |
| XFA_RESOLVENODE_BindNew)) { |
| if (pNodeHelper->m_pCreateParent) |
| resolveNodeRS->objects.emplace_back(pNodeHelper->m_pCreateParent.Get()); |
| else |
| pNodeHelper->CreateNodeForCondition(rndFind.m_wsCondition); |
| |
| resolveNodeRS->dwFlags = pNodeHelper->m_iCreateFlag; |
| if (resolveNodeRS->dwFlags == XFA_ResolveNode_RSType_CreateNodeOne) { |
| if (pNodeHelper->m_iCurAllStart != -1) |
| resolveNodeRS->dwFlags = XFA_ResolveNode_RSType_CreateNodeMidAll; |
| } |
| |
| if (!bNextCreate && (dwStyles & XFA_RESOLVENODE_CreateNode)) |
| resolveNodeRS->dwFlags = XFA_ResolveNode_RSType_ExistNodes; |
| |
| return !resolveNodeRS->objects.empty(); |
| } |
| return nNodes > 0; |
| } |
| |
| void CFXJSE_Engine::AddToCacheList(std::unique_ptr<CXFA_List> pList) { |
| m_CacheList.push_back(std::move(pList)); |
| } |
| |
| CFXJSE_Value* CFXJSE_Engine::GetOrCreateJSBindingFromMap(CXFA_Object* pObject) { |
| if (pObject->IsNode()) |
| RunVariablesScript(pObject->AsNode()); |
| |
| auto iter = m_mapObjectToValue.find(pObject); |
| if (iter != m_mapObjectToValue.end()) |
| return iter->second.get(); |
| |
| auto jsValue = pdfium::MakeUnique<CFXJSE_Value>(GetIsolate()); |
| jsValue->SetHostObject(pObject, m_pJsClass.Get()); |
| |
| CFXJSE_Value* pValue = jsValue.get(); |
| m_mapObjectToValue.insert(std::make_pair(pObject, std::move(jsValue))); |
| return pValue; |
| } |
| |
| void CFXJSE_Engine::RemoveJSBindingFromMap(CXFA_Object* pObject) { |
| auto iter = m_mapObjectToValue.find(pObject); |
| if (iter == m_mapObjectToValue.end()) |
| return; |
| |
| iter->second->ClearHostObject(); |
| m_mapObjectToValue.erase(iter); |
| } |
| |
| void CFXJSE_Engine::SetNodesOfRunScript(std::vector<CXFA_Node*>* pArray) { |
| m_pScriptNodeArray = pArray; |
| } |
| |
| void CFXJSE_Engine::AddNodesOfRunScript(CXFA_Node* pNode) { |
| if (m_pScriptNodeArray && !pdfium::ContainsValue(*m_pScriptNodeArray, pNode)) |
| m_pScriptNodeArray->push_back(pNode); |
| } |
| |
| CXFA_Object* CFXJSE_Engine::ToXFAObject(v8::Local<v8::Value> obj) { |
| if (obj.IsEmpty() || !obj->IsObject()) |
| return nullptr; |
| |
| CFXJSE_HostObject* pHostObj = |
| FXJSE_RetrieveObjectBinding(obj.As<v8::Object>()); |
| return pHostObj ? pHostObj->AsCXFAObject() : nullptr; |
| } |
| |
| v8::Local<v8::Value> CFXJSE_Engine::NewXFAObject( |
| CXFA_Object* obj, |
| v8::Global<v8::FunctionTemplate>& tmpl) { |
| v8::EscapableHandleScope scope(GetIsolate()); |
| v8::Local<v8::FunctionTemplate> klass = |
| v8::Local<v8::FunctionTemplate>::New(GetIsolate(), tmpl); |
| v8::Local<v8::Object> object = klass->InstanceTemplate() |
| ->NewInstance(m_JsContext->GetContext()) |
| .ToLocalChecked(); |
| FXJSE_UpdateObjectBinding(object, obj); |
| return scope.Escape(object); |
| } |