mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-15 07:21:07 +00:00
238 lines
6.8 KiB
C++
238 lines
6.8 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 "Class.hpp"
|
||
|
#include "Object.hpp"
|
||
|
#include "Utils.hpp"
|
||
|
#include "ScriptEngine.hpp"
|
||
|
|
||
|
#if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|
||
|
|
||
|
namespace se {
|
||
|
|
||
|
// --- Global Lookup for Constructor Functions
|
||
|
|
||
|
namespace {
|
||
|
// std::unordered_map<std::string, Class *> __clsMap;
|
||
|
JSContext* __cx = nullptr;
|
||
|
|
||
|
bool empty_constructor(JSContext* cx, uint32_t argc, JS::Value* vp)
|
||
|
{
|
||
|
assert(false);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
std::vector<Class*> __allClasses;
|
||
|
}
|
||
|
|
||
|
Class::Class()
|
||
|
: _parent(nullptr)
|
||
|
, _proto(nullptr)
|
||
|
, _parentProto(nullptr)
|
||
|
, _ctor(nullptr)
|
||
|
, _finalizeOp(nullptr)
|
||
|
{
|
||
|
memset(&_jsCls, 0, sizeof(_jsCls));
|
||
|
memset(&_classOps, 0, sizeof(_classOps));
|
||
|
|
||
|
__allClasses.push_back(this);
|
||
|
}
|
||
|
|
||
|
Class::~Class()
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
Class* Class::create(const char* className, Object* obj, Object* parentProto, JSNative ctor)
|
||
|
{
|
||
|
Class* cls = new Class();
|
||
|
if (cls != nullptr && !cls->init(className, obj, parentProto, ctor))
|
||
|
{
|
||
|
delete cls;
|
||
|
cls = nullptr;
|
||
|
}
|
||
|
return cls;
|
||
|
}
|
||
|
|
||
|
bool Class::init(const char* clsName, Object* parent, Object *parentProto, JSNative ctor)
|
||
|
{
|
||
|
_name = clsName;
|
||
|
|
||
|
_parent = parent;
|
||
|
if (_parent != nullptr)
|
||
|
_parent->incRef();
|
||
|
|
||
|
_parentProto = parentProto;
|
||
|
if (_parentProto != nullptr)
|
||
|
_parentProto->incRef();
|
||
|
|
||
|
_ctor = ctor;
|
||
|
if (_ctor == nullptr)
|
||
|
{
|
||
|
_ctor = empty_constructor;
|
||
|
}
|
||
|
|
||
|
// SE_LOGD("Class init ( %s ) ...\n", clsName);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void Class::destroy()
|
||
|
{
|
||
|
SAFE_DEC_REF(_parent);
|
||
|
SAFE_DEC_REF(_proto);
|
||
|
SAFE_DEC_REF(_parentProto);
|
||
|
}
|
||
|
|
||
|
bool Class::install()
|
||
|
{
|
||
|
// assert(__clsMap.find(_name) == __clsMap.end());
|
||
|
//
|
||
|
// __clsMap.emplace(_name, this);
|
||
|
|
||
|
_jsCls.name = _name;
|
||
|
if (_finalizeOp != nullptr)
|
||
|
{
|
||
|
_jsCls.flags = JSCLASS_HAS_PRIVATE | JSCLASS_FOREGROUND_FINALIZE; //IDEA: Use JSCLASS_BACKGROUND_FINALIZE to improve GC performance
|
||
|
_classOps.finalize = _finalizeOp;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_jsCls.flags = JSCLASS_HAS_PRIVATE;
|
||
|
}
|
||
|
|
||
|
_jsCls.cOps = &_classOps;
|
||
|
|
||
|
JSObject* parentObj = _parentProto != nullptr ? _parentProto->_getJSObject() : nullptr;
|
||
|
JS::RootedObject parent(__cx, _parent->_getJSObject());
|
||
|
JS::RootedObject parentProto(__cx, parentObj);
|
||
|
|
||
|
_funcs.push_back(JS_FS_END);
|
||
|
_properties.push_back(JS_PS_END);
|
||
|
_staticFuncs.push_back(JS_FS_END);
|
||
|
_staticProperties.push_back(JS_PS_END);
|
||
|
|
||
|
JSObject* jsobj = JS_InitClass(__cx, parent, parentProto, &_jsCls, _ctor, 0, _properties.data(), _funcs.data(), _staticProperties.data(), _staticFuncs.data());
|
||
|
if (jsobj != nullptr)
|
||
|
{
|
||
|
_proto = Object::_createJSObject(nullptr, jsobj);
|
||
|
// SE_LOGD("_proto: %p, name: %s\n", _proto, _name);
|
||
|
_proto->root();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool Class::defineFunction(const char *name, JSNative func)
|
||
|
{
|
||
|
JSFunctionSpec cb = JS_FN(name, func, 0, JSPROP_PERMANENT | JSPROP_ENUMERATE);
|
||
|
_funcs.push_back(cb);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Class::defineProperty(const char *name, JSNative getter, JSNative setter)
|
||
|
{
|
||
|
JSPropertySpec property = JS_PSGS(name, getter, setter, JSPROP_ENUMERATE | JSPROP_PERMANENT);
|
||
|
_properties.push_back(property);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Class::defineStaticFunction(const char *name, JSNative func)
|
||
|
{
|
||
|
JSFunctionSpec cb = JS_FN(name, func, 0, JSPROP_PERMANENT | JSPROP_ENUMERATE);
|
||
|
_staticFuncs.push_back(cb);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Class::defineStaticProperty(const char *name, JSNative getter, JSNative setter)
|
||
|
{
|
||
|
JSPropertySpec property = JS_PSGS(name, getter, setter, JSPROP_ENUMERATE | JSPROP_PERMANENT);
|
||
|
_staticProperties.push_back(property);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Class::defineFinalizeFunction(JSFinalizeOp func)
|
||
|
{
|
||
|
_finalizeOp = func;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// JSObject* Class::_createJSObject(const std::string &clsName, Class** outCls)
|
||
|
// {
|
||
|
// auto iter = __clsMap.find(clsName);
|
||
|
// if (iter == __clsMap.end())
|
||
|
// {
|
||
|
// *outCls = nullptr;
|
||
|
// return nullptr;
|
||
|
// }
|
||
|
//
|
||
|
// Class* thiz = iter->second;
|
||
|
// JS::RootedObject obj(__cx, _createJSObjectWithClass(thiz));
|
||
|
// *outCls = thiz;
|
||
|
// return obj;
|
||
|
// }
|
||
|
|
||
|
JSObject* Class::_createJSObjectWithClass(Class* cls)
|
||
|
{
|
||
|
JSObject* proto = cls->_proto != nullptr ? cls->_proto->_getJSObject() : nullptr;
|
||
|
JS::RootedObject jsProto(__cx, proto);
|
||
|
JS::RootedObject obj(__cx, JS_NewObjectWithGivenProto(__cx, &cls->_jsCls, jsProto));
|
||
|
return obj;
|
||
|
}
|
||
|
|
||
|
void Class::setContext(JSContext* cx)
|
||
|
{
|
||
|
__cx = cx;
|
||
|
}
|
||
|
|
||
|
Object *Class::getProto()
|
||
|
{
|
||
|
return _proto;
|
||
|
}
|
||
|
|
||
|
JSFinalizeOp Class::_getFinalizeCb() const
|
||
|
{
|
||
|
return _finalizeOp;
|
||
|
}
|
||
|
|
||
|
void Class::cleanup()
|
||
|
{
|
||
|
for (auto cls : __allClasses)
|
||
|
{
|
||
|
cls->destroy();
|
||
|
}
|
||
|
|
||
|
se::ScriptEngine::getInstance()->addAfterCleanupHook([](){
|
||
|
for (auto cls : __allClasses)
|
||
|
{
|
||
|
delete cls;
|
||
|
}
|
||
|
__allClasses.clear();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
} // namespace se {
|
||
|
|
||
|
#endif // #if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|