// Copyright 2017 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/cjx_hostpseudomodel.h"
#include <memory>
#include <vector>
#include "fxjs/cfxjse_engine.h"
#include "fxjs/cfxjse_value.h"
#include "fxjs/js_resources.h"
#include "xfa/fxfa/cxfa_ffnotify.h"
#include "xfa/fxfa/parser/cscript_hostpseudomodel.h"
#include "xfa/fxfa/parser/cxfa_layoutprocessor.h"
#include "xfa/fxfa/parser/cxfa_node.h"
#include "xfa/fxfa/parser/xfa_resolvenode_rs.h"
namespace {
int32_t FilterName(const WideStringView& wsExpression,
int32_t nStart,
WideString& wsFilter) {
ASSERT(nStart > -1);
int32_t iLength = wsExpression.GetLength();
if (nStart >= iLength)
return iLength;
wchar_t* pBuf = wsFilter.GetBuffer(iLength - nStart);
int32_t nCount = 0;
const wchar_t* pSrc = wsExpression.unterminated_c_str();
wchar_t wCur;
while (nStart < iLength) {
wCur = pSrc[nStart++];
if (wCur == ',')
break;
pBuf[nCount++] = wCur;
}
wsFilter.ReleaseBuffer(nCount);
wsFilter.Trim();
return nStart;
}
} // namespace
const CJX_MethodSpec CJX_HostPseudoModel::MethodSpecs[] = {
{"beep", beep_static},
{"documentCountInBatch", documentCountInBatch_static},
{"documentInBatch", documentInBatch_static},
{"exportData", exportData_static},
{"getFocus", getFocus_static},
{"gotoURL", gotoURL_static},
{"importData", importData_static},
{"messageBox", messageBox_static},
{"openList", openList_static},
{"pageDown", pageDown_static},
{"pageUp", pageUp_static},
{"print", print_static},
{"resetData", resetData_static},
{"response", response_static},
{"setFocus", setFocus_static}};
CJX_HostPseudoModel::CJX_HostPseudoModel(CScript_HostPseudoModel* model)
: CJX_Object(model) {
DefineMethods(MethodSpecs, FX_ArraySize(MethodSpecs));
}
CJX_HostPseudoModel::~CJX_HostPseudoModel() {}
void CJX_HostPseudoModel::appType(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowInvalidPropertyException();
return;
}
pValue->SetString("Exchange");
}
void CJX_HostPseudoModel::calculationsEnabled(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
if (bSetting) {
pNotify->GetDocEnvironment()->SetCalculationsEnabled(hDoc,
pValue->ToBoolean());
return;
}
pValue->SetBoolean(pNotify->GetDocEnvironment()->IsCalculationsEnabled(hDoc));
}
void CJX_HostPseudoModel::currentPage(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
if (bSetting) {
pNotify->GetDocEnvironment()->SetCurrentPage(hDoc, pValue->ToInteger());
return;
}
pValue->SetInteger(pNotify->GetDocEnvironment()->GetCurrentPage(hDoc));
}
void CJX_HostPseudoModel::language(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowException(L"Unable to set language value.");
return;
}
pValue->SetString(
pNotify->GetAppProvider()->GetLanguage().UTF8Encode().AsStringView());
}
void CJX_HostPseudoModel::numPages(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
if (bSetting) {
ThrowException(L"Unable to set numPages value.");
return;
}
pValue->SetInteger(pNotify->GetDocEnvironment()->CountPages(hDoc));
}
void CJX_HostPseudoModel::platform(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowException(L"Unable to set platform value.");
return;
}
pValue->SetString(
pNotify->GetAppProvider()->GetPlatform().UTF8Encode().AsStringView());
}
void CJX_HostPseudoModel::title(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return;
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
if (bSetting) {
pNotify->GetDocEnvironment()->SetTitle(hDoc, pValue->ToWideString());
return;
}
WideString wsTitle;
pNotify->GetDocEnvironment()->GetTitle(hDoc, wsTitle);
pValue->SetString(wsTitle.UTF8Encode().AsStringView());
}
void CJX_HostPseudoModel::validationsEnabled(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
if (bSetting) {
pNotify->GetDocEnvironment()->SetValidationsEnabled(hDoc,
pValue->ToBoolean());
return;
}
bool bEnabled = pNotify->GetDocEnvironment()->IsValidationsEnabled(hDoc);
pValue->SetBoolean(bEnabled);
}
void CJX_HostPseudoModel::variation(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return;
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowException(L"Unable to set variation value.");
return;
}
pValue->SetString("Full");
}
void CJX_HostPseudoModel::version(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowException(L"Unable to set version value.");
return;
}
pValue->SetString("11");
}
void CJX_HostPseudoModel::name(CFXJSE_Value* pValue,
bool bSetting,
XFA_Attribute eAttribute) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return;
if (bSetting) {
ThrowInvalidPropertyException();
return;
}
pValue->SetString(
pNotify->GetAppProvider()->GetAppName().UTF8Encode().AsStringView());
}
CJS_Return CJX_HostPseudoModel::gotoURL(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.size() != 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
WideString URL = runtime->ToWideString(params[0]);
pNotify->GetDocEnvironment()->GotoURL(hDoc, URL);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::openList(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.size() != 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_Node* pNode = nullptr;
if (params[0]->IsObject()) {
pNode = ToNode(runtime->ToXFAObject(params[0]));
} else if (params[0]->IsString()) {
CFXJSE_Engine* pScriptContext = GetDocument()->GetScriptContext();
if (!pScriptContext)
return CJS_Return(true);
CXFA_Object* pObject = pScriptContext->GetThisObject();
if (!pObject)
return CJS_Return(true);
uint32_t dwFlag = XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Parent |
XFA_RESOLVENODE_Siblings;
XFA_RESOLVENODE_RS resolveNodeRS;
bool iRet = pScriptContext->ResolveObjects(
pObject, runtime->ToWideString(params[0]).AsStringView(),
&resolveNodeRS, dwFlag, nullptr);
if (!iRet || !resolveNodeRS.objects.front()->IsNode())
return CJS_Return(true);
pNode = resolveNodeRS.objects.front()->AsNode();
}
CXFA_LayoutProcessor* pDocLayout = GetDocument()->GetDocLayout();
if (!pDocLayout)
return CJS_Return(true);
CXFA_FFWidget* hWidget =
pNotify->GetHWidget(pDocLayout->GetLayoutItem(pNode));
if (!hWidget)
return CJS_Return(true);
pNotify->GetDocEnvironment()->SetFocusWidget(pNotify->GetHDOC(), hWidget);
pNotify->OpenDropDownList(hWidget);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::response(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (params.empty() || params.size() > 4)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
WideString question;
if (params.size() >= 1)
question = runtime->ToWideString(params[0]);
WideString title;
if (params.size() >= 2)
title = runtime->ToWideString(params[1]);
WideString defaultAnswer;
if (params.size() >= 3)
defaultAnswer = runtime->ToWideString(params[2]);
bool mark = false;
if (params.size() >= 4)
mark = runtime->ToInt32(params[3]) != 0;
WideString answer =
pNotify->GetAppProvider()->Response(question, title, defaultAnswer, mark);
return CJS_Return(runtime->NewString(answer.UTF8Encode().AsStringView()));
}
CJS_Return CJX_HostPseudoModel::documentInBatch(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
return CJS_Return(runtime->NewNumber(0));
}
CJS_Return CJX_HostPseudoModel::resetData(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (params.size() > 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
WideString expression;
if (params.size() >= 1)
expression = runtime->ToWideString(params[0]);
if (expression.IsEmpty()) {
pNotify->ResetData();
return CJS_Return(true);
}
int32_t iStart = 0;
WideString wsName;
CXFA_Node* pNode = nullptr;
int32_t iExpLength = expression.GetLength();
while (iStart < iExpLength) {
iStart = FilterName(expression.AsStringView(), iStart, wsName);
CFXJSE_Engine* pScriptContext = GetDocument()->GetScriptContext();
if (!pScriptContext)
return CJS_Return(true);
CXFA_Object* pObject = pScriptContext->GetThisObject();
if (!pObject)
return CJS_Return(true);
uint32_t dwFlag = XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Parent |
XFA_RESOLVENODE_Siblings;
XFA_RESOLVENODE_RS resolveNodeRS;
bool iRet = pScriptContext->ResolveObjects(pObject, wsName.AsStringView(),
&resolveNodeRS, dwFlag, nullptr);
if (!iRet || !resolveNodeRS.objects.front()->IsNode())
continue;
pNode = resolveNodeRS.objects.front()->AsNode();
pNotify->ResetData(pNode->GetWidgetAcc());
}
if (!pNode)
pNotify->ResetData();
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::beep(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.size() > 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
uint32_t dwType = 4;
if (params.size() >= 1)
dwType = runtime->ToInt32(params[0]);
pNotify->GetAppProvider()->Beep(dwType);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::setFocus(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.size() != 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_Node* pNode = nullptr;
if (params.size() >= 1) {
if (params[0]->IsObject()) {
pNode = ToNode(runtime->ToXFAObject(params[0]));
} else if (params[0]->IsString()) {
CFXJSE_Engine* pScriptContext = GetDocument()->GetScriptContext();
if (!pScriptContext)
return CJS_Return(true);
CXFA_Object* pObject = pScriptContext->GetThisObject();
if (!pObject)
return CJS_Return(true);
uint32_t dwFlag = XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Parent |
XFA_RESOLVENODE_Siblings;
XFA_RESOLVENODE_RS resolveNodeRS;
bool iRet = pScriptContext->ResolveObjects(
pObject, runtime->ToWideString(params[0]).AsStringView(),
&resolveNodeRS, dwFlag, nullptr);
if (!iRet || !resolveNodeRS.objects.front()->IsNode())
return CJS_Return(true);
pNode = resolveNodeRS.objects.front()->AsNode();
}
}
pNotify->SetFocusWidgetNode(pNode);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::getFocus(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_Node* pNode = pNotify->GetFocusWidgetNode();
if (!pNode)
return CJS_Return(true);
CFXJSE_Value* value =
GetDocument()->GetScriptContext()->GetJSValueFromMap(pNode);
if (!value)
return CJS_Return(runtime->NewNull());
return CJS_Return(value->DirectGetValue().Get(runtime->GetIsolate()));
}
CJS_Return CJX_HostPseudoModel::messageBox(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.empty() || params.size() > 4)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
WideString message;
if (params.size() >= 1)
message = runtime->ToWideString(params[0]);
WideString title;
if (params.size() >= 2)
title = runtime->ToWideString(params[1]);
uint32_t messageType = XFA_MBICON_Error;
if (params.size() >= 3) {
messageType = runtime->ToInt32(params[2]);
if (messageType > XFA_MBICON_Status)
messageType = XFA_MBICON_Error;
}
uint32_t buttonType = XFA_MB_OK;
if (params.size() >= 4) {
buttonType = runtime->ToInt32(params[3]);
if (buttonType > XFA_MB_YesNoCancel)
buttonType = XFA_MB_OK;
}
int32_t iValue = pNotify->GetAppProvider()->MsgBox(message, title,
messageType, buttonType);
return CJS_Return(runtime->NewNumber(iValue));
}
CJS_Return CJX_HostPseudoModel::documentCountInBatch(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
return CJS_Return(runtime->NewNumber(0));
}
CJS_Return CJX_HostPseudoModel::print(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (!GetDocument()->GetScriptContext()->IsRunAtClient())
return CJS_Return(true);
if (params.size() != 8)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
uint32_t dwOptions = 0;
if (runtime->ToBoolean(params[0]))
dwOptions |= XFA_PRINTOPT_ShowDialog;
if (runtime->ToBoolean(params[3]))
dwOptions |= XFA_PRINTOPT_CanCancel;
if (runtime->ToBoolean(params[4]))
dwOptions |= XFA_PRINTOPT_ShrinkPage;
if (runtime->ToBoolean(params[5]))
dwOptions |= XFA_PRINTOPT_AsImage;
if (runtime->ToBoolean(params[6]))
dwOptions |= XFA_PRINTOPT_ReverseOrder;
if (runtime->ToBoolean(params[7]))
dwOptions |= XFA_PRINTOPT_PrintAnnot;
int32_t nStartPage = runtime->ToInt32(params[1]);
int32_t nEndPage = runtime->ToInt32(params[2]);
pNotify->GetDocEnvironment()->Print(pNotify->GetHDOC(), nStartPage, nEndPage,
dwOptions);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::importData(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (params.empty() || params.size() > 1)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::exportData(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
if (params.empty() || params.size() > 2)
return CJS_Return(JSGetStringFromID(JSMessage::kParamError));
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
WideString filePath;
if (params.size() >= 1)
filePath = runtime->ToWideString(params[0]);
bool XDP = true;
if (params.size() >= 2)
XDP = runtime->ToBoolean(params[1]);
pNotify->GetDocEnvironment()->ExportData(pNotify->GetHDOC(), filePath, XDP);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::pageUp(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
int32_t nCurPage = pNotify->GetDocEnvironment()->GetCurrentPage(hDoc);
int32_t nNewPage = 0;
if (nCurPage <= 1)
return CJS_Return(true);
nNewPage = nCurPage - 1;
pNotify->GetDocEnvironment()->SetCurrentPage(hDoc, nNewPage);
return CJS_Return(true);
}
CJS_Return CJX_HostPseudoModel::pageDown(
CJS_V8* runtime,
const std::vector<v8::Local<v8::Value>>& params) {
CXFA_FFNotify* pNotify = GetDocument()->GetNotify();
if (!pNotify)
return CJS_Return(true);
CXFA_FFDoc* hDoc = pNotify->GetHDOC();
int32_t nCurPage = pNotify->GetDocEnvironment()->GetCurrentPage(hDoc);
int32_t nPageCount = pNotify->GetDocEnvironment()->CountPages(hDoc);
if (!nPageCount || nCurPage == nPageCount)
return CJS_Return(true);
int32_t nNewPage = 0;
if (nCurPage >= nPageCount)
nNewPage = nPageCount - 1;
else
nNewPage = nCurPage + 1;
pNotify->GetDocEnvironment()->SetCurrentPage(hDoc, nNewPage);
return CJS_Return(true);
}