mirror of https://github.com/axmolengine/axmol.git
Merge pull request #14213 from pandamicro/v3
Improved JSCallbackWrapper with generational GC and fix the issue
This commit is contained in:
commit
2d75269a98
|
@ -878,49 +878,48 @@ jsval anonEvaluate(JSContext *cx, JS::HandleObject thisObj, const char* string)
|
||||||
}
|
}
|
||||||
|
|
||||||
JSCallbackWrapper::JSCallbackWrapper()
|
JSCallbackWrapper::JSCallbackWrapper()
|
||||||
: _jsCallback(JSVAL_VOID), _jsThisObj(JSVAL_VOID), _extraData(JSVAL_VOID)
|
|
||||||
{
|
{
|
||||||
|
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
|
_jsCallback.construct(cx, JS::NullHandleValue);
|
||||||
|
_jsThisObj.construct(cx, JS::NullHandleValue);
|
||||||
|
_extraData.construct(cx, JS::NullHandleValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
JSCallbackWrapper::~JSCallbackWrapper()
|
JSCallbackWrapper::~JSCallbackWrapper()
|
||||||
{
|
{
|
||||||
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
_jsCallback.destroyIfConstructed();
|
||||||
JS::RemoveValueRoot(cx, &_jsCallback);
|
_jsThisObj.destroyIfConstructed();
|
||||||
JS::RemoveValueRoot(cx, &_jsThisObj);
|
_extraData.destroyIfConstructed();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSCallbackWrapper::setJSCallbackFunc(jsval func) {
|
void JSCallbackWrapper::setJSCallbackFunc(JS::HandleValue func) {
|
||||||
_jsCallback = func;
|
if (!func.isNullOrUndefined())
|
||||||
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
_jsCallback.ref() = func;
|
||||||
// Root the callback function.
|
|
||||||
JS::AddNamedValueRoot(cx, &_jsCallback, "JSCallbackWrapper_callback_func");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSCallbackWrapper::setJSCallbackThis(jsval thisObj) {
|
void JSCallbackWrapper::setJSCallbackThis(JS::HandleValue thisObj) {
|
||||||
_jsThisObj = thisObj;
|
if (!thisObj.isNullOrUndefined())
|
||||||
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
_jsThisObj.ref() = thisObj;
|
||||||
// Root the this object.
|
|
||||||
JS::AddNamedValueRoot(cx, &_jsThisObj, "JSCallbackWrapper_callback_this");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSCallbackWrapper::setJSExtraData(jsval data) {
|
void JSCallbackWrapper::setJSExtraData(JS::HandleValue data) {
|
||||||
_extraData = data;
|
if (!data.isNullOrUndefined())
|
||||||
|
_extraData.ref() = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
const jsval& JSCallbackWrapper::getJSCallbackFunc() const
|
const jsval JSCallbackWrapper::getJSCallbackFunc() const
|
||||||
{
|
{
|
||||||
return _jsCallback.get();
|
return _jsCallback.ref().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
const jsval& JSCallbackWrapper::getJSCallbackThis() const
|
const jsval JSCallbackWrapper::getJSCallbackThis() const
|
||||||
{
|
{
|
||||||
return _jsThisObj.get();
|
return _jsThisObj.ref().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
const jsval& JSCallbackWrapper::getJSExtraData() const
|
const jsval JSCallbackWrapper::getJSExtraData() const
|
||||||
{
|
{
|
||||||
return _extraData.get();
|
return _extraData.ref().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
// cc.CallFunc.create( func, this, [data])
|
// cc.CallFunc.create( func, this, [data])
|
||||||
|
@ -932,82 +931,52 @@ static bool js_callFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
|
|
||||||
std::shared_ptr<JSCallbackWrapper> tmpCobj(new JSCallbackWrapper());
|
std::shared_ptr<JSCallbackWrapper> tmpCobj(new JSCallbackWrapper());
|
||||||
|
|
||||||
tmpCobj->setJSCallbackFunc(args.get(0));
|
JS::RootedValue callback(cx, args.get(0));
|
||||||
|
tmpCobj->setJSCallbackFunc(callback);
|
||||||
if(argc >= 2) {
|
if(argc >= 2) {
|
||||||
tmpCobj->setJSCallbackThis(args.get(1));
|
JS::RootedValue thisObj(cx, args.get(1));
|
||||||
} if(argc == 3) {
|
tmpCobj->setJSCallbackThis(thisObj);
|
||||||
tmpCobj->setJSExtraData(args.get(2));
|
}
|
||||||
|
if(argc >= 3) {
|
||||||
|
JS::RootedValue data(cx, args.get(2));
|
||||||
|
tmpCobj->setJSExtraData(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
CallFuncN *ret = CallFuncN::create([=](Node* sender){
|
CallFuncN *ret = CallFuncN::create([=](Node* sender){
|
||||||
// const jsval& jsvalThis = tmpCobj->getJSCallbackThis();
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
// const jsval& jsvalCallback = tmpCobj->getJSCallbackFunc();
|
|
||||||
// const jsval& jsvalExtraData = tmpCobj->getJSExtraData();
|
|
||||||
JS::RootedValue jsvalThis(cx, tmpCobj->getJSCallbackThis());
|
JS::RootedValue jsvalThis(cx, tmpCobj->getJSCallbackThis());
|
||||||
|
JS::RootedObject thisObj(cx, jsvalThis.toObjectOrNull());
|
||||||
JS::RootedValue jsvalCallback(cx, tmpCobj->getJSCallbackFunc());
|
JS::RootedValue jsvalCallback(cx, tmpCobj->getJSCallbackFunc());
|
||||||
JS::RootedValue jsvalExtraData(cx, tmpCobj->getJSExtraData());
|
JS::RootedValue jsvalExtraData(cx, tmpCobj->getJSExtraData());
|
||||||
|
|
||||||
bool hasExtraData = !jsvalExtraData.isUndefined();
|
JS::RootedValue senderVal(cx);
|
||||||
JS::RootedObject thisObj(cx, jsvalThis.toObjectOrNull());
|
if (sender)
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
|
||||||
|
|
||||||
if(sender)
|
|
||||||
{
|
{
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::Node>(cx, sender);
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::Node>(cx, sender);
|
||||||
|
senderVal.set(OBJECT_TO_JSVAL(proxy->obj));
|
||||||
JS::RootedValue retval(cx);
|
|
||||||
if(jsvalCallback != JSVAL_VOID)
|
|
||||||
{
|
|
||||||
if (hasExtraData)
|
|
||||||
{
|
|
||||||
jsval valArr[2];
|
|
||||||
valArr[0] = OBJECT_TO_JSVAL(proxy->obj);
|
|
||||||
valArr[1] = jsvalExtraData;
|
|
||||||
|
|
||||||
//TODO: really need root?
|
|
||||||
// JS_AddValueRoot(cx, valArr);
|
|
||||||
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
|
||||||
// JS_RemoveValueRoot(cx, valArr);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
jsval senderVal = OBJECT_TO_JSVAL(proxy->obj);
|
|
||||||
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(1, &senderVal);
|
|
||||||
// JS_AddValueRoot(cx, &senderVal);
|
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
|
||||||
|
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
|
||||||
// JS_RemoveValueRoot(cx, &senderVal);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
JS::RootedValue callRet(cx);
|
senderVal.set(JS::NullValue());
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, JS::HandleValueArray::empty(), &callRet);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// I think the JSCallFuncWrapper isn't needed.
|
if (!jsvalCallback.isNullOrUndefined())
|
||||||
// Since an action will be run by a cc.Node, it will be released at the Node::cleanup.
|
{
|
||||||
// By James Chen
|
JS::RootedValue retval(cx);
|
||||||
// JSCallFuncWrapper::setTargetForNativeNode(node, (JSCallFuncWrapper *)this);
|
|
||||||
|
jsval valArr[2];
|
||||||
|
valArr[0] = senderVal;
|
||||||
|
valArr[1] = jsvalExtraData;
|
||||||
|
|
||||||
|
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
||||||
|
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CallFunc>(cx, ret);
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CallFunc>(cx, ret);
|
||||||
args.rval().set(OBJECT_TO_JSVAL(proxy->obj));
|
args.rval().set(OBJECT_TO_JSVAL(proxy->obj));
|
||||||
|
|
||||||
JS_SetReservedSlot(proxy->obj, 0, args.get(0));
|
|
||||||
if(argc > 1) {
|
|
||||||
JS_SetReservedSlot(proxy->obj, 1, args.get(1));
|
|
||||||
}
|
|
||||||
// if(argc == 3) {
|
|
||||||
// JS_SetReservedSlot(proxy->obj, 2, args.get(2));
|
|
||||||
// }
|
|
||||||
|
|
||||||
// test->execute();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
JS_ReportError(cx, "Invalid number of arguments");
|
JS_ReportError(cx, "Invalid number of arguments");
|
||||||
|
@ -1028,87 +997,66 @@ bool js_cocos2dx_CallFunc_initWithFunction(JSContext *cx, uint32_t argc, jsval *
|
||||||
|
|
||||||
std::shared_ptr<JSCallbackWrapper> tmpCobj(new JSCallbackWrapper());
|
std::shared_ptr<JSCallbackWrapper> tmpCobj(new JSCallbackWrapper());
|
||||||
|
|
||||||
tmpCobj->setJSCallbackFunc(args.get(0));
|
JS::RootedValue callback(cx, args.get(0));
|
||||||
|
tmpCobj->setJSCallbackFunc(callback);
|
||||||
if(argc >= 2) {
|
if(argc >= 2) {
|
||||||
tmpCobj->setJSCallbackThis(args.get(1));
|
JS::RootedValue thisObj(cx, args.get(1));
|
||||||
} if(argc == 3) {
|
tmpCobj->setJSCallbackThis(thisObj);
|
||||||
tmpCobj->setJSExtraData(args.get(2));
|
}
|
||||||
|
if(argc >= 3) {
|
||||||
|
JS::RootedValue data(cx, args.get(2));
|
||||||
|
tmpCobj->setJSExtraData(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
action->initWithFunction([=](Node* sender){
|
action->initWithFunction([=](Node* sender){
|
||||||
// const jsval& jsvalThis = tmpCobj->getJSCallbackThis();
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
// const jsval& jsvalCallback = tmpCobj->getJSCallbackFunc();
|
|
||||||
// const jsval& jsvalExtraData = tmpCobj->getJSExtraData();
|
|
||||||
JS::RootedValue jsvalThis(cx, tmpCobj->getJSCallbackThis());
|
JS::RootedValue jsvalThis(cx, tmpCobj->getJSCallbackThis());
|
||||||
|
JS::RootedObject thisObj(cx, jsvalThis.toObjectOrNull());
|
||||||
JS::RootedValue jsvalCallback(cx, tmpCobj->getJSCallbackFunc());
|
JS::RootedValue jsvalCallback(cx, tmpCobj->getJSCallbackFunc());
|
||||||
JS::RootedValue jsvalExtraData(cx, tmpCobj->getJSExtraData());
|
JS::RootedValue jsvalExtraData(cx, tmpCobj->getJSExtraData());
|
||||||
|
|
||||||
bool hasExtraData = !jsvalExtraData.isUndefined();
|
JS::RootedValue senderVal(cx);
|
||||||
JS::RootedObject thisObj(cx, jsvalThis.toObjectOrNull());
|
if (sender)
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
|
||||||
|
|
||||||
if(sender)
|
|
||||||
{
|
{
|
||||||
js_proxy_t *senderProxy = js_get_or_create_proxy<cocos2d::Node>(cx, sender);
|
js_proxy_t *senderProxy = js_get_or_create_proxy<cocos2d::Node>(cx, sender);
|
||||||
|
senderVal.set(OBJECT_TO_JSVAL(senderProxy->obj));
|
||||||
JS::RootedValue retval(cx);
|
|
||||||
if(jsvalCallback != JSVAL_VOID)
|
|
||||||
{
|
|
||||||
if (hasExtraData)
|
|
||||||
{
|
|
||||||
jsval valArr[2];
|
|
||||||
valArr[0] = OBJECT_TO_JSVAL(senderProxy->obj);
|
|
||||||
valArr[1] = jsvalExtraData;
|
|
||||||
|
|
||||||
// JS_AddValueRoot(cx, valArr);
|
|
||||||
// JS_CallFunctionValue(cx, thisObj, jsvalCallback, 2, valArr, &retval);
|
|
||||||
// JS_RemoveValueRoot(cx, valArr);
|
|
||||||
|
|
||||||
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
jsval senderVal = OBJECT_TO_JSVAL(senderProxy->obj);
|
|
||||||
// JS_AddValueRoot(cx, &senderVal);
|
|
||||||
// JS_CallFunctionValue(cx, thisObj, jsvalCallback, 1, &senderVal, &retval);
|
|
||||||
// JS_RemoveValueRoot(cx, &senderVal);
|
|
||||||
|
|
||||||
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(1, &senderVal);
|
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
JS::RootedValue ret(cx);
|
senderVal.set(JS::NullValue());
|
||||||
JS_CallFunctionValue(cx, thisObj, jsvalCallback, JS::HandleValueArray::empty(), &ret);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!jsvalCallback.isNullOrUndefined())
|
||||||
|
{
|
||||||
|
JS::RootedValue retval(cx);
|
||||||
|
|
||||||
// I think the JSCallFuncWrapper isn't needed.
|
jsval valArr[2];
|
||||||
// Since an action will be run by a cc.Node, it will be released at the Node::cleanup.
|
valArr[0] = senderVal;
|
||||||
// By James Chen
|
valArr[1] = jsvalExtraData;
|
||||||
// JSCallFuncWrapper::setTargetForNativeNode(node, (JSCallFuncWrapper *)this);
|
|
||||||
|
JS::HandleValueArray callArgs = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
||||||
|
JS_CallFunctionValue(cx, thisObj, jsvalCallback, callArgs, &retval);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
JS_SetReservedSlot(proxy->obj, 0, args.get(0));
|
|
||||||
if(argc > 1) {
|
|
||||||
JS_SetReservedSlot(proxy->obj, 1, args.get(1));
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
JS_ReportError(cx, "Invalid number of arguments");
|
JS_ReportError(cx, "Invalid number of arguments");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JSScheduleWrapper::JSScheduleWrapper()
|
||||||
|
: _pTarget(NULL)
|
||||||
|
, _priority(0)
|
||||||
|
, _isUpdateSchedule(false)
|
||||||
|
{
|
||||||
|
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
|
_pPureJSTarget.construct(cx);
|
||||||
|
}
|
||||||
|
|
||||||
JSScheduleWrapper::~JSScheduleWrapper()
|
JSScheduleWrapper::~JSScheduleWrapper()
|
||||||
{
|
{
|
||||||
if (_pPureJSTarget.get()) {
|
_pPureJSTarget.destroyIfConstructed();
|
||||||
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
JS::RemoveObjectRoot(cx, &_pPureJSTarget);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSScheduleWrapper::setTargetForSchedule(JS::HandleValue sched, JSScheduleWrapper *target) {
|
void JSScheduleWrapper::setTargetForSchedule(JS::HandleValue sched, JSScheduleWrapper *target) {
|
||||||
|
@ -1399,43 +1347,25 @@ void JSScheduleWrapper::scheduleFunc(float dt)
|
||||||
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
|
|
||||||
//XXX: really need root?
|
|
||||||
// bool ok = JS_AddValueRoot(cx, &data);
|
|
||||||
// if (!ok) {
|
|
||||||
// CCLOG("scheduleFunc: Root value fails.");
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
|
||||||
if(!_jsCallback.isNullOrUndefined()) {
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
|
if(!callback.isNullOrUndefined()) {
|
||||||
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(1, &data);
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(1, &data);
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
JS_CallFunctionValue(cx, JS::RootedObject(cx, _jsThisObj.toObjectOrNull()), JS::RootedValue(cx, _jsCallback.get()), args, &retval);
|
JS::RootedObject callbackTarget(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
|
JS_CallFunctionValue(cx, callbackTarget, callback, args, &retval);
|
||||||
}
|
}
|
||||||
|
|
||||||
// JS_RemoveValueRoot(cx, &data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSScheduleWrapper::update(float dt)
|
void JSScheduleWrapper::update(float dt)
|
||||||
{
|
{
|
||||||
jsval data = DOUBLE_TO_JSVAL(dt);
|
jsval data = DOUBLE_TO_JSVAL(dt);
|
||||||
|
|
||||||
//XXX: really need root?
|
ScriptingCore::getInstance()->executeFunctionWithOwner(getJSCallbackThis(), "update", 1, &data);
|
||||||
// JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
|
|
||||||
// bool ok = JS_AddValueRoot(cx, &data);
|
|
||||||
// if (!ok) {
|
|
||||||
// CCLOG("scheduleFunc: Root value fails.");
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
|
|
||||||
ScriptingCore::getInstance()->executeFunctionWithOwner(_jsThisObj, "update", 1, &data);
|
|
||||||
|
|
||||||
// JS_RemoveValueRoot(cx, &data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ref* JSScheduleWrapper::getTarget()
|
Ref* JSScheduleWrapper::getTarget()
|
||||||
{
|
{
|
||||||
return _pTarget;
|
return _pTarget;
|
||||||
}
|
}
|
||||||
|
@ -1447,15 +1377,12 @@ void JSScheduleWrapper::setTarget(Ref* pTarget)
|
||||||
|
|
||||||
void JSScheduleWrapper::setPureJSTarget(JS::HandleObject pPureJSTarget)
|
void JSScheduleWrapper::setPureJSTarget(JS::HandleObject pPureJSTarget)
|
||||||
{
|
{
|
||||||
CCASSERT(_pPureJSTarget == NULL, "The pure js target has been set");
|
_pPureJSTarget.ref() = pPureJSTarget;
|
||||||
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
_pPureJSTarget = pPureJSTarget;
|
|
||||||
JS::AddNamedObjectRoot(cx, &_pPureJSTarget, "Pure JS target");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JSObject* JSScheduleWrapper::getPureJSTarget()
|
JSObject* JSScheduleWrapper::getPureJSTarget()
|
||||||
{
|
{
|
||||||
return _pPureJSTarget.get();
|
return _pPureJSTarget.ref().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSScheduleWrapper::setPriority(int priority)
|
void JSScheduleWrapper::setPriority(int priority)
|
||||||
|
@ -1550,8 +1477,8 @@ bool js_CCNode_scheduleOnce(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
||||||
|
|
||||||
// JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
JS::RootedValue thisValue(cx, args.thisv());
|
||||||
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
JS::RootedObject obj(cx, thisValue.toObjectOrNull());
|
||||||
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
||||||
cocos2d::Node *node = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
cocos2d::Node *node = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
||||||
|
|
||||||
|
@ -1586,7 +1513,7 @@ bool js_CCNode_scheduleOnce(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
{
|
{
|
||||||
tmpCobj = new JSScheduleWrapper();
|
tmpCobj = new JSScheduleWrapper();
|
||||||
tmpCobj->autorelease();
|
tmpCobj->autorelease();
|
||||||
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
tmpCobj->setJSCallbackThis(thisValue);
|
||||||
tmpCobj->setJSCallbackFunc(args.get(0));
|
tmpCobj->setJSCallbackFunc(args.get(0));
|
||||||
tmpCobj->setTarget(node);
|
tmpCobj->setTarget(node);
|
||||||
|
|
||||||
|
@ -1630,8 +1557,8 @@ bool js_CCNode_schedule(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
||||||
|
|
||||||
// JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
JS::RootedValue thisValue(cx, args.thisv());
|
||||||
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
JS::RootedObject obj(cx, thisValue.toObjectOrNull());
|
||||||
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
||||||
cocos2d::Node *node = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
cocos2d::Node *node = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
||||||
Scheduler *sched = node->getScheduler();
|
Scheduler *sched = node->getScheduler();
|
||||||
|
@ -1679,7 +1606,7 @@ bool js_CCNode_schedule(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
{
|
{
|
||||||
tmpCobj = new JSScheduleWrapper();
|
tmpCobj = new JSScheduleWrapper();
|
||||||
tmpCobj->autorelease();
|
tmpCobj->autorelease();
|
||||||
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
tmpCobj->setJSCallbackThis(thisValue);
|
||||||
tmpCobj->setJSCallbackFunc(args.get(0));
|
tmpCobj->setJSCallbackFunc(args.get(0));
|
||||||
tmpCobj->setTarget(node);
|
tmpCobj->setTarget(node);
|
||||||
JSScheduleWrapper::setTargetForSchedule(args.get(0), tmpCobj);
|
JSScheduleWrapper::setTargetForSchedule(args.get(0), tmpCobj);
|
||||||
|
@ -1710,8 +1637,8 @@ bool js_cocos2dx_CCNode_scheduleUpdateWithPriority(JSContext *cx, uint32_t argc,
|
||||||
{
|
{
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
// JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
JS::RootedValue thisValue(cx, args.thisv());
|
||||||
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
JS::RootedObject obj(cx, thisValue.toObjectOrNull());
|
||||||
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
||||||
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
||||||
JSB_PRECONDITION2( cobj, cx, false, "Invalid Native Object");
|
JSB_PRECONDITION2( cobj, cx, false, "Invalid Native Object");
|
||||||
|
@ -1753,7 +1680,7 @@ bool js_cocos2dx_CCNode_scheduleUpdateWithPriority(JSContext *cx, uint32_t argc,
|
||||||
{
|
{
|
||||||
tmpCobj = new JSScheduleWrapper();
|
tmpCobj = new JSScheduleWrapper();
|
||||||
tmpCobj->autorelease();
|
tmpCobj->autorelease();
|
||||||
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
tmpCobj->setJSCallbackThis(thisValue);
|
||||||
tmpCobj->setJSCallbackFunc(jsUpdateFunc);
|
tmpCobj->setJSCallbackFunc(jsUpdateFunc);
|
||||||
tmpCobj->setTarget(cobj);
|
tmpCobj->setTarget(cobj);
|
||||||
tmpCobj->setUpdateSchedule(true);
|
tmpCobj->setUpdateSchedule(true);
|
||||||
|
@ -1775,7 +1702,6 @@ bool js_cocos2dx_CCNode_scheduleUpdateWithPriority(JSContext *cx, uint32_t argc,
|
||||||
bool js_cocos2dx_CCNode_unscheduleUpdate(JSContext *cx, uint32_t argc, jsval *vp)
|
bool js_cocos2dx_CCNode_unscheduleUpdate(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
{
|
{
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
||||||
// JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
||||||
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
||||||
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
||||||
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
||||||
|
@ -1785,8 +1711,6 @@ bool js_cocos2dx_CCNode_unscheduleUpdate(JSContext *cx, uint32_t argc, jsval *vp
|
||||||
{
|
{
|
||||||
cobj->unscheduleUpdate();
|
cobj->unscheduleUpdate();
|
||||||
do {
|
do {
|
||||||
// JSObject *tmpObj = obj;
|
|
||||||
|
|
||||||
__Array *arr = JSScheduleWrapper::getTargetForJSObject(obj);
|
__Array *arr = JSScheduleWrapper::getTargetForJSObject(obj);
|
||||||
// If there aren't any targets, just return true.
|
// If there aren't any targets, just return true.
|
||||||
// Otherwise, the for loop will break immediately.
|
// Otherwise, the for loop will break immediately.
|
||||||
|
@ -1817,8 +1741,8 @@ bool js_cocos2dx_CCNode_scheduleUpdate(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
{
|
{
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
// JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
JS::RootedValue thisValue(cx, args.thisv());
|
||||||
JS::RootedObject obj(cx, args.thisv().toObjectOrNull());
|
JS::RootedObject obj(cx, thisValue.toObjectOrNull());
|
||||||
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
js_proxy_t *proxy = jsb_get_js_proxy(obj);
|
||||||
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
cocos2d::Node* cobj = (cocos2d::Node *)(proxy ? proxy->ptr : NULL);
|
||||||
JSB_PRECONDITION2( cobj, cx, false, "Invalid Native Object");
|
JSB_PRECONDITION2( cobj, cx, false, "Invalid Native Object");
|
||||||
|
@ -1857,7 +1781,7 @@ bool js_cocos2dx_CCNode_scheduleUpdate(JSContext *cx, uint32_t argc, jsval *vp)
|
||||||
{
|
{
|
||||||
tmpCobj = new JSScheduleWrapper();
|
tmpCobj = new JSScheduleWrapper();
|
||||||
tmpCobj->autorelease();
|
tmpCobj->autorelease();
|
||||||
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
tmpCobj->setJSCallbackThis(thisValue);
|
||||||
tmpCobj->setJSCallbackFunc(jsUpdateFunc);
|
tmpCobj->setJSCallbackFunc(jsUpdateFunc);
|
||||||
tmpCobj->setTarget(cobj);
|
tmpCobj->setTarget(cobj);
|
||||||
tmpCobj->setUpdateSchedule(true);
|
tmpCobj->setUpdateSchedule(true);
|
||||||
|
|
|
@ -124,24 +124,24 @@ class JSCallbackWrapper: public cocos2d::Ref {
|
||||||
public:
|
public:
|
||||||
JSCallbackWrapper();
|
JSCallbackWrapper();
|
||||||
virtual ~JSCallbackWrapper();
|
virtual ~JSCallbackWrapper();
|
||||||
void setJSCallbackFunc(jsval obj);
|
void setJSCallbackFunc(JS::HandleValue callback);
|
||||||
void setJSCallbackThis(jsval thisObj);
|
void setJSCallbackThis(JS::HandleValue thisObj);
|
||||||
void setJSExtraData(jsval data);
|
void setJSExtraData(JS::HandleValue data);
|
||||||
|
|
||||||
const jsval& getJSCallbackFunc() const;
|
const jsval getJSCallbackFunc() const;
|
||||||
const jsval& getJSCallbackThis() const;
|
const jsval getJSCallbackThis() const;
|
||||||
const jsval& getJSExtraData() const;
|
const jsval getJSExtraData() const;
|
||||||
protected:
|
protected:
|
||||||
JS::Heap<JS::Value> _jsCallback;
|
mozilla::Maybe<JS::PersistentRootedValue> _jsCallback;
|
||||||
JS::Heap<JS::Value> _jsThisObj;
|
mozilla::Maybe<JS::PersistentRootedValue> _jsThisObj;
|
||||||
JS::Heap<JS::Value> _extraData;
|
mozilla::Maybe<JS::PersistentRootedValue> _extraData;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
class JSScheduleWrapper: public JSCallbackWrapper {
|
class JSScheduleWrapper: public JSCallbackWrapper {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
JSScheduleWrapper() : _pTarget(NULL), _pPureJSTarget(NULL), _priority(0), _isUpdateSchedule(false) {}
|
JSScheduleWrapper();
|
||||||
virtual ~JSScheduleWrapper();
|
virtual ~JSScheduleWrapper();
|
||||||
|
|
||||||
static void setTargetForSchedule(JS::HandleValue sched, JSScheduleWrapper *target);
|
static void setTargetForSchedule(JS::HandleValue sched, JSScheduleWrapper *target);
|
||||||
|
@ -178,7 +178,7 @@ public:
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
Ref* _pTarget;
|
Ref* _pTarget;
|
||||||
JS::Heap<JSObject*> _pPureJSTarget;
|
mozilla::Maybe<JS::PersistentRootedObject> _pPureJSTarget;
|
||||||
int _priority;
|
int _priority;
|
||||||
bool _isUpdateSchedule;
|
bool _isUpdateSchedule;
|
||||||
};
|
};
|
||||||
|
|
|
@ -34,16 +34,17 @@ public:
|
||||||
|
|
||||||
void animationCompleteCallback()
|
void animationCompleteCallback()
|
||||||
{
|
{
|
||||||
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
if(!_jsCallback.isNullOrUndefined() && !_jsThisObj.isNullOrUndefined())
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
|
JS::RootedValue thisObj(cx, getJSCallbackThis());
|
||||||
|
if(!callback.isNullOrUndefined() && !thisObj.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
|
||||||
JS_CallFunctionValue(cx, JS::RootedObject(cx, _jsThisObj.toObjectOrNull()), JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::empty(), &retval);
|
JS_CallFunctionValue(cx, JS::RootedObject(cx, thisObj.toObjectOrNull()), JS::RootedValue(cx, callback), JS::HandleValueArray::empty(), &retval);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -28,55 +28,19 @@
|
||||||
|
|
||||||
class JSArmatureWrapper: public JSCallbackWrapper {
|
class JSArmatureWrapper: public JSCallbackWrapper {
|
||||||
public:
|
public:
|
||||||
JSArmatureWrapper();
|
|
||||||
virtual ~JSArmatureWrapper();
|
|
||||||
|
|
||||||
virtual void setJSCallbackThis(jsval thisObj);
|
|
||||||
|
|
||||||
void movementCallbackFunc(cocostudio::Armature *armature, cocostudio::MovementEventType movementType, const std::string& movementID);
|
void movementCallbackFunc(cocostudio::Armature *armature, cocostudio::MovementEventType movementType, const std::string& movementID);
|
||||||
void frameCallbackFunc(cocostudio::Bone *bone, const std::string& evt, int originFrameIndex, int currentFrameIndex);
|
void frameCallbackFunc(cocostudio::Bone *bone, const std::string& evt, int originFrameIndex, int currentFrameIndex);
|
||||||
void addArmatureFileInfoAsyncCallbackFunc(float percent);
|
void addArmatureFileInfoAsyncCallbackFunc(float percent);
|
||||||
|
|
||||||
private:
|
|
||||||
bool m_bNeedUnroot;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
JSArmatureWrapper::JSArmatureWrapper()
|
|
||||||
: m_bNeedUnroot(false)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JSArmatureWrapper::~JSArmatureWrapper()
|
|
||||||
{
|
|
||||||
if (m_bNeedUnroot)
|
|
||||||
{
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
JS::RemoveValueRoot(cx, &_jsThisObj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void JSArmatureWrapper::setJSCallbackThis(jsval obj)
|
|
||||||
{
|
|
||||||
JSCallbackWrapper::setJSCallbackThis(obj);
|
|
||||||
|
|
||||||
JSObject *thisObj = obj.toObjectOrNull();
|
|
||||||
js_proxy *p = jsb_get_js_proxy(thisObj);
|
|
||||||
if (!p)
|
|
||||||
{
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
m_bNeedUnroot = true;
|
|
||||||
m_bNeedUnroot &= JS::AddValueRoot(cx, &_jsThisObj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void JSArmatureWrapper::movementCallbackFunc(cocostudio::Armature *armature, cocostudio::MovementEventType movementType, const std::string& movementID)
|
void JSArmatureWrapper::movementCallbackFunc(cocostudio::Armature *armature, cocostudio::MovementEventType movementType, const std::string& movementID)
|
||||||
{
|
{
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
JS::RootedObject thisObj(cx, _jsThisObj.toObjectOrNull());
|
JS::RootedObject thisObj(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy(cx, armature);
|
js_proxy_t *proxy = js_get_or_create_proxy(cx, armature);
|
||||||
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
if (_jsCallback != JSVAL_VOID)
|
if (!callback.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
int movementEventType = (int)movementType;
|
int movementEventType = (int)movementType;
|
||||||
jsval movementVal = INT_TO_JSVAL(movementEventType);
|
jsval movementVal = INT_TO_JSVAL(movementEventType);
|
||||||
|
@ -88,30 +52,25 @@ void JSArmatureWrapper::movementCallbackFunc(cocostudio::Armature *armature, coc
|
||||||
valArr[1] = movementVal;
|
valArr[1] = movementVal;
|
||||||
valArr[2] = idVal;
|
valArr[2] = idVal;
|
||||||
|
|
||||||
//JS_AddValueRoot(cx, valArr);
|
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
|
||||||
JS_CallFunctionValue(cx, thisObj, JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::fromMarkedLocation(3, valArr), &retval);
|
JS_CallFunctionValue(cx, thisObj, callback, JS::HandleValueArray::fromMarkedLocation(3, valArr), &retval);
|
||||||
//JS_RemoveValueRoot(cx, valArr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void JSArmatureWrapper::addArmatureFileInfoAsyncCallbackFunc(float percent)
|
void JSArmatureWrapper::addArmatureFileInfoAsyncCallbackFunc(float percent)
|
||||||
{
|
{
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
JS::RootedObject thisObj(cx, _jsThisObj.toObjectOrNull());
|
JS::RootedObject thisObj(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
if (_jsCallback != JSVAL_VOID)
|
if (!callback.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
jsval percentVal = DOUBLE_TO_JSVAL(percent);
|
jsval percentVal = DOUBLE_TO_JSVAL(percent);
|
||||||
|
|
||||||
//JS_AddValueRoot(cx, &percentVal);
|
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
|
||||||
JS_CallFunctionValue(cx, thisObj, JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::fromMarkedLocation(1, &percentVal), &retval);
|
JS_CallFunctionValue(cx, thisObj, callback, JS::HandleValueArray::fromMarkedLocation(1, &percentVal), &retval);
|
||||||
//JS_RemoveValueRoot(cx, &percentVal);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -121,10 +80,11 @@ void JSArmatureWrapper::frameCallbackFunc(cocostudio::Bone *bone, const std::str
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
JS::RootedObject thisObj(cx, _jsThisObj.toObjectOrNull());
|
JS::RootedObject thisObj(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy(cx, bone);
|
js_proxy_t *proxy = js_get_or_create_proxy(cx, bone);
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
if (_jsCallback != JSVAL_VOID)
|
if (!callback.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
jsval nameVal = std_string_to_jsval(cx, evt);
|
jsval nameVal = std_string_to_jsval(cx, evt);
|
||||||
jsval originIndexVal = INT_TO_JSVAL(originFrameIndex);
|
jsval originIndexVal = INT_TO_JSVAL(originFrameIndex);
|
||||||
|
@ -136,10 +96,7 @@ void JSArmatureWrapper::frameCallbackFunc(cocostudio::Bone *bone, const std::str
|
||||||
valArr[2] = originIndexVal;
|
valArr[2] = originIndexVal;
|
||||||
valArr[3] = currentIndexVal;
|
valArr[3] = currentIndexVal;
|
||||||
|
|
||||||
//JS_AddValueRoot(cx, valArr);
|
JS_CallFunctionValue(cx, thisObj, callback, JS::HandleValueArray::fromMarkedLocation(4, valArr), &retval);
|
||||||
|
|
||||||
JS_CallFunctionValue(cx, thisObj, JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::fromMarkedLocation(4, valArr), &retval);
|
|
||||||
//JS_RemoveValueRoot(cx, valArr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -173,7 +130,8 @@ static bool js_cocos2dx_ArmatureAnimation_setMovementEventCallFunc(JSContext *cx
|
||||||
tmpObj->setJSCallbackFunc(args.get(0));
|
tmpObj->setJSCallbackFunc(args.get(0));
|
||||||
if (argc == 1)
|
if (argc == 1)
|
||||||
{
|
{
|
||||||
tmpObj->setJSCallbackThis(JSVAL_NULL);
|
JS::RootedValue nullVal(cx, JS::NullValue());
|
||||||
|
tmpObj->setJSCallbackThis(nullVal);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -219,7 +177,8 @@ static bool js_cocos2dx_ArmatureAnimation_setFrameEventCallFunc(JSContext *cx, u
|
||||||
tmpObj->setJSCallbackFunc(args.get(0));
|
tmpObj->setJSCallbackFunc(args.get(0));
|
||||||
if (argc == 1)
|
if (argc == 1)
|
||||||
{
|
{
|
||||||
tmpObj->setJSCallbackThis(JSVAL_NULL);
|
JS::RootedValue nullVal(cx, JS::NullValue());
|
||||||
|
tmpObj->setJSCallbackThis(nullVal);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -417,7 +376,7 @@ static bool js_cocos2dx_studio_Frame_getEasingParams(JSContext *cx, uint32_t arg
|
||||||
bool ok = true;
|
bool ok = true;
|
||||||
for(size_t i = 0; i < ret.size(); ++i)
|
for(size_t i = 0; i < ret.size(); ++i)
|
||||||
{
|
{
|
||||||
ok &= JS_SetElement(cx, jsobj, i, ret[i]);
|
ok &= JS_SetElement(cx, jsobj, (int)i, ret[i]);
|
||||||
}
|
}
|
||||||
JSB_PRECONDITION2(ok, cx, false, "js_cocos2dx_studio_Frame_getEasingParams : Error processing arguments");
|
JSB_PRECONDITION2(ok, cx, false, "js_cocos2dx_studio_Frame_getEasingParams : Error processing arguments");
|
||||||
|
|
||||||
|
|
|
@ -564,10 +564,11 @@ public:
|
||||||
|
|
||||||
void animationCallbackFunc(spine::SkeletonAnimation* node, int trackIndex, spEventType type, spEvent* event, int loopCount) const {
|
void animationCallbackFunc(spine::SkeletonAnimation* node, int trackIndex, spEventType type, spEvent* event, int loopCount) const {
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
JS::RootedObject thisObj(cx, _jsThisObj.toObjectOrNull());
|
JS::RootedObject thisObj(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy(cx, node);
|
js_proxy_t *proxy = js_get_or_create_proxy(cx, node);
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
if (_jsCallback != JSVAL_VOID)
|
if (!callback.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
jsval nodeVal = OBJECT_TO_JSVAL(proxy->obj);
|
jsval nodeVal = OBJECT_TO_JSVAL(proxy->obj);
|
||||||
jsval trackIndexVal = INT_TO_JSVAL(trackIndex);
|
jsval trackIndexVal = INT_TO_JSVAL(trackIndex);
|
||||||
|
@ -585,10 +586,8 @@ public:
|
||||||
valArr[3] = eventVal;
|
valArr[3] = eventVal;
|
||||||
valArr[4] = loopCountVal;
|
valArr[4] = loopCountVal;
|
||||||
|
|
||||||
//JS_AddValueRoot(cx, valArr);
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
JS_CallFunctionValue(cx, thisObj, JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::fromMarkedLocation(5, valArr), &retval);
|
JS_CallFunctionValue(cx, thisObj, callback, JS::HandleValueArray::fromMarkedLocation(5, valArr), &retval);
|
||||||
//JS_RemoveValueRoot(cx, valArr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -31,53 +31,17 @@ using namespace cocos2d::ui;
|
||||||
|
|
||||||
class JSStudioEventListenerWrapper: public JSCallbackWrapper {
|
class JSStudioEventListenerWrapper: public JSCallbackWrapper {
|
||||||
public:
|
public:
|
||||||
JSStudioEventListenerWrapper();
|
|
||||||
virtual ~JSStudioEventListenerWrapper();
|
|
||||||
|
|
||||||
virtual void setJSCallbackThis(jsval thisObj);
|
|
||||||
|
|
||||||
virtual void eventCallbackFunc(Ref*,int);
|
virtual void eventCallbackFunc(Ref*,int);
|
||||||
|
|
||||||
private:
|
|
||||||
bool m_bNeedUnroot;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
JSStudioEventListenerWrapper::JSStudioEventListenerWrapper()
|
|
||||||
: m_bNeedUnroot(false)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JSStudioEventListenerWrapper::~JSStudioEventListenerWrapper()
|
|
||||||
{
|
|
||||||
if (m_bNeedUnroot)
|
|
||||||
{
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
JS::RemoveValueRoot(cx, &_jsThisObj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void JSStudioEventListenerWrapper::setJSCallbackThis(jsval jsThisObj)
|
|
||||||
{
|
|
||||||
JSCallbackWrapper::setJSCallbackThis(jsThisObj);
|
|
||||||
|
|
||||||
JSObject *thisObj = jsThisObj.toObjectOrNull();
|
|
||||||
js_proxy *p = jsb_get_js_proxy(thisObj);
|
|
||||||
if (!p)
|
|
||||||
{
|
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
||||||
m_bNeedUnroot = true;
|
|
||||||
m_bNeedUnroot &= JS::AddValueRoot(cx, &_jsThisObj);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void JSStudioEventListenerWrapper::eventCallbackFunc(Ref* sender,int eventType)
|
void JSStudioEventListenerWrapper::eventCallbackFunc(Ref* sender,int eventType)
|
||||||
{
|
{
|
||||||
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
||||||
JS::RootedObject thisObj(cx, _jsThisObj.isNullOrUndefined() ? NULL : _jsThisObj.toObjectOrNull());
|
JS::RootedObject thisObj(cx, getJSCallbackThis().toObjectOrNull());
|
||||||
|
JS::RootedValue callback(cx, getJSCallbackFunc());
|
||||||
js_proxy_t *proxy = js_get_or_create_proxy(cx, sender);
|
js_proxy_t *proxy = js_get_or_create_proxy(cx, sender);
|
||||||
JS::RootedValue retval(cx);
|
JS::RootedValue retval(cx);
|
||||||
if (!_jsCallback.isNullOrUndefined())
|
if (!callback.isNullOrUndefined())
|
||||||
{
|
{
|
||||||
jsval touchVal = INT_TO_JSVAL(eventType);
|
jsval touchVal = INT_TO_JSVAL(eventType);
|
||||||
|
|
||||||
|
@ -85,12 +49,8 @@ void JSStudioEventListenerWrapper::eventCallbackFunc(Ref* sender,int eventType)
|
||||||
valArr[0] = OBJECT_TO_JSVAL(proxy->obj);
|
valArr[0] = OBJECT_TO_JSVAL(proxy->obj);
|
||||||
valArr[1] = touchVal;
|
valArr[1] = touchVal;
|
||||||
|
|
||||||
//JS::AddValueRoot(cx, valArr);
|
|
||||||
|
|
||||||
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
||||||
|
JS_CallFunctionValue(cx, thisObj, callback, JS::HandleValueArray::fromMarkedLocation(2, valArr), &retval);
|
||||||
JS_CallFunctionValue(cx, thisObj, JS::RootedValue(cx, _jsCallback), JS::HandleValueArray::fromMarkedLocation(2, valArr), &retval);
|
|
||||||
//JS::RemoveValueRoot(cx, valArr);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue