mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-15 07:21:07 +00:00
379 lines
12 KiB
C++
379 lines
12 KiB
C++
|
/****************************************************************************
|
||
|
Copyright (c) 2016 Chukong Technologies Inc.
|
||
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
||
|
|
||
|
http://www.cocos2d-x.org
|
||
|
|
||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
of this software and associated documentation files (the "Software"), to deal
|
||
|
in the Software without restriction, including without limitation the rights
|
||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
|
copies of the Software, and to permit persons to whom the Software is
|
||
|
furnished to do so, subject to the following conditions:
|
||
|
|
||
|
The above copyright notice and this permission notice shall be included in
|
||
|
all copies or substantial portions of the Software.
|
||
|
|
||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
|
THE SOFTWARE.
|
||
|
****************************************************************************/
|
||
|
#include "Utils.hpp"
|
||
|
|
||
|
#if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_JSC
|
||
|
|
||
|
#include "Object.hpp"
|
||
|
#include "ScriptEngine.hpp"
|
||
|
#include "../HandleObject.hpp"
|
||
|
|
||
|
namespace se {
|
||
|
|
||
|
namespace internal {
|
||
|
|
||
|
namespace {
|
||
|
JSContextRef __cx = nullptr;
|
||
|
}
|
||
|
|
||
|
void setContext(JSContextRef cx)
|
||
|
{
|
||
|
__cx = cx;
|
||
|
}
|
||
|
|
||
|
bool defineProperty(Object* obj, const char* name, JSObjectCallAsFunctionCallback jsGetter, JSObjectCallAsFunctionCallback jsSetter)
|
||
|
{
|
||
|
bool ok = true;
|
||
|
Object* globalObject = ScriptEngine::getInstance()->getGlobalObject();
|
||
|
Value v;
|
||
|
ok = globalObject->getProperty("Object", &v);
|
||
|
if (!ok || !v.isObject())
|
||
|
{
|
||
|
SE_LOGD("ERROR: couldn't find Object\n");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
Value definePropertyFunc;
|
||
|
ok = v.toObject()->getProperty("defineProperty", &definePropertyFunc);
|
||
|
if (!ok || !v.isObject())
|
||
|
{
|
||
|
SE_LOGD("ERROR: couldn't find Object.defineProperty\n");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
ValueArray args;
|
||
|
args.reserve(3);
|
||
|
args.push_back(Value(obj));
|
||
|
args.push_back(Value(name));
|
||
|
|
||
|
HandleObject desc(Object::createPlainObject());
|
||
|
|
||
|
JSObjectRef getter = nullptr;
|
||
|
if (jsGetter != nullptr)
|
||
|
{
|
||
|
getter = JSObjectMakeFunctionWithCallback(__cx, nullptr, jsGetter);
|
||
|
}
|
||
|
JSObjectRef setter = nullptr;
|
||
|
|
||
|
if (jsSetter != nullptr)
|
||
|
{
|
||
|
setter = JSObjectMakeFunctionWithCallback(__cx, nullptr, jsSetter);
|
||
|
}
|
||
|
|
||
|
assert(getter != nullptr || setter != nullptr);
|
||
|
|
||
|
if (getter != nullptr)
|
||
|
{
|
||
|
HandleObject getterObj(Object::_createJSObject(nullptr, getter));
|
||
|
desc->setProperty("get", Value(getterObj));
|
||
|
}
|
||
|
|
||
|
if (setter != nullptr)
|
||
|
{
|
||
|
HandleObject setterObj(Object::_createJSObject(nullptr, setter));
|
||
|
desc->setProperty("set", Value(setterObj));
|
||
|
}
|
||
|
|
||
|
args.push_back(Value(desc));
|
||
|
definePropertyFunc.toObject()->call(args, nullptr);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void jsToSeArgs(JSContextRef cx, unsigned short argc, const JSValueRef* argv, ValueArray* outArr)
|
||
|
{
|
||
|
outArr->reserve(argc);
|
||
|
for (unsigned short i = 0; i < argc; ++i)
|
||
|
{
|
||
|
Value v;
|
||
|
jsToSeValue(cx, argv[i], &v);
|
||
|
outArr->push_back(v);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void seToJsArgs(JSContextRef cx, const ValueArray& args, JSValueRef* outArr)
|
||
|
{
|
||
|
for (size_t i = 0, len = args.size(); i < len; ++i)
|
||
|
{
|
||
|
seToJsValue(cx, args[i], &outArr[i]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void jsToSeValue(JSContextRef cx, JSValueRef jsValue, Value* data)
|
||
|
{
|
||
|
if (JSValueIsNull(cx, jsValue))
|
||
|
{
|
||
|
data->setNull();
|
||
|
}
|
||
|
else if (JSValueIsUndefined(cx, jsValue))
|
||
|
{
|
||
|
data->setUndefined();
|
||
|
}
|
||
|
else if (JSValueIsNumber(cx, jsValue))
|
||
|
{
|
||
|
JSValueRef exception = nullptr;
|
||
|
double number = JSValueToNumber(cx, jsValue, &exception);
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
data->setNumber(number);
|
||
|
}
|
||
|
}
|
||
|
else if (JSValueIsBoolean(cx, jsValue))
|
||
|
{
|
||
|
data->setBoolean(JSValueToBoolean(cx, jsValue));
|
||
|
}
|
||
|
else if (JSValueIsString(cx, jsValue))
|
||
|
{
|
||
|
std::string str;
|
||
|
forceConvertJsValueToStdString(cx, jsValue, &str);
|
||
|
data->setString(str);
|
||
|
}
|
||
|
else if (JSValueIsObject(cx, jsValue))
|
||
|
{
|
||
|
JSValueRef exception = nullptr;
|
||
|
JSObjectRef jsobj = JSValueToObject(cx, jsValue, &exception);
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
void* nativePtr = internal::getPrivate(jsobj);
|
||
|
Object* obj = nullptr;
|
||
|
if (nativePtr != nullptr)
|
||
|
{
|
||
|
obj = Object::getObjectWithPtr(nativePtr);
|
||
|
}
|
||
|
|
||
|
if (obj == nullptr)
|
||
|
{
|
||
|
obj = Object::_createJSObject(nullptr, jsobj);
|
||
|
}
|
||
|
data->setObject(obj, true);
|
||
|
obj->decRef();
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
assert(false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void seToJsValue(JSContextRef cx, const Value& v, JSValueRef* jsval)
|
||
|
{
|
||
|
switch (v.getType()) {
|
||
|
case Value::Type::Null:
|
||
|
*jsval = JSValueMakeNull(cx);
|
||
|
break;
|
||
|
|
||
|
case Value::Type::Number:
|
||
|
*jsval = JSValueMakeNumber(cx, v.toNumber());
|
||
|
break;
|
||
|
|
||
|
case Value::Type::String:
|
||
|
{
|
||
|
JSStringRef str = JSStringCreateWithUTF8CString(v.toString().c_str());
|
||
|
*jsval = JSValueMakeString(cx, str);
|
||
|
JSStringRelease(str);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case Value::Type::Boolean:
|
||
|
*jsval = JSValueMakeBoolean(cx, v.toBoolean());
|
||
|
break;
|
||
|
|
||
|
case Value::Type::Object:
|
||
|
*jsval = v.toObject()->_getJSObject();
|
||
|
break;
|
||
|
|
||
|
default: // Undefined
|
||
|
*jsval = JSValueMakeUndefined(cx);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void forceConvertJsValueToStdString(JSContextRef cx, JSValueRef jsval, std::string* ret, bool ignoreException/* = false */)
|
||
|
{
|
||
|
JSValueRef exception = nullptr;
|
||
|
JSStringRef jsstr = JSValueToStringCopy(cx, jsval, &exception);
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
// NOTE: ignoreException is true in ScriptEngine::_formatException because it's already in the _clearException process.
|
||
|
// Adds this flag to avoid infinite loop of _clearException -> _formatException -> forceConvertJsValueToStdString -> _clearException -> ...
|
||
|
if (!ignoreException)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
ret->clear();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
jsStringToStdString(cx, jsstr, ret);
|
||
|
}
|
||
|
|
||
|
if (jsstr != nullptr)
|
||
|
JSStringRelease(jsstr);
|
||
|
}
|
||
|
|
||
|
void jsStringToStdString(JSContextRef cx, JSStringRef jsStr, std::string* ret)
|
||
|
{
|
||
|
size_t len = JSStringGetLength(jsStr);
|
||
|
const size_t BUF_SIZE = len * 4 + 1;
|
||
|
char* buf = (char*)malloc(BUF_SIZE);
|
||
|
memset(buf, 0, BUF_SIZE);
|
||
|
JSStringGetUTF8CString(jsStr, buf, BUF_SIZE);
|
||
|
*ret = buf;
|
||
|
free(buf);
|
||
|
}
|
||
|
|
||
|
const char* KEY_PRIVATE_DATA = "__cc_private_data";
|
||
|
|
||
|
bool hasPrivate(JSObjectRef obj)
|
||
|
{
|
||
|
void* data = JSObjectGetPrivate(obj);
|
||
|
if (data != nullptr)
|
||
|
return true;
|
||
|
|
||
|
JSStringRef key = JSStringCreateWithUTF8CString(KEY_PRIVATE_DATA);
|
||
|
bool found = JSObjectHasProperty(__cx, obj, key);
|
||
|
JSStringRelease(key);
|
||
|
|
||
|
return found;
|
||
|
}
|
||
|
|
||
|
void setPrivate(JSObjectRef obj, void* data, JSObjectFinalizeCallback finalizeCb)
|
||
|
{
|
||
|
bool ok = JSObjectSetPrivate(obj, data);
|
||
|
if (ok)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
assert(finalizeCb);
|
||
|
Object* privateObj = Object::createObjectWithClass(__jsb_CCPrivateData_class);
|
||
|
privateObj->root();
|
||
|
|
||
|
internal::PrivateData* privateData = (internal::PrivateData*)malloc(sizeof(internal::PrivateData));
|
||
|
privateData->data = data;
|
||
|
privateData->finalizeCb = finalizeCb;
|
||
|
ok = JSObjectSetPrivate(privateObj->_getJSObject(), privateData);
|
||
|
assert(ok);
|
||
|
|
||
|
JSStringRef key = JSStringCreateWithUTF8CString(KEY_PRIVATE_DATA);
|
||
|
JSValueRef exception = nullptr;
|
||
|
JSObjectSetProperty(__cx, obj, key, privateObj->_getJSObject(), kJSPropertyAttributeDontEnum, &exception);
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
JSStringRelease(key);
|
||
|
|
||
|
privateObj->unroot();
|
||
|
privateObj->decRef();
|
||
|
}
|
||
|
|
||
|
void* getPrivate(JSObjectRef obj)
|
||
|
{
|
||
|
void* data = JSObjectGetPrivate(obj);
|
||
|
if (data != nullptr)
|
||
|
return data;
|
||
|
|
||
|
JSStringRef key = JSStringCreateWithUTF8CString(KEY_PRIVATE_DATA);
|
||
|
bool found = JSObjectHasProperty(__cx, obj, key);
|
||
|
if (found)
|
||
|
{
|
||
|
JSValueRef exception = nullptr;
|
||
|
JSValueRef privateDataVal = JSObjectGetProperty(__cx, obj, key, &exception);
|
||
|
do
|
||
|
{
|
||
|
if (exception != nullptr)
|
||
|
break;
|
||
|
|
||
|
JSObjectRef jsobj = JSValueToObject(__cx, privateDataVal, &exception);
|
||
|
if (exception != nullptr)
|
||
|
break;
|
||
|
|
||
|
internal::PrivateData* privateData = (internal::PrivateData*)JSObjectGetPrivate(jsobj);
|
||
|
assert(privateData != nullptr);
|
||
|
data = privateData->data;
|
||
|
} while(false);
|
||
|
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
}
|
||
|
JSStringRelease(key);
|
||
|
return data;
|
||
|
}
|
||
|
|
||
|
void clearPrivate(JSObjectRef obj)
|
||
|
{
|
||
|
void* data = JSObjectGetPrivate(obj);
|
||
|
if (data != nullptr)
|
||
|
{
|
||
|
JSObjectSetPrivate(obj, nullptr);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
JSStringRef key = JSStringCreateWithUTF8CString(KEY_PRIVATE_DATA); //IDEA: cache the key string
|
||
|
if (JSObjectHasProperty(__cx, obj, key))
|
||
|
{
|
||
|
JSValueRef exception = nullptr;
|
||
|
do
|
||
|
{
|
||
|
JSValueRef value = JSObjectGetProperty(__cx, obj, key, &exception);
|
||
|
if (exception != nullptr)
|
||
|
break;
|
||
|
|
||
|
JSObjectRef propertyObj = JSValueToObject(__cx, value, &exception);
|
||
|
if (exception != nullptr)
|
||
|
break;
|
||
|
|
||
|
internal::PrivateData* privateData = (internal::PrivateData*)JSObjectGetPrivate(propertyObj);
|
||
|
free(privateData);
|
||
|
JSObjectSetPrivate(propertyObj, nullptr);
|
||
|
bool ok = JSObjectDeleteProperty(__cx, obj, key, nullptr);
|
||
|
assert(ok);
|
||
|
} while (false);
|
||
|
|
||
|
if (exception != nullptr)
|
||
|
{
|
||
|
ScriptEngine::getInstance()->_clearException(exception);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
JSStringRelease(key);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}} // namespace se { namespace internal {
|
||
|
|
||
|
#endif // #if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_JSC
|