1 // Copyright 2014 The PDFium Authors
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 
7 #include "fxjs/cjs_app.h"
8 
9 #include <stdint.h>
10 
11 #include <algorithm>
12 #include <utility>
13 
14 #include "core/fxcrt/fixed_zeroed_data_vector.h"
15 #include "core/fxcrt/stl_util.h"
16 #include "fpdfsdk/cpdfsdk_formfillenvironment.h"
17 #include "fpdfsdk/cpdfsdk_interactiveform.h"
18 #include "fxjs/cjs_document.h"
19 #include "fxjs/cjs_timerobj.h"
20 #include "fxjs/global_timer.h"
21 #include "fxjs/ijs_event_context.h"
22 #include "fxjs/js_resources.h"
23 #include "v8/include/v8-container.h"
24 
25 namespace {
26 
27 constexpr wchar_t kStrViewerType[] = L"pdfium";
28 constexpr wchar_t kStrViewerVariation[] = L"Full";
29 constexpr wchar_t kStrPlatform[] = L"WIN";
30 constexpr wchar_t kStrLanguage[] = L"ENU";
31 constexpr int kNumViewerVersion = 8;
32 constexpr int kNumViewerVersionXfa = 11;
33 constexpr int kNumFormsVersion = 7;
34 
35 }  // namespace
36 
37 const JSPropertySpec CJS_App::PropertySpecs[] = {
38     {"activeDocs", get_active_docs_static, set_active_docs_static},
39     {"calculate", get_calculate_static, set_calculate_static},
40     {"formsVersion", get_forms_version_static, set_forms_version_static},
41     {"fs", get_fs_static, set_fs_static},
42     {"fullscreen", get_fullscreen_static, set_fullscreen_static},
43     {"language", get_language_static, set_language_static},
44     {"media", get_media_static, set_media_static},
45     {"platform", get_platform_static, set_platform_static},
46     {"runtimeHighlight", get_runtime_highlight_static,
47      set_runtime_highlight_static},
48     {"viewerType", get_viewer_type_static, set_viewer_type_static},
49     {"viewerVariation", get_viewer_variation_static,
50      set_viewer_variation_static},
51     {"viewerVersion", get_viewer_version_static, set_viewer_version_static}};
52 
53 const JSMethodSpec CJS_App::MethodSpecs[] = {
54     {"alert", alert_static},
55     {"beep", beep_static},
56     {"browseForDoc", browseForDoc_static},
57     {"clearInterval", clearInterval_static},
58     {"clearTimeOut", clearTimeOut_static},
59     {"execDialog", execDialog_static},
60     {"execMenuItem", execMenuItem_static},
61     {"findComponent", findComponent_static},
62     {"goBack", goBack_static},
63     {"goForward", goForward_static},
64     {"launchURL", launchURL_static},
65     {"mailMsg", mailMsg_static},
66     {"newFDF", newFDF_static},
67     {"newDoc", newDoc_static},
68     {"openDoc", openDoc_static},
69     {"openFDF", openFDF_static},
70     {"popUpMenuEx", popUpMenuEx_static},
71     {"popUpMenu", popUpMenu_static},
72     {"response", response_static},
73     {"setInterval", setInterval_static},
74     {"setTimeOut", setTimeOut_static}};
75 
76 uint32_t CJS_App::ObjDefnID = 0;
77 
78 const char CJS_App::kName[] = "app";
79 
80 // static
GetObjDefnID()81 uint32_t CJS_App::GetObjDefnID() {
82   return ObjDefnID;
83 }
84 
85 // static
DefineJSObjects(CFXJS_Engine * pEngine)86 void CJS_App::DefineJSObjects(CFXJS_Engine* pEngine) {
87   ObjDefnID = pEngine->DefineObj(CJS_App::kName, FXJSOBJTYPE_STATIC,
88                                  JSConstructor<CJS_App>, JSDestructor);
89   DefineProps(pEngine, ObjDefnID, PropertySpecs);
90   DefineMethods(pEngine, ObjDefnID, MethodSpecs);
91 }
92 
CJS_App(v8::Local<v8::Object> pObject,CJS_Runtime * pRuntime)93 CJS_App::CJS_App(v8::Local<v8::Object> pObject, CJS_Runtime* pRuntime)
94     : CJS_Object(pObject, pRuntime) {}
95 
96 CJS_App::~CJS_App() = default;
97 
get_active_docs(CJS_Runtime * pRuntime)98 CJS_Result CJS_App::get_active_docs(CJS_Runtime* pRuntime) {
99   v8::Local<v8::Object> pObj = pRuntime->GetThisObj();
100   auto pJSDocument = JSGetObject<CJS_Document>(pRuntime->GetIsolate(), pObj);
101   if (!pJSDocument)
102     return CJS_Result::Failure(JSMessage::kObjectTypeError);
103   v8::Local<v8::Array> aDocs = pRuntime->NewArray();
104   pRuntime->PutArrayElement(aDocs, 0, pJSDocument->ToV8Object());
105   if (pRuntime->GetArrayLength(aDocs) > 0)
106     return CJS_Result::Success(aDocs);
107 
108   return CJS_Result::Success(pRuntime->NewUndefined());
109 }
110 
set_active_docs(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)111 CJS_Result CJS_App::set_active_docs(CJS_Runtime* pRuntime,
112                                     v8::Local<v8::Value> vp) {
113   return CJS_Result::Failure(JSMessage::kNotSupportedError);
114 }
115 
get_calculate(CJS_Runtime * pRuntime)116 CJS_Result CJS_App::get_calculate(CJS_Runtime* pRuntime) {
117   return CJS_Result::Success(pRuntime->NewBoolean(m_bCalculate));
118 }
119 
set_calculate(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)120 CJS_Result CJS_App::set_calculate(CJS_Runtime* pRuntime,
121                                   v8::Local<v8::Value> vp) {
122   m_bCalculate = pRuntime->ToBoolean(vp);
123   pRuntime->GetFormFillEnv()->GetInteractiveForm()->EnableCalculate(
124       m_bCalculate);
125   return CJS_Result::Success();
126 }
127 
get_forms_version(CJS_Runtime * pRuntime)128 CJS_Result CJS_App::get_forms_version(CJS_Runtime* pRuntime) {
129   return CJS_Result::Success(pRuntime->NewNumber(kNumFormsVersion));
130 }
131 
set_forms_version(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)132 CJS_Result CJS_App::set_forms_version(CJS_Runtime* pRuntime,
133                                       v8::Local<v8::Value> vp) {
134   return CJS_Result::Failure(JSMessage::kNotSupportedError);
135 }
136 
get_viewer_type(CJS_Runtime * pRuntime)137 CJS_Result CJS_App::get_viewer_type(CJS_Runtime* pRuntime) {
138   return CJS_Result::Success(pRuntime->NewString(kStrViewerType));
139 }
140 
set_viewer_type(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)141 CJS_Result CJS_App::set_viewer_type(CJS_Runtime* pRuntime,
142                                     v8::Local<v8::Value> vp) {
143   return CJS_Result::Failure(JSMessage::kNotSupportedError);
144 }
145 
get_viewer_variation(CJS_Runtime * pRuntime)146 CJS_Result CJS_App::get_viewer_variation(CJS_Runtime* pRuntime) {
147   return CJS_Result::Success(pRuntime->NewString(kStrViewerVariation));
148 }
149 
set_viewer_variation(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)150 CJS_Result CJS_App::set_viewer_variation(CJS_Runtime* pRuntime,
151                                          v8::Local<v8::Value> vp) {
152   return CJS_Result::Failure(JSMessage::kNotSupportedError);
153 }
154 
get_viewer_version(CJS_Runtime * pRuntime)155 CJS_Result CJS_App::get_viewer_version(CJS_Runtime* pRuntime) {
156   CPDF_Document::Extension* pContext =
157       pRuntime->GetFormFillEnv()->GetDocExtension();
158   int version = pContext && pContext->ContainsExtensionForm()
159                     ? kNumViewerVersionXfa
160                     : kNumViewerVersion;
161   return CJS_Result::Success(pRuntime->NewNumber(version));
162 }
163 
set_viewer_version(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)164 CJS_Result CJS_App::set_viewer_version(CJS_Runtime* pRuntime,
165                                        v8::Local<v8::Value> vp) {
166   return CJS_Result::Failure(JSMessage::kNotSupportedError);
167 }
168 
get_platform(CJS_Runtime * pRuntime)169 CJS_Result CJS_App::get_platform(CJS_Runtime* pRuntime) {
170   CPDFSDK_FormFillEnvironment* pFormFillEnv = pRuntime->GetFormFillEnv();
171   if (pFormFillEnv) {
172     WideString platform = pFormFillEnv->GetPlatform();
173     if (!platform.IsEmpty())
174       return CJS_Result::Success(pRuntime->NewString(platform.AsStringView()));
175   }
176   return CJS_Result::Success(pRuntime->NewString(kStrPlatform));
177 }
178 
set_platform(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)179 CJS_Result CJS_App::set_platform(CJS_Runtime* pRuntime,
180                                  v8::Local<v8::Value> vp) {
181   return CJS_Result::Failure(JSMessage::kNotSupportedError);
182 }
183 
get_language(CJS_Runtime * pRuntime)184 CJS_Result CJS_App::get_language(CJS_Runtime* pRuntime) {
185   CPDFSDK_FormFillEnvironment* pFormFillEnv = pRuntime->GetFormFillEnv();
186   if (pFormFillEnv) {
187     WideString language = pFormFillEnv->GetLanguage();
188     if (!language.IsEmpty())
189       return CJS_Result::Success(pRuntime->NewString(language.AsStringView()));
190   }
191   return CJS_Result::Success(pRuntime->NewString(kStrLanguage));
192 }
193 
set_language(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)194 CJS_Result CJS_App::set_language(CJS_Runtime* pRuntime,
195                                  v8::Local<v8::Value> vp) {
196   return CJS_Result::Failure(JSMessage::kNotSupportedError);
197 }
198 
199 // creates a new fdf object that contains no data
200 // comment: need reader support
201 // note:
202 // CFDF_Document * CPDFSDK_FormFillEnvironment::NewFDF();
newFDF(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)203 CJS_Result CJS_App::newFDF(CJS_Runtime* pRuntime,
204                            const std::vector<v8::Local<v8::Value>>& params) {
205   return CJS_Result::Success();
206 }
207 
208 // opens a specified pdf document and returns its document object
209 // comment:need reader support
210 // note: as defined in js reference, the proto of this function's fourth
211 // parmeters, how old an fdf document while do not show it.
212 // CFDF_Document * CPDFSDK_FormFillEnvironment::OpenFDF(string strPath,bool
213 // bUserConv);
214 
openFDF(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)215 CJS_Result CJS_App::openFDF(CJS_Runtime* pRuntime,
216                             const std::vector<v8::Local<v8::Value>>& params) {
217   return CJS_Result::Success();
218 }
219 
alert(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)220 CJS_Result CJS_App::alert(CJS_Runtime* pRuntime,
221                           const std::vector<v8::Local<v8::Value>>& params) {
222   std::vector<v8::Local<v8::Value>> newParams = ExpandKeywordParams(
223       pRuntime, params, 4, "cMsg", "nIcon", "nType", "cTitle");
224 
225   if (!IsExpandedParamKnown(newParams[0]))
226     return CJS_Result::Failure(JSMessage::kParamError);
227 
228   CPDFSDK_FormFillEnvironment* pFormFillEnv = pRuntime->GetFormFillEnv();
229   if (!pFormFillEnv)
230     return CJS_Result::Success(pRuntime->NewNumber(0));
231 
232   WideString swMsg;
233   if (newParams[0]->IsArray()) {
234     v8::Local<v8::Array> carray = pRuntime->ToArray(newParams[0]);
235     swMsg = L"[";
236     for (size_t i = 0; i < pRuntime->GetArrayLength(carray); ++i) {
237       if (i)
238         swMsg += L", ";
239 
240       swMsg += pRuntime->ToWideString(pRuntime->GetArrayElement(carray, i));
241     }
242     swMsg += L"]";
243   } else {
244     swMsg = pRuntime->ToWideString(newParams[0]);
245   }
246 
247   int iIcon = JSPLATFORM_ALERT_ICON_DEFAULT;
248   if (IsExpandedParamKnown(newParams[1]))
249     iIcon = pRuntime->ToInt32(newParams[1]);
250 
251   int iType = JSPLATFORM_ALERT_BUTTON_DEFAULT;
252   if (IsExpandedParamKnown(newParams[2]))
253     iType = pRuntime->ToInt32(newParams[2]);
254 
255   WideString swTitle;
256   if (IsExpandedParamKnown(newParams[3]))
257     swTitle = pRuntime->ToWideString(newParams[3]);
258   else
259     swTitle = JSGetStringFromID(JSMessage::kAlert);
260 
261   pRuntime->BeginBlock();
262   pFormFillEnv->KillFocusAnnot({});
263   v8::Local<v8::Value> ret = pRuntime->NewNumber(
264       pFormFillEnv->JS_appAlert(swMsg, swTitle, iType, iIcon));
265   pRuntime->EndBlock();
266 
267   return CJS_Result::Success(ret);
268 }
269 
beep(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)270 CJS_Result CJS_App::beep(CJS_Runtime* pRuntime,
271                          const std::vector<v8::Local<v8::Value>>& params) {
272   if (params.size() != 1)
273     return CJS_Result::Failure(JSMessage::kParamError);
274 
275   int type = JSPLATFORM_BEEP_DEFAULT;
276   if (IsExpandedParamKnown(params[0]))
277     type = pRuntime->ToInt32(params[0]);
278 
279   pRuntime->GetFormFillEnv()->JS_appBeep(type);
280   return CJS_Result::Success();
281 }
282 
findComponent(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)283 CJS_Result CJS_App::findComponent(
284     CJS_Runtime* pRuntime,
285     const std::vector<v8::Local<v8::Value>>& params) {
286   return CJS_Result::Success();
287 }
288 
popUpMenuEx(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)289 CJS_Result CJS_App::popUpMenuEx(
290     CJS_Runtime* pRuntime,
291     const std::vector<v8::Local<v8::Value>>& params) {
292   return CJS_Result::Failure(JSMessage::kNotSupportedError);
293 }
294 
get_fs(CJS_Runtime * pRuntime)295 CJS_Result CJS_App::get_fs(CJS_Runtime* pRuntime) {
296   return CJS_Result::Failure(JSMessage::kNotSupportedError);
297 }
298 
set_fs(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)299 CJS_Result CJS_App::set_fs(CJS_Runtime* pRuntime, v8::Local<v8::Value> vp) {
300   return CJS_Result::Failure(JSMessage::kNotSupportedError);
301 }
302 
setInterval(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)303 CJS_Result CJS_App::setInterval(
304     CJS_Runtime* pRuntime,
305     const std::vector<v8::Local<v8::Value>>& params) {
306   if (params.size() == 0 || params.size() > 2)
307     return CJS_Result::Failure(JSMessage::kParamError);
308 
309   WideString script = pRuntime->ToWideString(params[0]);
310   if (script.IsEmpty())
311     return CJS_Result::Failure(JSMessage::kInvalidInputError);
312 
313   uint32_t dwInterval = params.size() > 1 ? pRuntime->ToInt32(params[1]) : 1000;
314   auto timerRef = std::make_unique<GlobalTimer>(
315       this, pRuntime, GlobalTimer::Type::kRepeating, script, dwInterval, 0);
316   GlobalTimer* pTimerRef = timerRef.get();
317   m_Timers.insert(std::move(timerRef));
318 
319   v8::Local<v8::Object> pRetObj = pRuntime->NewFXJSBoundObject(
320       CJS_TimerObj::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC);
321   if (pRetObj.IsEmpty())
322     return CJS_Result::Failure(JSMessage::kBadObjectError);
323 
324   auto* pJS_TimerObj = static_cast<CJS_TimerObj*>(
325       CFXJS_Engine::GetObjectPrivate(pRuntime->GetIsolate(), pRetObj));
326 
327   pJS_TimerObj->SetTimer(pTimerRef);
328   return CJS_Result::Success(pRetObj);
329 }
330 
setTimeOut(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)331 CJS_Result CJS_App::setTimeOut(
332     CJS_Runtime* pRuntime,
333     const std::vector<v8::Local<v8::Value>>& params) {
334   if (params.size() == 0 || params.size() > 2)
335     return CJS_Result::Failure(JSMessage::kParamError);
336 
337   WideString script = pRuntime->ToWideString(params[0]);
338   if (script.IsEmpty())
339     return CJS_Result::Failure(JSMessage::kInvalidInputError);
340 
341   uint32_t dwTimeOut = params.size() > 1 ? pRuntime->ToInt32(params[1]) : 1000;
342   auto timerRef =
343       std::make_unique<GlobalTimer>(this, pRuntime, GlobalTimer::Type::kOneShot,
344                                     script, dwTimeOut, dwTimeOut);
345   GlobalTimer* pTimerRef = timerRef.get();
346   m_Timers.insert(std::move(timerRef));
347 
348   v8::Local<v8::Object> pRetObj = pRuntime->NewFXJSBoundObject(
349       CJS_TimerObj::GetObjDefnID(), FXJSOBJTYPE_DYNAMIC);
350   if (pRetObj.IsEmpty())
351     return CJS_Result::Failure(JSMessage::kBadObjectError);
352 
353   auto* pJS_TimerObj = static_cast<CJS_TimerObj*>(
354       CFXJS_Engine::GetObjectPrivate(pRuntime->GetIsolate(), pRetObj));
355 
356   pJS_TimerObj->SetTimer(pTimerRef);
357   return CJS_Result::Success(pRetObj);
358 }
359 
clearTimeOut(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)360 CJS_Result CJS_App::clearTimeOut(
361     CJS_Runtime* pRuntime,
362     const std::vector<v8::Local<v8::Value>>& params) {
363   if (params.size() != 1)
364     return CJS_Result::Failure(JSMessage::kParamError);
365 
366   CJS_App::ClearTimerCommon(pRuntime, params[0]);
367   return CJS_Result::Success();
368 }
369 
clearInterval(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)370 CJS_Result CJS_App::clearInterval(
371     CJS_Runtime* pRuntime,
372     const std::vector<v8::Local<v8::Value>>& params) {
373   if (params.size() != 1)
374     return CJS_Result::Failure(JSMessage::kParamError);
375 
376   CJS_App::ClearTimerCommon(pRuntime, params[0]);
377   return CJS_Result::Success();
378 }
379 
ClearTimerCommon(CJS_Runtime * pRuntime,v8::Local<v8::Value> param)380 void CJS_App::ClearTimerCommon(CJS_Runtime* pRuntime,
381                                v8::Local<v8::Value> param) {
382   if (!param->IsObject())
383     return;
384 
385   v8::Local<v8::Object> pObj = pRuntime->ToObject(param);
386   auto pTimer = JSGetObject<CJS_TimerObj>(pRuntime->GetIsolate(), pObj);
387   if (!pTimer)
388     return;
389 
390   GlobalTimer::Cancel(pTimer->GetTimerID());
391 }
392 
execMenuItem(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)393 CJS_Result CJS_App::execMenuItem(
394     CJS_Runtime* pRuntime,
395     const std::vector<v8::Local<v8::Value>>& params) {
396   return CJS_Result::Failure(JSMessage::kNotSupportedError);
397 }
398 
TimerProc(GlobalTimer * pTimer)399 void CJS_App::TimerProc(GlobalTimer* pTimer) {
400   CJS_Runtime* pRuntime = pTimer->GetRuntime();
401   if (pRuntime && (!pTimer->IsOneShot() || pTimer->GetTimeOut() > 0))
402     RunJsScript(pRuntime, pTimer->GetJScript());
403 }
404 
CancelProc(GlobalTimer * pTimer)405 void CJS_App::CancelProc(GlobalTimer* pTimer) {
406   m_Timers.erase(fxcrt::MakeFakeUniquePtr(pTimer));
407 }
408 
RunJsScript(CJS_Runtime * pRuntime,const WideString & wsScript)409 void CJS_App::RunJsScript(CJS_Runtime* pRuntime, const WideString& wsScript) {
410   if (pRuntime->IsBlocking())
411     return;
412 
413   IJS_Runtime::ScopedEventContext pContext(pRuntime);
414   pContext->OnExternal_Exec();
415   pContext->RunScript(wsScript);
416 }
417 
goBack(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)418 CJS_Result CJS_App::goBack(CJS_Runtime* pRuntime,
419                            const std::vector<v8::Local<v8::Value>>& params) {
420   // Not supported, but do not return error.
421   return CJS_Result::Success();
422 }
423 
goForward(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)424 CJS_Result CJS_App::goForward(CJS_Runtime* pRuntime,
425                               const std::vector<v8::Local<v8::Value>>& params) {
426   // Not supported, but do not return error.
427   return CJS_Result::Success();
428 }
429 
mailMsg(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)430 CJS_Result CJS_App::mailMsg(CJS_Runtime* pRuntime,
431                             const std::vector<v8::Local<v8::Value>>& params) {
432   std::vector<v8::Local<v8::Value>> newParams = ExpandKeywordParams(
433       pRuntime, params, 6, "bUI", "cTo", "cCc", "cBcc", "cSubject", "cMsg");
434 
435   if (!IsExpandedParamKnown(newParams[0]))
436     return CJS_Result::Failure(JSMessage::kParamError);
437 
438   bool bUI = pRuntime->ToBoolean(newParams[0]);
439   WideString cTo;
440   if (IsExpandedParamKnown(newParams[1])) {
441     cTo = pRuntime->ToWideString(newParams[1]);
442   } else {
443     // cTo parameter required when UI not invoked.
444     if (!bUI)
445       return CJS_Result::Failure(JSMessage::kParamError);
446   }
447 
448   WideString cCc;
449   if (IsExpandedParamKnown(newParams[2]))
450     cCc = pRuntime->ToWideString(newParams[2]);
451 
452   WideString cBcc;
453   if (IsExpandedParamKnown(newParams[3]))
454     cBcc = pRuntime->ToWideString(newParams[3]);
455 
456   WideString cSubject;
457   if (IsExpandedParamKnown(newParams[4]))
458     cSubject = pRuntime->ToWideString(newParams[4]);
459 
460   WideString cMsg;
461   if (IsExpandedParamKnown(newParams[5]))
462     cMsg = pRuntime->ToWideString(newParams[5]);
463 
464   pRuntime->BeginBlock();
465   pRuntime->GetFormFillEnv()->JS_docmailForm(pdfium::span<const uint8_t>(), bUI,
466                                              cTo, cSubject, cCc, cBcc, cMsg);
467   pRuntime->EndBlock();
468   return CJS_Result::Success();
469 }
470 
launchURL(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)471 CJS_Result CJS_App::launchURL(CJS_Runtime* pRuntime,
472                               const std::vector<v8::Local<v8::Value>>& params) {
473   // Unsafe, not supported, but do not return error.
474   return CJS_Result::Success();
475 }
476 
get_runtime_highlight(CJS_Runtime * pRuntime)477 CJS_Result CJS_App::get_runtime_highlight(CJS_Runtime* pRuntime) {
478   return CJS_Result::Success(pRuntime->NewBoolean(m_bRuntimeHighLight));
479 }
480 
set_runtime_highlight(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)481 CJS_Result CJS_App::set_runtime_highlight(CJS_Runtime* pRuntime,
482                                           v8::Local<v8::Value> vp) {
483   m_bRuntimeHighLight = pRuntime->ToBoolean(vp);
484   return CJS_Result::Success();
485 }
486 
get_fullscreen(CJS_Runtime * pRuntime)487 CJS_Result CJS_App::get_fullscreen(CJS_Runtime* pRuntime) {
488   return CJS_Result::Failure(JSMessage::kNotSupportedError);
489 }
490 
set_fullscreen(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)491 CJS_Result CJS_App::set_fullscreen(CJS_Runtime* pRuntime,
492                                    v8::Local<v8::Value> vp) {
493   return CJS_Result::Failure(JSMessage::kNotSupportedError);
494 }
495 
popUpMenu(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)496 CJS_Result CJS_App::popUpMenu(CJS_Runtime* pRuntime,
497                               const std::vector<v8::Local<v8::Value>>& params) {
498   return CJS_Result::Failure(JSMessage::kNotSupportedError);
499 }
500 
browseForDoc(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)501 CJS_Result CJS_App::browseForDoc(
502     CJS_Runtime* pRuntime,
503     const std::vector<v8::Local<v8::Value>>& params) {
504   // Unsafe, not supported, but do not return an error.
505   return CJS_Result::Success();
506 }
507 
SysPathToPDFPath(const WideString & sOldPath)508 WideString CJS_App::SysPathToPDFPath(const WideString& sOldPath) {
509   WideString sRet = L"/";
510   for (const wchar_t& c : sOldPath) {
511     if (c != L':')
512       sRet += (c == L'\\') ? L'/' : c;
513   }
514   return sRet;
515 }
516 
newDoc(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)517 CJS_Result CJS_App::newDoc(CJS_Runtime* pRuntime,
518                            const std::vector<v8::Local<v8::Value>>& params) {
519   return CJS_Result::Failure(JSMessage::kNotSupportedError);
520 }
521 
openDoc(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)522 CJS_Result CJS_App::openDoc(CJS_Runtime* pRuntime,
523                             const std::vector<v8::Local<v8::Value>>& params) {
524   return CJS_Result::Failure(JSMessage::kNotSupportedError);
525 }
526 
response(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)527 CJS_Result CJS_App::response(CJS_Runtime* pRuntime,
528                              const std::vector<v8::Local<v8::Value>>& params) {
529   std::vector<v8::Local<v8::Value>> newParams =
530       ExpandKeywordParams(pRuntime, params, 5, "cQuestion", "cTitle",
531                           "cDefault", "bPassword", "cLabel");
532 
533   if (!IsExpandedParamKnown(newParams[0]))
534     return CJS_Result::Failure(JSMessage::kParamError);
535 
536   WideString swQuestion = pRuntime->ToWideString(newParams[0]);
537   WideString swTitle = L"PDF";
538   if (IsExpandedParamKnown(newParams[1]))
539     swTitle = pRuntime->ToWideString(newParams[1]);
540 
541   WideString swDefault;
542   if (IsExpandedParamKnown(newParams[2]))
543     swDefault = pRuntime->ToWideString(newParams[2]);
544 
545   bool bPassword = false;
546   if (IsExpandedParamKnown(newParams[3]))
547     bPassword = pRuntime->ToBoolean(newParams[3]);
548 
549   WideString swLabel;
550   if (IsExpandedParamKnown(newParams[4]))
551     swLabel = pRuntime->ToWideString(newParams[4]);
552 
553   constexpr int kMaxWideChars = 1024;
554   constexpr int kMaxBytes = kMaxWideChars * sizeof(uint16_t);
555   FixedZeroedDataVector<uint16_t> buffer(kMaxWideChars);
556   pdfium::span<uint16_t> buffer_span = buffer.writable_span();
557   int byte_length = pRuntime->GetFormFillEnv()->JS_appResponse(
558       swQuestion, swTitle, swDefault, swLabel, bPassword,
559       pdfium::as_writable_bytes(buffer_span));
560   if (byte_length < 0 || byte_length > kMaxBytes)
561     return CJS_Result::Failure(JSMessage::kParamTooLongError);
562 
563   buffer_span = buffer_span.first(
564       std::min<size_t>(kMaxWideChars, byte_length / sizeof(uint16_t)));
565   return CJS_Result::Success(pRuntime->NewString(
566       WideString::FromUTF16LE(buffer_span.data(), buffer_span.size())
567           .AsStringView()));
568 }
569 
get_media(CJS_Runtime * pRuntime)570 CJS_Result CJS_App::get_media(CJS_Runtime* pRuntime) {
571   return CJS_Result::Failure(JSMessage::kNotSupportedError);
572 }
573 
set_media(CJS_Runtime * pRuntime,v8::Local<v8::Value> vp)574 CJS_Result CJS_App::set_media(CJS_Runtime* pRuntime, v8::Local<v8::Value> vp) {
575   return CJS_Result::Failure(JSMessage::kNotSupportedError);
576 }
577 
execDialog(CJS_Runtime * pRuntime,const std::vector<v8::Local<v8::Value>> & params)578 CJS_Result CJS_App::execDialog(
579     CJS_Runtime* pRuntime,
580     const std::vector<v8::Local<v8::Value>>& params) {
581   return CJS_Result::Success();
582 }
583