John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 1 | // Copyright 2014 PDFium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
Lei Zhang | a6d9f0e | 2015-06-13 00:48:38 -0700 | [diff] [blame] | 4 | |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 5 | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| 6 | |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 7 | #include "../../include/javascript/IJavaScript.h" |
Tom Sepez | 3a83266 | 2015-03-02 12:59:05 -0800 | [diff] [blame] | 8 | #include "../../include/javascript/JS_Context.h" |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 9 | #include "../../include/javascript/JS_Define.h" |
Tom Sepez | 3a83266 | 2015-03-02 12:59:05 -0800 | [diff] [blame] | 10 | #include "../../include/javascript/JS_EventHandler.h" |
| 11 | #include "../../include/javascript/JS_GlobalData.h" |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 12 | #include "../../include/javascript/JS_Object.h" |
| 13 | #include "../../include/javascript/JS_Value.h" |
Tom Sepez | 3a83266 | 2015-03-02 12:59:05 -0800 | [diff] [blame] | 14 | #include "../../include/javascript/JavaScript.h" |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 15 | #include "../../include/javascript/global.h" |
Bo Xu | fdc00a7 | 2014-10-28 23:03:33 -0700 | [diff] [blame] | 16 | #include "../../include/fpdfxfa/fpdfxfa_app.h" |
Tom Sepez | 3a83266 | 2015-03-02 12:59:05 -0800 | [diff] [blame] | 17 | #include "../../include/javascript/resource.h" |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 18 | |
| 19 | /* ---------------------------- global ---------------------------- */ |
| 20 | |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 21 | // Helper class for compile-time calculation of hash values in order to |
| 22 | // avoid having global object initializers. |
Lei Zhang | 659b8d3 | 2015-02-27 19:49:57 -0800 | [diff] [blame] | 23 | template <unsigned ACC, wchar_t... Ns> |
| 24 | struct CHash; |
| 25 | |
| 26 | // Only needed to hash single-character strings. |
| 27 | template <wchar_t N> |
| 28 | struct CHash<N> { |
| 29 | static const unsigned value = N; |
| 30 | }; |
| 31 | |
| 32 | template <unsigned ACC, wchar_t N> |
| 33 | struct CHash<ACC, N> { |
| 34 | static const unsigned value = (ACC * 1313LLU + N) & 0xFFFFFFFF; |
| 35 | }; |
| 36 | |
| 37 | template <unsigned ACC, wchar_t N, wchar_t... Ns> |
| 38 | struct CHash<ACC, N, Ns...> { |
| 39 | static const unsigned value = CHash<CHash<ACC, N>::value, Ns...>::value; |
| 40 | }; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 41 | |
| 42 | extern const unsigned int JSCONST_nStringHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 43 | CHash<'s', 't', 'r', 'i', 'n', 'g'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 44 | extern const unsigned int JSCONST_nNumberHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 45 | CHash<'n', 'u', 'm', 'b', 'e', 'r'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 46 | extern const unsigned int JSCONST_nBoolHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 47 | CHash<'b', 'o', 'o', 'l', 'e', 'a', 'n'>::value; |
| 48 | extern const unsigned int JSCONST_nDateHash = CHash<'d', 'a', 't', 'e'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 49 | extern const unsigned int JSCONST_nObjectHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 50 | CHash<'o', 'b', 'j', 'e', 'c', 't'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 51 | extern const unsigned int JSCONST_nFXobjHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 52 | CHash<'f', 'x', 'o', 'b', 'j'>::value; |
| 53 | extern const unsigned int JSCONST_nNullHash = CHash<'n', 'u', 'l', 'l'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 54 | extern const unsigned int JSCONST_nUndefHash = |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 55 | CHash<'u', 'n', 'd', 'e', 'f', 'i', 'n', 'e', 'd'>::value; |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 56 | |
| 57 | #ifdef _DEBUG |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 58 | class HashVerify { |
| 59 | public: |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 60 | HashVerify(); |
| 61 | } g_hashVerify; |
| 62 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 63 | HashVerify::HashVerify() { |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 64 | ASSERT(JSCONST_nStringHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 65 | JS_CalcHash(VALUE_NAME_STRING, wcslen(VALUE_NAME_STRING))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 66 | ASSERT(JSCONST_nNumberHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 67 | JS_CalcHash(VALUE_NAME_NUMBER, wcslen(VALUE_NAME_NUMBER))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 68 | ASSERT(JSCONST_nBoolHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 69 | JS_CalcHash(VALUE_NAME_BOOLEAN, wcslen(VALUE_NAME_BOOLEAN))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 70 | ASSERT(JSCONST_nDateHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 71 | JS_CalcHash(VALUE_NAME_DATE, wcslen(VALUE_NAME_DATE))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 72 | ASSERT(JSCONST_nObjectHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 73 | JS_CalcHash(VALUE_NAME_OBJECT, wcslen(VALUE_NAME_OBJECT))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 74 | ASSERT(JSCONST_nFXobjHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 75 | JS_CalcHash(VALUE_NAME_FXOBJ, wcslen(VALUE_NAME_FXOBJ))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 76 | ASSERT(JSCONST_nNullHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 77 | JS_CalcHash(VALUE_NAME_NULL, wcslen(VALUE_NAME_NULL))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 78 | ASSERT(JSCONST_nUndefHash == |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 79 | JS_CalcHash(VALUE_NAME_UNDEFINED, wcslen(VALUE_NAME_UNDEFINED))); |
Bruce Dawson | e68d624 | 2015-01-05 15:18:21 -0800 | [diff] [blame] | 80 | } |
| 81 | #endif |
| 82 | |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 83 | BEGIN_JS_STATIC_CONST(CJS_Global) |
| 84 | END_JS_STATIC_CONST() |
| 85 | |
| 86 | BEGIN_JS_STATIC_PROP(CJS_Global) |
| 87 | END_JS_STATIC_PROP() |
| 88 | |
| 89 | BEGIN_JS_STATIC_METHOD(CJS_Global) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 90 | JS_STATIC_METHOD_ENTRY(setPersistent) |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 91 | END_JS_STATIC_METHOD() |
| 92 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 93 | IMPLEMENT_SPECIAL_JS_CLASS(CJS_Global, JSGlobalAlternate, global); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 94 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 95 | FX_BOOL CJS_Global::InitInstance(IFXJS_Context* cc) { |
| 96 | CJS_Context* pContext = (CJS_Context*)cc; |
| 97 | ASSERT(pContext != NULL); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 98 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 99 | JSGlobalAlternate* pGlobal = (JSGlobalAlternate*)GetEmbedObject(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 100 | ASSERT(pGlobal != NULL); |
Lei Zhang | a6d9f0e | 2015-06-13 00:48:38 -0700 | [diff] [blame] | 101 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 102 | pGlobal->Initial(pContext->GetReaderApp()); |
Lei Zhang | a6d9f0e | 2015-06-13 00:48:38 -0700 | [diff] [blame] | 103 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 104 | return TRUE; |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 105 | }; |
| 106 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 107 | JSGlobalAlternate::JSGlobalAlternate(CJS_Object* pJSObject) |
| 108 | : CJS_EmbedObj(pJSObject), m_pApp(NULL) { |
| 109 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 110 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 111 | JSGlobalAlternate::~JSGlobalAlternate() { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 112 | DestroyGlobalPersisitentVariables(); |
| 113 | CPDFXFA_App::GetInstance()->GetRuntimeFactory()->ReleaseGlobalData(); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 114 | } |
| 115 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 116 | void JSGlobalAlternate::Initial(CPDFDoc_Environment* pApp) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 117 | m_pApp = pApp; |
| 118 | m_pGlobalData = |
| 119 | CPDFXFA_App::GetInstance()->GetRuntimeFactory()->NewGlobalData(pApp); |
| 120 | UpdateGlobalPersistentVariables(); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 121 | } |
Tom Sepez | bdeeb8a | 2015-05-27 12:25:00 -0700 | [diff] [blame] | 122 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 123 | FX_BOOL JSGlobalAlternate::QueryProperty(const FX_WCHAR* propname) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 124 | return CFX_WideString(propname) != L"setPersistent"; |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 125 | } |
| 126 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 127 | FX_BOOL JSGlobalAlternate::DelProperty(IFXJS_Context* cc, |
| 128 | const FX_WCHAR* propname, |
| 129 | CFX_WideString& sError) { |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 130 | auto it = m_mapGlobal.find(CFX_ByteString::FromUnicode(propname)); |
| 131 | if (it == m_mapGlobal.end()) |
| 132 | return FALSE; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 133 | |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 134 | it->second->bDeleted = TRUE; |
| 135 | return TRUE; |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 136 | } |
| 137 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 138 | FX_BOOL JSGlobalAlternate::DoProperty(IFXJS_Context* cc, |
| 139 | const FX_WCHAR* propname, |
| 140 | CJS_PropValue& vp, |
| 141 | CFX_WideString& sError) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 142 | if (vp.IsSetting()) { |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 143 | CFX_ByteString sPropName = CFX_ByteString::FromUnicode(propname); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 144 | switch (vp.GetType()) { |
| 145 | case VT_number: { |
| 146 | double dData; |
| 147 | vp >> dData; |
| 148 | return SetGlobalVariables(sPropName, JS_GLOBALDATA_TYPE_NUMBER, dData, |
| 149 | false, "", v8::Local<v8::Object>(), FALSE); |
| 150 | } |
| 151 | case VT_boolean: { |
| 152 | bool bData; |
| 153 | vp >> bData; |
| 154 | return SetGlobalVariables(sPropName, JS_GLOBALDATA_TYPE_BOOLEAN, 0, |
| 155 | bData, "", v8::Local<v8::Object>(), FALSE); |
| 156 | } |
| 157 | case VT_string: { |
| 158 | CFX_ByteString sData; |
| 159 | vp >> sData; |
| 160 | return SetGlobalVariables(sPropName, JS_GLOBALDATA_TYPE_STRING, 0, |
| 161 | false, sData, v8::Local<v8::Object>(), FALSE); |
| 162 | } |
| 163 | case VT_object: { |
| 164 | JSObject pData; |
| 165 | vp >> pData; |
| 166 | return SetGlobalVariables(sPropName, JS_GLOBALDATA_TYPE_OBJECT, 0, |
| 167 | false, "", pData, FALSE); |
| 168 | } |
| 169 | case VT_null: { |
| 170 | return SetGlobalVariables(sPropName, JS_GLOBALDATA_TYPE_NULL, 0, false, |
| 171 | "", v8::Local<v8::Object>(), FALSE); |
| 172 | } |
| 173 | case VT_undefined: { |
| 174 | DelProperty(cc, propname, sError); |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 175 | return TRUE; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 176 | } |
| 177 | default: |
| 178 | break; |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 179 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 180 | } else { |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 181 | auto it = m_mapGlobal.find(CFX_ByteString::FromUnicode(propname)); |
| 182 | if (it == m_mapGlobal.end()) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 183 | vp.SetNull(); |
| 184 | return TRUE; |
| 185 | } |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 186 | JSGlobalData* pData = it->second; |
| 187 | if (pData->bDeleted) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 188 | vp.SetNull(); |
| 189 | return TRUE; |
| 190 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 191 | switch (pData->nType) { |
| 192 | case JS_GLOBALDATA_TYPE_NUMBER: |
| 193 | vp << pData->dData; |
| 194 | return TRUE; |
| 195 | case JS_GLOBALDATA_TYPE_BOOLEAN: |
| 196 | vp << pData->bData; |
| 197 | return TRUE; |
| 198 | case JS_GLOBALDATA_TYPE_STRING: |
| 199 | vp << pData->sData; |
| 200 | return TRUE; |
| 201 | case JS_GLOBALDATA_TYPE_OBJECT: { |
| 202 | v8::Local<v8::Object> obj = |
| 203 | v8::Local<v8::Object>::New(vp.GetIsolate(), pData->pData); |
| 204 | vp << obj; |
| 205 | return TRUE; |
| 206 | } |
| 207 | case JS_GLOBALDATA_TYPE_NULL: |
| 208 | vp.SetNull(); |
| 209 | return TRUE; |
| 210 | default: |
| 211 | break; |
| 212 | } |
| 213 | } |
| 214 | return FALSE; |
| 215 | } |
| 216 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 217 | FX_BOOL JSGlobalAlternate::setPersistent(IFXJS_Context* cc, |
| 218 | const CJS_Parameters& params, |
| 219 | CJS_Value& vRet, |
| 220 | CFX_WideString& sError) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 221 | CJS_Context* pContext = static_cast<CJS_Context*>(cc); |
| 222 | if (params.size() != 2) { |
| 223 | sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR); |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 224 | return FALSE; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 225 | } |
| 226 | |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 227 | auto it = m_mapGlobal.find(params[0].ToCFXByteString()); |
| 228 | if (it != m_mapGlobal.end()) { |
| 229 | JSGlobalData* pData = it->second; |
| 230 | if (!pData->bDeleted) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 231 | pData->bPersistent = params[1].ToBool(); |
| 232 | return TRUE; |
| 233 | } |
| 234 | } |
| 235 | |
| 236 | sError = JSGetStringFromID(pContext, IDS_STRING_JSNOGLOBAL); |
| 237 | return FALSE; |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 238 | } |
| 239 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 240 | void JSGlobalAlternate::UpdateGlobalPersistentVariables() { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 241 | ASSERT(m_pGlobalData != NULL); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 242 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 243 | for (int i = 0, sz = m_pGlobalData->GetSize(); i < sz; i++) { |
| 244 | CJS_GlobalData_Element* pData = m_pGlobalData->GetAt(i); |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 245 | ASSERT(pData != NULL); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 246 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 247 | switch (pData->data.nType) { |
| 248 | case JS_GLOBALDATA_TYPE_NUMBER: |
| 249 | SetGlobalVariables(pData->data.sKey, JS_GLOBALDATA_TYPE_NUMBER, |
| 250 | pData->data.dData, false, "", |
| 251 | v8::Local<v8::Object>(), pData->bPersistent == 1); |
| 252 | JS_PutObjectNumber(NULL, (JSFXObject)(*m_pJSObject), |
| 253 | pData->data.sKey.UTF8Decode().c_str(), |
| 254 | pData->data.dData); |
| 255 | break; |
| 256 | case JS_GLOBALDATA_TYPE_BOOLEAN: |
| 257 | SetGlobalVariables(pData->data.sKey, JS_GLOBALDATA_TYPE_BOOLEAN, 0, |
| 258 | (bool)(pData->data.bData == 1), "", |
| 259 | v8::Local<v8::Object>(), pData->bPersistent == 1); |
| 260 | JS_PutObjectBoolean(NULL, (JSFXObject)(*m_pJSObject), |
| 261 | pData->data.sKey.UTF8Decode().c_str(), |
| 262 | (bool)(pData->data.bData == 1)); |
| 263 | break; |
| 264 | case JS_GLOBALDATA_TYPE_STRING: |
| 265 | SetGlobalVariables(pData->data.sKey, JS_GLOBALDATA_TYPE_STRING, 0, |
| 266 | false, pData->data.sData, v8::Local<v8::Object>(), |
| 267 | pData->bPersistent == 1); |
| 268 | JS_PutObjectString(NULL, (JSFXObject)(*m_pJSObject), |
| 269 | pData->data.sKey.UTF8Decode().c_str(), |
| 270 | pData->data.sData.UTF8Decode().c_str()); |
| 271 | break; |
| 272 | case JS_GLOBALDATA_TYPE_OBJECT: { |
| 273 | IJS_Runtime* pRuntime = JS_GetRuntime((JSFXObject)(*m_pJSObject)); |
| 274 | v8::Local<v8::Object> pObj = JS_NewFxDynamicObj(pRuntime, NULL, -1); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 275 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 276 | PutObjectProperty(pObj, &pData->data); |
| 277 | |
| 278 | SetGlobalVariables(pData->data.sKey, JS_GLOBALDATA_TYPE_OBJECT, 0, |
| 279 | false, "", (JSObject)pObj, pData->bPersistent == 1); |
| 280 | JS_PutObjectObject(NULL, (JSFXObject)(*m_pJSObject), |
| 281 | pData->data.sKey.UTF8Decode().c_str(), |
| 282 | (JSObject)pObj); |
| 283 | } break; |
| 284 | case JS_GLOBALDATA_TYPE_NULL: |
| 285 | SetGlobalVariables(pData->data.sKey, JS_GLOBALDATA_TYPE_NULL, 0, false, |
| 286 | "", v8::Local<v8::Object>(), |
| 287 | pData->bPersistent == 1); |
| 288 | JS_PutObjectNull(NULL, (JSFXObject)(*m_pJSObject), |
| 289 | pData->data.sKey.UTF8Decode().c_str()); |
| 290 | break; |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 291 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 292 | } |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 293 | } |
| 294 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 295 | void JSGlobalAlternate::CommitGlobalPersisitentVariables() { |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 296 | ASSERT(m_pGlobalData); |
| 297 | for (auto it = m_mapGlobal.begin(); it != m_mapGlobal.end(); ++it) { |
| 298 | CFX_ByteString name = it->first; |
| 299 | JSGlobalData* pData = it->second; |
| 300 | if (pData->bDeleted) { |
| 301 | m_pGlobalData->DeleteGlobalVariable(name); |
| 302 | } else { |
| 303 | switch (pData->nType) { |
| 304 | case JS_GLOBALDATA_TYPE_NUMBER: |
| 305 | m_pGlobalData->SetGlobalVariableNumber(name, pData->dData); |
| 306 | m_pGlobalData->SetGlobalVariablePersistent(name, pData->bPersistent); |
| 307 | break; |
| 308 | case JS_GLOBALDATA_TYPE_BOOLEAN: |
| 309 | m_pGlobalData->SetGlobalVariableBoolean(name, pData->bData); |
| 310 | m_pGlobalData->SetGlobalVariablePersistent(name, pData->bPersistent); |
| 311 | break; |
| 312 | case JS_GLOBALDATA_TYPE_STRING: |
| 313 | m_pGlobalData->SetGlobalVariableString(name, pData->sData); |
| 314 | m_pGlobalData->SetGlobalVariablePersistent(name, pData->bPersistent); |
| 315 | break; |
| 316 | case JS_GLOBALDATA_TYPE_OBJECT: |
| 317 | // if (pData->pData) |
| 318 | { |
| 319 | CJS_GlobalVariableArray array; |
| 320 | v8::Local<v8::Object> obj = v8::Local<v8::Object>::New( |
| 321 | GetJSObject()->GetIsolate(), pData->pData); |
| 322 | ObjectToArray(obj, array); |
| 323 | m_pGlobalData->SetGlobalVariableObject(name, array); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 324 | m_pGlobalData->SetGlobalVariablePersistent(name, |
| 325 | pData->bPersistent); |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 326 | } |
| 327 | break; |
| 328 | case JS_GLOBALDATA_TYPE_NULL: |
| 329 | m_pGlobalData->SetGlobalVariableNull(name); |
| 330 | m_pGlobalData->SetGlobalVariablePersistent(name, pData->bPersistent); |
| 331 | break; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 332 | } |
| 333 | } |
| 334 | } |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 335 | } |
| 336 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 337 | void JSGlobalAlternate::ObjectToArray(v8::Local<v8::Object> pObj, |
| 338 | CJS_GlobalVariableArray& array) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 339 | v8::Local<v8::Context> context = pObj->CreationContext(); |
| 340 | v8::Isolate* isolate = context->GetIsolate(); |
| 341 | v8::Local<v8::Array> pKeyList = JS_GetObjectElementNames(isolate, pObj); |
| 342 | int nObjElements = pKeyList->Length(); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 343 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 344 | for (int i = 0; i < nObjElements; i++) { |
| 345 | CFX_WideString ws = |
| 346 | JS_ToString(isolate, JS_GetArrayElement(isolate, pKeyList, i)); |
| 347 | CFX_ByteString sKey = ws.UTF8Encode(); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 348 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 349 | v8::Local<v8::Value> v = JS_GetObjectElement(isolate, pObj, ws.c_str()); |
| 350 | FXJSVALUETYPE vt = GET_VALUE_TYPE(v); |
| 351 | switch (vt) { |
| 352 | case VT_number: { |
| 353 | CJS_KeyValue* pObjElement = new CJS_KeyValue; |
| 354 | pObjElement->nType = JS_GLOBALDATA_TYPE_NUMBER; |
| 355 | pObjElement->sKey = sKey; |
| 356 | pObjElement->dData = JS_ToNumber(isolate, v); |
| 357 | array.Add(pObjElement); |
| 358 | } break; |
| 359 | case VT_boolean: { |
| 360 | CJS_KeyValue* pObjElement = new CJS_KeyValue; |
| 361 | pObjElement->nType = JS_GLOBALDATA_TYPE_BOOLEAN; |
| 362 | pObjElement->sKey = sKey; |
| 363 | pObjElement->dData = JS_ToBoolean(isolate, v); |
| 364 | array.Add(pObjElement); |
| 365 | } break; |
| 366 | case VT_string: { |
| 367 | CFX_ByteString sValue = |
| 368 | CJS_Value(isolate, v, VT_string).ToCFXByteString(); |
| 369 | CJS_KeyValue* pObjElement = new CJS_KeyValue; |
| 370 | pObjElement->nType = JS_GLOBALDATA_TYPE_STRING; |
| 371 | pObjElement->sKey = sKey; |
| 372 | pObjElement->sData = sValue; |
| 373 | array.Add(pObjElement); |
| 374 | } break; |
| 375 | case VT_object: { |
| 376 | CJS_KeyValue* pObjElement = new CJS_KeyValue; |
| 377 | pObjElement->nType = JS_GLOBALDATA_TYPE_OBJECT; |
| 378 | pObjElement->sKey = sKey; |
| 379 | ObjectToArray(JS_ToObject(isolate, v), pObjElement->objData); |
| 380 | array.Add(pObjElement); |
| 381 | } break; |
| 382 | case VT_null: { |
| 383 | CJS_KeyValue* pObjElement = new CJS_KeyValue; |
| 384 | pObjElement->nType = JS_GLOBALDATA_TYPE_NULL; |
| 385 | pObjElement->sKey = sKey; |
| 386 | array.Add(pObjElement); |
| 387 | } break; |
| 388 | default: |
| 389 | break; |
| 390 | } |
| 391 | } |
| 392 | } |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 393 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 394 | void JSGlobalAlternate::PutObjectProperty(v8::Local<v8::Object> pObj, |
| 395 | CJS_KeyValue* pData) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 396 | ASSERT(pData != NULL); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 397 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 398 | for (int i = 0, sz = pData->objData.Count(); i < sz; i++) { |
| 399 | CJS_KeyValue* pObjData = pData->objData.GetAt(i); |
| 400 | ASSERT(pObjData != NULL); |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 401 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 402 | switch (pObjData->nType) { |
| 403 | case JS_GLOBALDATA_TYPE_NUMBER: |
| 404 | JS_PutObjectNumber(NULL, (JSObject)pObj, |
| 405 | pObjData->sKey.UTF8Decode().c_str(), |
| 406 | pObjData->dData); |
| 407 | break; |
| 408 | case JS_GLOBALDATA_TYPE_BOOLEAN: |
| 409 | JS_PutObjectBoolean(NULL, (JSObject)pObj, |
| 410 | pObjData->sKey.UTF8Decode().c_str(), |
| 411 | (bool)(pObjData->bData == 1)); |
| 412 | break; |
| 413 | case JS_GLOBALDATA_TYPE_STRING: |
| 414 | JS_PutObjectString(NULL, (JSObject)pObj, |
| 415 | pObjData->sKey.UTF8Decode().c_str(), |
| 416 | pObjData->sData.UTF8Decode().c_str()); |
| 417 | break; |
| 418 | case JS_GLOBALDATA_TYPE_OBJECT: { |
| 419 | IJS_Runtime* pRuntime = JS_GetRuntime((JSFXObject)(*m_pJSObject)); |
| 420 | v8::Local<v8::Object> pNewObj = JS_NewFxDynamicObj(pRuntime, NULL, -1); |
| 421 | PutObjectProperty(pNewObj, pObjData); |
| 422 | JS_PutObjectObject(NULL, (JSObject)pObj, |
| 423 | pObjData->sKey.UTF8Decode().c_str(), |
| 424 | (JSObject)pNewObj); |
| 425 | } break; |
| 426 | case JS_GLOBALDATA_TYPE_NULL: |
| 427 | JS_PutObjectNull(NULL, (JSObject)pObj, |
| 428 | pObjData->sKey.UTF8Decode().c_str()); |
| 429 | break; |
| 430 | } |
| 431 | } |
| 432 | } |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 433 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 434 | void JSGlobalAlternate::DestroyGlobalPersisitentVariables() { |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 435 | for (const auto& pair : m_mapGlobal) { |
| 436 | delete pair.second; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 437 | } |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 438 | m_mapGlobal.clear(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 439 | } |
| 440 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 441 | FX_BOOL JSGlobalAlternate::SetGlobalVariables(const FX_CHAR* propname, |
| 442 | int nType, |
| 443 | double dData, |
| 444 | bool bData, |
| 445 | const CFX_ByteString& sData, |
| 446 | JSObject pData, |
| 447 | bool bDefaultPersistent) { |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 448 | if (!propname) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 449 | return FALSE; |
| 450 | |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 451 | auto it = m_mapGlobal.find(propname); |
| 452 | if (it != m_mapGlobal.end()) { |
| 453 | JSGlobalData* pTemp = it->second; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 454 | if (pTemp->bDeleted || pTemp->nType != nType) { |
| 455 | pTemp->dData = 0; |
| 456 | pTemp->bData = 0; |
| 457 | pTemp->sData = ""; |
| 458 | pTemp->nType = nType; |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 459 | } |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 460 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 461 | pTemp->bDeleted = FALSE; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 462 | switch (nType) { |
| 463 | case JS_GLOBALDATA_TYPE_NUMBER: { |
| 464 | pTemp->dData = dData; |
| 465 | } break; |
| 466 | case JS_GLOBALDATA_TYPE_BOOLEAN: { |
| 467 | pTemp->bData = bData; |
| 468 | } break; |
| 469 | case JS_GLOBALDATA_TYPE_STRING: { |
| 470 | pTemp->sData = sData; |
| 471 | } break; |
| 472 | case JS_GLOBALDATA_TYPE_OBJECT: { |
| 473 | pTemp->pData.Reset(JS_GetRuntime(pData), pData); |
| 474 | } break; |
| 475 | case JS_GLOBALDATA_TYPE_NULL: |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 476 | break; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 477 | default: |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 478 | return FALSE; |
| 479 | } |
Tom Sepez | 2f2ffec | 2015-07-23 14:42:09 -0700 | [diff] [blame] | 480 | return TRUE; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 481 | } |
| 482 | |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 483 | JSGlobalData* pNewData = NULL; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 484 | |
| 485 | switch (nType) { |
| 486 | case JS_GLOBALDATA_TYPE_NUMBER: { |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 487 | pNewData = new JSGlobalData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 488 | pNewData->nType = JS_GLOBALDATA_TYPE_NUMBER; |
| 489 | pNewData->dData = dData; |
| 490 | pNewData->bPersistent = bDefaultPersistent; |
| 491 | } break; |
| 492 | case JS_GLOBALDATA_TYPE_BOOLEAN: { |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 493 | pNewData = new JSGlobalData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 494 | pNewData->nType = JS_GLOBALDATA_TYPE_BOOLEAN; |
| 495 | pNewData->bData = bData; |
| 496 | pNewData->bPersistent = bDefaultPersistent; |
| 497 | } break; |
| 498 | case JS_GLOBALDATA_TYPE_STRING: { |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 499 | pNewData = new JSGlobalData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 500 | pNewData->nType = JS_GLOBALDATA_TYPE_STRING; |
| 501 | pNewData->sData = sData; |
| 502 | pNewData->bPersistent = bDefaultPersistent; |
| 503 | } break; |
| 504 | case JS_GLOBALDATA_TYPE_OBJECT: { |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 505 | pNewData = new JSGlobalData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 506 | pNewData->nType = JS_GLOBALDATA_TYPE_OBJECT; |
| 507 | pNewData->pData.Reset(JS_GetRuntime(pData), pData); |
| 508 | pNewData->bPersistent = bDefaultPersistent; |
| 509 | } break; |
| 510 | case JS_GLOBALDATA_TYPE_NULL: { |
Tom Sepez | 7dc5cc1 | 2015-08-17 12:15:26 -0700 | [diff] [blame] | 511 | pNewData = new JSGlobalData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 512 | pNewData->nType = JS_GLOBALDATA_TYPE_NULL; |
| 513 | pNewData->bPersistent = bDefaultPersistent; |
| 514 | } break; |
| 515 | default: |
| 516 | return FALSE; |
| 517 | } |
| 518 | |
Tom Sepez | 09d33bc | 2015-08-19 09:49:24 -0700 | [diff] [blame] | 519 | m_mapGlobal[propname] = pNewData; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 520 | return TRUE; |
John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 521 | } |
Tom Sepez | ccc9483 | 2015-02-17 13:30:23 -0800 | [diff] [blame] | 522 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 523 | FXJSVALUETYPE GET_VALUE_TYPE(v8::Local<v8::Value> p) { |
Tom Sepez | ccc9483 | 2015-02-17 13:30:23 -0800 | [diff] [blame] | 524 | const unsigned int nHash = JS_CalcHash(JS_GetTypeof(p)); |
| 525 | |
| 526 | if (nHash == JSCONST_nUndefHash) |
| 527 | return VT_undefined; |
| 528 | if (nHash == JSCONST_nNullHash) |
| 529 | return VT_null; |
| 530 | if (nHash == JSCONST_nStringHash) |
| 531 | return VT_string; |
| 532 | if (nHash == JSCONST_nNumberHash) |
| 533 | return VT_number; |
| 534 | if (nHash == JSCONST_nBoolHash) |
| 535 | return VT_boolean; |
| 536 | if (nHash == JSCONST_nDateHash) |
| 537 | return VT_date; |
| 538 | if (nHash == JSCONST_nObjectHash) |
| 539 | return VT_object; |
| 540 | if (nHash == JSCONST_nFXobjHash) |
| 541 | return VT_fxobject; |
| 542 | |
| 543 | return VT_unknown; |
| 544 | } |