2012-08-28 10:22:36 +08:00
|
|
|
#include "cocos2d.h"
|
|
|
|
#include "cocos2d_specifics.hpp"
|
|
|
|
#include <typeinfo>
|
2013-01-11 17:44:39 +08:00
|
|
|
#include "js_bindings_config.h"
|
2012-08-28 10:22:36 +08:00
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
schedFunc_proxy_t *_schedFunc_target_ht = NULL;
|
|
|
|
schedTarget_proxy_t *_schedTarget_native_ht = NULL;
|
2012-10-23 02:07:23 +08:00
|
|
|
callfuncTarget_proxy_t *_callfuncTarget_native_ht = NULL;
|
2012-10-10 05:59:16 +08:00
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSTouchDelegate::TouchDelegateMap JSTouchDelegate::sTouchDelegateMap;
|
|
|
|
|
|
|
|
void JSTouchDelegate::setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate)
|
|
|
|
{
|
|
|
|
CCAssert(sTouchDelegateMap.find(pJSObj) == sTouchDelegateMap.end(), "");
|
|
|
|
sTouchDelegateMap.insert(TouchDelegatePair(pJSObj, pDelegate));
|
|
|
|
}
|
|
|
|
|
|
|
|
JSTouchDelegate* JSTouchDelegate::getDelegateForJSObject(JSObject* pJSObj)
|
|
|
|
{
|
|
|
|
JSTouchDelegate* pRet = NULL;
|
|
|
|
TouchDelegateMap::iterator iter = sTouchDelegateMap.find(pJSObj);
|
|
|
|
if (iter != sTouchDelegateMap.end())
|
|
|
|
{
|
|
|
|
pRet = iter->second;
|
|
|
|
}
|
|
|
|
return pRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::removeDelegateForJSObject(JSObject* pJSObj)
|
|
|
|
{
|
|
|
|
TouchDelegateMap::iterator iter = sTouchDelegateMap.find(pJSObj);
|
|
|
|
CCAssert(iter != sTouchDelegateMap.end(), "");
|
|
|
|
sTouchDelegateMap.erase(pJSObj);
|
|
|
|
}
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
void JSTouchDelegate::setJSObject(JSObject *obj) {
|
|
|
|
_mObj = obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::registerStandardDelegate() {
|
|
|
|
CCDirector* pDirector = CCDirector::sharedDirector();
|
|
|
|
pDirector->getTouchDispatcher()->addStandardDelegate(this,0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::registerTargettedDelegate(int priority, bool swallowsTouches) {
|
|
|
|
CCDirector* pDirector = CCDirector::sharedDirector();
|
|
|
|
pDirector->getTouchDispatcher()->addTargetedDelegate(this,
|
|
|
|
priority,
|
|
|
|
swallowsTouches);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
void JSTouchDelegate::unregisterTouchDelegate()
|
|
|
|
{
|
|
|
|
CCDirector* pDirector = CCDirector::sharedDirector();
|
|
|
|
pDirector->getTouchDispatcher()->removeDelegate(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSTouchDelegate::ccTouchBegan(CCTouch *pTouch, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
jsval retval;
|
|
|
|
bool bRet = false;
|
|
|
|
|
|
|
|
js_proxy_t* p = NULL;
|
|
|
|
JS_GET_NATIVE_PROXY(p, _mObj);
|
|
|
|
CCAssert(p, "js object has been unrooted.");
|
|
|
|
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchEvent(CCTOUCHBEGAN,
|
|
|
|
pTouch, _mObj, retval);
|
|
|
|
if(JSVAL_IS_BOOLEAN(retval)) {
|
|
|
|
bRet = JSVAL_TO_BOOLEAN(retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bRet;
|
|
|
|
};
|
|
|
|
// optional
|
|
|
|
|
|
|
|
void JSTouchDelegate::ccTouchMoved(CCTouch *pTouch, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
|
|
|
|
//jsval retval;
|
|
|
|
js_proxy_t* p = NULL;
|
|
|
|
JS_GET_NATIVE_PROXY(p, _mObj);
|
|
|
|
CCAssert(p, "js object has been unrooted.");
|
|
|
|
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchEvent(CCTOUCHMOVED,
|
|
|
|
pTouch, _mObj);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::ccTouchEnded(CCTouch *pTouch, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
|
|
|
|
js_proxy_t* p = NULL;
|
|
|
|
JS_GET_NATIVE_PROXY(p, _mObj);
|
|
|
|
CCAssert(p, "js object has been unrooted.");
|
|
|
|
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchEvent(CCTOUCHENDED,
|
|
|
|
pTouch, _mObj);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::ccTouchCancelled(CCTouch *pTouch, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
js_proxy_t* p = NULL;
|
|
|
|
JS_GET_NATIVE_PROXY(p, _mObj);
|
|
|
|
CCAssert(p, "js object has been unrooted.");
|
|
|
|
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchEvent(CCTOUCHCANCELLED,
|
|
|
|
pTouch, _mObj);
|
|
|
|
}
|
|
|
|
|
|
|
|
// optional
|
|
|
|
void JSTouchDelegate::ccTouchesBegan(CCSet *pTouches, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchesEvent(CCTOUCHBEGAN,
|
|
|
|
pTouches, _mObj);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSTouchDelegate::ccTouchesMoved(CCSet *pTouches, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchesEvent(CCTOUCHMOVED,
|
|
|
|
pTouches, _mObj);
|
|
|
|
}
|
|
|
|
void JSTouchDelegate::ccTouchesEnded(CCSet *pTouches, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchesEvent(CCTOUCHENDED,
|
|
|
|
pTouches, _mObj);
|
|
|
|
}
|
|
|
|
void JSTouchDelegate::ccTouchesCancelled(CCSet *pTouches, CCEvent *pEvent) {
|
|
|
|
CC_UNUSED_PARAM(pEvent);
|
|
|
|
ScriptingCore::getInstance()->executeCustomTouchesEvent(CCTOUCHCANCELLED,
|
|
|
|
pTouches, _mObj);
|
|
|
|
}
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
static void addCallBackAndThis(JSObject *obj, jsval callback, jsval &thisObj) {
|
|
|
|
if(callback != JSVAL_VOID) {
|
|
|
|
ScriptingCore::getInstance()->setReservedSpot(0, obj, callback);
|
|
|
|
}
|
|
|
|
if(thisObj != JSVAL_VOID) {
|
|
|
|
ScriptingCore::getInstance()->setReservedSpot(1, obj, thisObj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
JSObject* bind_menu_item(JSContext *cx, T* nativeObj, jsval callback, jsval thisObj) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, nativeObj);
|
|
|
|
if (p) {
|
|
|
|
addCallBackAndThis(p->obj, callback, thisObj);
|
|
|
|
return p->obj;
|
|
|
|
} else {
|
|
|
|
js_type_class_t *classType = js_get_type_from_native<T>(nativeObj);
|
|
|
|
assert(classType);
|
|
|
|
JSObject *tmp = JS_NewObject(cx, classType->jsclass, classType->proto, classType->parentProto);
|
|
|
|
|
|
|
|
// bind nativeObj <-> JSObject
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_NEW_PROXY(proxy, nativeObj, tmp);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JS_AddNamedObjectRoot(cx, &proxy->obj, typeid(*nativeObj).name());
|
2012-08-28 10:22:36 +08:00
|
|
|
addCallBackAndThis(tmp, callback, thisObj);
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCNode_getChildren(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *thisObj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
if (thisObj) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, thisObj);
|
|
|
|
if (proxy) {
|
|
|
|
cocos2d::CCNode *node = (cocos2d::CCNode *)(proxy->ptr ? proxy->ptr : NULL);
|
|
|
|
cocos2d::CCArray *children = node->getChildren();
|
2012-12-24 11:55:04 +08:00
|
|
|
unsigned int count = children ? children->count() : 0;
|
|
|
|
JSObject *jsarr = JS_NewArrayObject(cx, count, NULL);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
|
2012-12-24 11:55:04 +08:00
|
|
|
for (unsigned int i=0; i < count; i++) {
|
2012-08-28 10:22:36 +08:00
|
|
|
cocos2d::CCNode *child = (cocos2d::CCNode*)children->objectAtIndex(i);
|
|
|
|
js_proxy_t *childProxy = js_get_or_create_proxy<cocos2d::CCNode>(cx, child);
|
|
|
|
jsval childVal = OBJECT_TO_JSVAL(childProxy->obj);
|
|
|
|
JS_SetElement(cx, jsarr, i, &childVal);
|
|
|
|
}
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(jsarr));
|
2013-01-11 17:44:39 +08:00
|
|
|
return JS_TRUE;
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "Error in js_cocos2dx_CCNode_getChildren");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCMenu_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
if (argc > 0) {
|
|
|
|
cocos2d::CCArray* array = cocos2d::CCArray::create();
|
2012-12-20 17:18:49 +08:00
|
|
|
uint32_t i = 0;
|
2012-08-28 10:22:36 +08:00
|
|
|
while (i < argc) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[i]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCObject *item = (cocos2d::CCObject*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, item)
|
|
|
|
array->addObject(item);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
cocos2d::CCMenu* ret = cocos2d::CCMenu::createWithArray(array);
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCMenu>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
if (argc == 0) {
|
|
|
|
cocos2d::CCMenu* ret = cocos2d::CCMenu::create();
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCMenu>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSequence_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
if (argc > 0) {
|
|
|
|
cocos2d::CCArray* array = cocos2d::CCArray::create();
|
2012-12-20 17:18:49 +08:00
|
|
|
uint32_t i = 0;
|
2012-08-28 10:22:36 +08:00
|
|
|
while (i < argc) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[i]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCObject *item = (cocos2d::CCObject*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, item)
|
|
|
|
array->addObject(item);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
cocos2d::CCFiniteTimeAction* ret = cocos2d::CCSequence::create(array);
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCFiniteTimeAction>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSpawn_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
if (argc > 0) {
|
|
|
|
cocos2d::CCArray* array = cocos2d::CCArray::create();
|
2012-12-20 17:18:49 +08:00
|
|
|
uint32_t i = 0;
|
2012-08-28 10:22:36 +08:00
|
|
|
while (i < argc) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[i]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCObject *item = (cocos2d::CCObject*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, item)
|
|
|
|
array->addObject(item);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
cocos2d::CCFiniteTimeAction* ret = cocos2d::CCSpawn::create(array);
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCFiniteTimeAction>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCMenuItem_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
cocos2d::CCMenuItem* ret = cocos2d::CCMenuItem::create();
|
2012-11-03 01:23:23 +08:00
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItem>(cx, ret, argv[0], argc == 2? argv[1] : JSVAL_VOID);
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// "create" in JS
|
2012-11-05 18:44:28 +08:00
|
|
|
// cc.MenuItemSprite.create( normalSprite, selectedSprite, [disabledSprite], [callback_fn], [this]
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItemSprite_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2012-11-05 18:44:28 +08:00
|
|
|
if (argc >= 2 && argc <= 5) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj;
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCNode* arg0 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg0);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(argv[1]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCNode* arg1 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg1);
|
|
|
|
|
|
|
|
int last = 2;
|
2012-11-05 23:22:59 +08:00
|
|
|
JSBool thirdArgIsCallback = JS_FALSE;
|
|
|
|
|
|
|
|
jsval jsCallback = JSVAL_VOID;
|
|
|
|
jsval jsThis = JSVAL_VOID;
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
cocos2d::CCNode* arg2 = NULL;
|
2012-11-05 23:22:59 +08:00
|
|
|
if (argc >= 3) {
|
2012-08-28 10:22:36 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(argv[2]);
|
2012-11-05 23:22:59 +08:00
|
|
|
thirdArgIsCallback = JS_ObjectIsFunction(cx, tmpObj);
|
|
|
|
if (!thirdArgIsCallback) {
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
arg2 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg2);
|
|
|
|
last = 3;
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
cocos2d::CCMenuItemSprite* ret = cocos2d::CCMenuItemSprite::create(arg0, arg1, arg2);
|
2012-11-05 23:22:59 +08:00
|
|
|
if (argc >= 3) {
|
|
|
|
if (thirdArgIsCallback) {
|
|
|
|
//cc.MenuItemSprite.create( normalSprite, selectedSprite, callback_fn, [this] )
|
|
|
|
jsCallback = argv[last++];
|
|
|
|
if (argc == 4) {
|
|
|
|
jsThis = argv[last];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
//cc.MenuItemSprite.create( normalSprite, selectedSprite, disabledSprite, callback_fn, [this] )
|
|
|
|
if (argc >= 4) {
|
|
|
|
jsCallback = argv[last++];
|
|
|
|
if (argc == 5) {
|
|
|
|
jsThis = argv[last];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItemSprite>(cx, ret, jsCallback, jsThis);
|
2012-11-03 01:23:23 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "Invalid number of arguments. Expecting: 2 <= args <= 5");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// "create" in JS
|
2012-11-05 18:44:28 +08:00
|
|
|
// cc.MenuItemImage.create( normalImage, selectedImage, [disabledImage], callback_fn, [this]
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItemImage_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2012-11-05 18:44:28 +08:00
|
|
|
if (argc >= 2 && argc <= 5) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2012-11-16 03:14:37 +08:00
|
|
|
JSStringWrapper arg0(argv[0]);
|
|
|
|
JSStringWrapper arg1(argv[1]);
|
|
|
|
JSStringWrapper arg2;
|
2012-11-05 23:22:59 +08:00
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
bool thirdArgIsString = true;
|
2012-11-05 23:22:59 +08:00
|
|
|
|
|
|
|
jsval jsCallback = JSVAL_VOID;
|
|
|
|
jsval jsThis = JSVAL_VOID;
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
int last = 2;
|
2012-11-05 23:22:59 +08:00
|
|
|
if (argc >= 3) {
|
2012-11-16 03:14:37 +08:00
|
|
|
thirdArgIsString = argv[2].isString();
|
2012-11-05 23:22:59 +08:00
|
|
|
if (thirdArgIsString) {
|
2012-11-16 03:14:37 +08:00
|
|
|
arg2.set(argv[2], cx);
|
2012-11-05 23:22:59 +08:00
|
|
|
last = 3;
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
cocos2d::CCMenuItemImage* ret = cocos2d::CCMenuItemImage::create(arg0, arg1, arg2);
|
2012-11-05 23:22:59 +08:00
|
|
|
|
|
|
|
if (argc >= 3) {
|
|
|
|
if (!thirdArgIsString) {
|
|
|
|
//cc.MenuItemImage.create( normalImage, selectedImage, callback_fn, [this] )
|
|
|
|
jsCallback = argv[last++];
|
|
|
|
if (argc == 4) {
|
|
|
|
jsThis = argv[last];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
//cc.MenuItemImage.create( normalImage, selectedImage, disabledImage, callback_fn, [this] )
|
|
|
|
if (argc >= 4) {
|
|
|
|
jsCallback = argv[last++];
|
|
|
|
if (argc == 5) {
|
|
|
|
jsThis = argv[last];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItemImage>(cx, ret, jsCallback, jsThis);
|
2012-11-03 01:23:23 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "Invalid number of arguments. Expecting: 2 <= args <= 5");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// "create" in JS:
|
2012-11-05 18:44:28 +08:00
|
|
|
// cc.MenuItemLabel.create( label, callback_fn, [this] );
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItemLabel_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2012-11-05 18:44:28 +08:00
|
|
|
if (argc >= 1 && argc <= 3) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCNode* arg0 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg0)
|
|
|
|
cocos2d::CCMenuItemLabel* ret = cocos2d::CCMenuItemLabel::create(arg0);
|
2012-11-05 18:44:28 +08:00
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItemLabel>(cx, ret, (argc >= 2 ? argv[1] : JSVAL_VOID), (argc == 3 ? argv[2] : JSVAL_VOID) );
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d or %d", argc, 1, 2, 3);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCMenuItemAtlasFont_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 5) {
|
2013-01-11 17:44:39 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2012-11-16 03:14:37 +08:00
|
|
|
JSStringWrapper arg0(argv[0]);
|
|
|
|
JSStringWrapper arg1(argv[1]);
|
2013-01-11 17:44:39 +08:00
|
|
|
int arg2; ok &= jsval_to_int32(cx, argv[2], &arg2);
|
|
|
|
int arg3; ok &= jsval_to_int32(cx, argv[3], &arg3);
|
|
|
|
int arg4; ok &= jsval_to_int32(cx, argv[4], &arg4);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
cocos2d::CCMenuItemAtlasFont* ret = cocos2d::CCMenuItemAtlasFont::create(arg0, arg1, arg2, arg3, arg4);
|
2012-11-03 01:23:23 +08:00
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItemAtlasFont>(cx, ret, (argc >= 6 ? argv[5] : JSVAL_VOID), (argc == 7 ? argv[6] : JSVAL_VOID));
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// "create" in JS
|
2012-11-05 18:44:28 +08:00
|
|
|
// cc.MenuItemFont.create( string, callback_fn, [this] );
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItemFont_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2012-11-05 18:44:28 +08:00
|
|
|
if (argc >= 1 && argc <= 3) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2012-11-16 03:14:37 +08:00
|
|
|
JSStringWrapper arg0(argv[0]);
|
2012-08-28 10:22:36 +08:00
|
|
|
cocos2d::CCMenuItemFont* ret = cocos2d::CCMenuItemFont::create(arg0);
|
2012-11-05 18:44:28 +08:00
|
|
|
JSObject *obj = bind_menu_item<cocos2d::CCMenuItemFont>(cx, ret, (argc >= 2 ? argv[1] : JSVAL_VOID), (argc == 3 ? argv[2] : JSVAL_VOID));
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d or %d", argc, 1, 2, 3);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-09-19 09:56:54 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItemToggle_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2012-09-19 09:56:54 +08:00
|
|
|
if (argc >= 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
cocos2d::CCMenuItemToggle* ret = cocos2d::CCMenuItemToggle::create();
|
2012-12-20 17:18:49 +08:00
|
|
|
|
|
|
|
for (uint32_t i=0; i < argc; i++) {
|
2012-09-19 09:56:54 +08:00
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[i]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
cocos2d::CCMenuItem* item = (cocos2d::CCMenuItem*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, item)
|
|
|
|
if(i == 0) ret->initWithItem(item);
|
|
|
|
else ret->addSubItem(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
jsval jsret;
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_PROXY(proxy, ret);
|
|
|
|
if (proxy) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
proxy = js_get_or_create_proxy<cocos2d::CCMenuItemToggle>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-09-19 09:56:54 +08:00
|
|
|
return JS_FALSE;
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// "setCallback" in JS
|
2012-11-05 18:44:28 +08:00
|
|
|
// item.setCallback( callback_fn, [this]);
|
2012-10-23 02:07:23 +08:00
|
|
|
template<class T>
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_cocos2dx_setCallback(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
|
2012-11-05 18:44:28 +08:00
|
|
|
if(argc == 1 || argc == 2) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
2012-11-05 18:44:28 +08:00
|
|
|
jsval jsThis = JSVAL_VOID;
|
|
|
|
jsval jsFunc = argv[0];
|
|
|
|
if (argc == 2) {
|
|
|
|
jsThis = argv[1];
|
|
|
|
}
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
2012-10-23 02:07:23 +08:00
|
|
|
T* item = (T*)(proxy ? proxy->ptr : NULL);
|
2012-08-28 10:22:36 +08:00
|
|
|
TEST_NATIVE_OBJECT(cx, item)
|
2012-11-05 18:44:28 +08:00
|
|
|
bind_menu_item(cx, item, jsFunc, jsThis);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d", argc, 1, 2);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-10-23 02:07:23 +08:00
|
|
|
JSBool js_cocos2dx_CCMenuItem_setCallback(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return js_cocos2dx_setCallback<cocos2d::CCMenuItem>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCAnimation_create(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
if (argc <= 3) {
|
|
|
|
cocos2d::CCArray* arg0;
|
|
|
|
if (argc > 0) {
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccarray(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
2012-10-23 02:07:23 +08:00
|
|
|
cocos2d::CCAnimation* ret;
|
2012-08-28 10:22:36 +08:00
|
|
|
double arg1 = 0.0f;
|
|
|
|
if (argc > 0 && argc == 2) {
|
|
|
|
if (argc == 2) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &arg1);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
ret = cocos2d::CCAnimation::createWithSpriteFrames(arg0, arg1);
|
|
|
|
} else if (argc == 3) {
|
2012-08-28 10:22:36 +08:00
|
|
|
unsigned int loops;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &arg1);
|
|
|
|
ok &= jsval_to_uint32(cx, argv[1], &loops);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
ret = cocos2d::CCAnimation::create(arg0, arg1, loops);
|
2012-10-10 05:59:16 +08:00
|
|
|
} else if (argc == 1) {
|
|
|
|
ret = cocos2d::CCAnimation::createWithSpriteFrames(arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
} else if (argc == 0) {
|
|
|
|
ret = cocos2d::CCAnimation::create();
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval jsret;
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_PROXY(proxy, ret);
|
|
|
|
if (proxy) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
proxy = js_get_or_create_proxy<cocos2d::CCAnimation>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-30 21:15:01 +08:00
|
|
|
JSBool js_cocos2dx_CCLayerMultiplex_create(JSContext *cx, uint32_t argc, jsval *vp)
|
2013-01-11 13:54:57 +08:00
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
cocos2d::CCArray* arg0;
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
ok &= jsvals_variadic_to_ccarray(cx, argv, argc, &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2013-01-11 13:54:57 +08:00
|
|
|
cocos2d::CCLayerMultiplex* ret = cocos2d::CCLayerMultiplex::createWithArray(arg0);
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCLayerMultiplex>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
2012-11-30 21:15:01 +08:00
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
|
|
|
|
JSBool js_cocos2dx_JSTouchDelegate_registerStandardDelegate(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSObject* jsobj = NULL;
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JSTouchDelegate *touch = new JSTouchDelegate();
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
touch->autorelease();
|
2012-08-28 10:22:36 +08:00
|
|
|
touch->registerStandardDelegate();
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
jsobj = (argc == 1 ? JSVAL_TO_OBJECT(argv[0]) : JSVAL_TO_OBJECT(JSVAL_VOID));
|
|
|
|
touch->setJSObject(jsobj);
|
|
|
|
JSTouchDelegate::setDelegateForJSObject(jsobj, touch);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting >= 1", argc);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_JSTouchDelegate_registerTargettedDelegate(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSObject* jsobj = NULL;
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JSTouchDelegate *touch = new JSTouchDelegate();
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
touch->autorelease();
|
2012-08-28 10:22:36 +08:00
|
|
|
touch->registerTargettedDelegate((argc >= 1 ? JSVAL_TO_INT(argv[0]) : 0), (argc >= 2 ? JSVAL_TO_BOOLEAN(argv[1]) : true));
|
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
jsobj = (argc == 3 ? JSVAL_TO_OBJECT(argv[2]) : JSVAL_TO_OBJECT(JSVAL_VOID));
|
|
|
|
touch->setJSObject(jsobj);
|
|
|
|
JSTouchDelegate::setDelegateForJSObject(jsobj, touch);
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting >=1", argc);
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSBool js_cocos2dx_JSTouchDelegate_unregisterTouchDelegate(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc == 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject* jsobj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JSTouchDelegate* pDelegate = JSTouchDelegate::getDelegateForJSObject(jsobj);
|
|
|
|
if (pDelegate)
|
|
|
|
{
|
|
|
|
pDelegate->unregisterTouchDelegate();
|
|
|
|
JSTouchDelegate::removeDelegateForJSObject(jsobj);
|
|
|
|
}
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
|
|
|
|
JSBool js_cocos2dx_swap_native_object(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc == 2) {
|
|
|
|
// get the native object from the second object to the first object
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *one = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JSObject *two = JSVAL_TO_OBJECT(argv[1]);
|
|
|
|
js_proxy_t *nproxy;
|
|
|
|
JS_GET_NATIVE_PROXY(nproxy, two);
|
|
|
|
void *ptrTwo = (nproxy ? nproxy->ptr : NULL);
|
|
|
|
if (nproxy) {
|
|
|
|
js_proxy_t *jsproxy;
|
|
|
|
JS_GET_PROXY(jsproxy, ptrTwo);
|
|
|
|
if (jsproxy) {
|
2012-09-19 09:44:21 +08:00
|
|
|
JS_RemoveObjectRoot(cx, &nproxy->obj);
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_REMOVE_PROXY(jsproxy, nproxy);
|
|
|
|
JS_NEW_PROXY(nproxy, ptrTwo, one);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
JS_AddNamedObjectRoot(cx, &nproxy->obj, typeid(*((CCObject*)nproxy->ptr)).name());
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCNode_copy(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc == 0) {
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
cocos2d::CCObject *node = (cocos2d::CCObject *)(proxy ? proxy->ptr : NULL);
|
2012-08-28 10:22:36 +08:00
|
|
|
TEST_NATIVE_OBJECT(cx, node)
|
|
|
|
cocos2d::CCObject *ret = node->copy();
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
proxy = js_get_or_create_proxy<cocos2d::CCObject>(cx, ret);
|
|
|
|
if (ret && proxy) {
|
|
|
|
ret->autorelease();
|
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(proxy->obj));
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSObject* getObjectFromNamespace(JSContext* cx, JSObject *ns, const char *name) {
|
|
|
|
jsval out;
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-08-28 10:22:36 +08:00
|
|
|
if (JS_GetProperty(cx, ns, name, &out) == JS_TRUE) {
|
|
|
|
JSObject *obj;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToObject(cx, out, &obj);
|
|
|
|
JSB_PRECONDITION2(ok, cx, NULL, "Error processing arguments");
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
jsval anonEvaluate(JSContext *cx, JSObject *thisObj, const char* string) {
|
|
|
|
jsval out;
|
|
|
|
if (JS_EvaluateScript(cx, thisObj, string, strlen(string), "(string)", 1, &out) == JS_TRUE) {
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
return JSVAL_VOID;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_platform(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSString *str = JS_NewStringCopyZ(cx, "mobile");
|
|
|
|
jsval out = STRING_TO_JSVAL(str);
|
|
|
|
JS_SET_RVAL(cx, vp, out);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSCallbackWrapper::JSCallbackWrapper()
|
|
|
|
: jsCallback(JSVAL_VOID), jsThisObj(JSVAL_VOID), extraData(JSVAL_VOID)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
JSCallbackWrapper::~JSCallbackWrapper()
|
|
|
|
{
|
|
|
|
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
|
|
JS_RemoveValueRoot(cx, &jsCallback);
|
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
void JSCallbackWrapper::setJSCallbackFunc(jsval func) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsCallback = func;
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSContext* cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
|
|
// Root the callback function.
|
|
|
|
JS_AddNamedValueRoot(cx, &jsCallback, "JSCallbackWrapper_callback_func");
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
void JSCallbackWrapper::setJSCallbackThis(jsval thisObj) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsThisObj = thisObj;
|
|
|
|
}
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
void JSCallbackWrapper::setJSExtraData(jsval data) {
|
2012-10-25 05:05:29 +08:00
|
|
|
extraData = data;
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
const jsval& JSCallbackWrapper::getJSCallbackFunc() const
|
|
|
|
{
|
|
|
|
return jsCallback;
|
|
|
|
}
|
|
|
|
|
|
|
|
const jsval& JSCallbackWrapper::getJSCallbackThis() const
|
|
|
|
{
|
|
|
|
return jsThisObj;
|
|
|
|
}
|
|
|
|
|
|
|
|
const jsval& JSCallbackWrapper::getJSExtraData() const
|
|
|
|
{
|
|
|
|
return extraData;
|
|
|
|
}
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
void JSCallFuncWrapper::setTargetForNativeNode(CCNode *pNode, JSCallFuncWrapper *target) {
|
2012-10-23 02:07:23 +08:00
|
|
|
callfuncTarget_proxy_t *t;
|
|
|
|
HASH_FIND_PTR(_callfuncTarget_native_ht, &pNode, t);
|
|
|
|
|
|
|
|
CCArray *arr;
|
|
|
|
if(!t) {
|
|
|
|
arr = new CCArray();
|
|
|
|
} else {
|
|
|
|
arr = t->obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
arr->addObject(target);
|
|
|
|
|
|
|
|
callfuncTarget_proxy_t *p = (callfuncTarget_proxy_t *)malloc(sizeof(callfuncTarget_proxy_t));
|
|
|
|
assert(p);
|
|
|
|
p->ptr = (void *)pNode;
|
|
|
|
p->obj = arr;
|
|
|
|
HASH_ADD_PTR(_callfuncTarget_native_ht, ptr, p);
|
|
|
|
}
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
CCArray * JSCallFuncWrapper::getTargetForNativeNode(CCNode *pNode) {
|
2012-10-23 02:07:23 +08:00
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
callfuncTarget_proxy_t *t;
|
2012-10-23 02:07:23 +08:00
|
|
|
HASH_FIND_PTR(_callfuncTarget_native_ht, &pNode, t);
|
|
|
|
if(!t) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return t->obj;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 18:44:28 +08:00
|
|
|
void JSCallFuncWrapper::callbackFunc(CCNode *node) const {
|
|
|
|
|
2012-11-30 18:15:29 +08:00
|
|
|
bool hasExtraData = !JSVAL_IS_VOID(extraData);
|
|
|
|
JSObject* thisObj = JSVAL_IS_VOID(jsThisObj) ? NULL : JSVAL_TO_OBJECT(jsThisObj);
|
2012-11-05 18:44:28 +08:00
|
|
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCNode>(cx, node);
|
2012-10-23 02:07:23 +08:00
|
|
|
|
2012-11-05 18:44:28 +08:00
|
|
|
jsval retval;
|
2012-11-30 18:15:29 +08:00
|
|
|
if(jsCallback != JSVAL_VOID)
|
|
|
|
{
|
|
|
|
if (hasExtraData)
|
|
|
|
{
|
|
|
|
jsval valArr[2];
|
|
|
|
valArr[0] = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
valArr[1] = extraData;
|
|
|
|
|
|
|
|
JS_AddValueRoot(cx, valArr);
|
|
|
|
JS_CallFunctionValue(cx, thisObj, jsCallback, 2, valArr, &retval);
|
|
|
|
JS_RemoveValueRoot(cx, valArr);
|
2012-11-05 18:44:28 +08:00
|
|
|
}
|
2012-11-30 18:15:29 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
jsval senderVal = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
JS_AddValueRoot(cx, &senderVal);
|
|
|
|
JS_CallFunctionValue(cx, thisObj, jsCallback, 1, &senderVal, &retval);
|
|
|
|
JS_RemoveValueRoot(cx, &senderVal);
|
2012-11-05 18:44:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
// I think the JSCallFuncWrapper isn't needed.
|
|
|
|
// Since an action will be run by a cc.Node, it will be released at the CCNode::cleanup.
|
|
|
|
// By James Chen
|
|
|
|
// JSCallFuncWrapper::setTargetForNativeNode(node, (JSCallFuncWrapper *)this);
|
2012-11-05 18:44:28 +08:00
|
|
|
}
|
|
|
|
|
2012-11-16 03:14:37 +08:00
|
|
|
// cc.CallFunc.create( func, this, [data])
|
2012-11-05 18:44:28 +08:00
|
|
|
// cc.CallFunc.create( func )
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_callFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
|
2012-11-05 18:44:28 +08:00
|
|
|
if (argc >= 1 && argc <= 3) {
|
2012-08-28 10:22:36 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
JSCallFuncWrapper *tmpCobj = new JSCallFuncWrapper();
|
2012-10-23 02:07:23 +08:00
|
|
|
tmpCobj->autorelease();
|
|
|
|
|
2012-11-05 18:44:28 +08:00
|
|
|
tmpCobj->setJSCallbackFunc(argv[0]);
|
2012-08-28 10:22:36 +08:00
|
|
|
if(argc >= 2) {
|
2012-11-05 18:44:28 +08:00
|
|
|
tmpCobj->setJSCallbackThis(argv[1]);
|
2012-08-28 10:22:36 +08:00
|
|
|
} if(argc == 3) {
|
2012-11-01 02:08:37 +08:00
|
|
|
tmpCobj->setJSExtraData(argv[2]);
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CCCallFunc *ret = (CCCallFunc *)CCCallFuncN::create((CCObject *)tmpCobj,
|
2012-11-01 02:08:37 +08:00
|
|
|
callfuncN_selector(JSCallFuncWrapper::callbackFunc));
|
2012-08-28 10:22:36 +08:00
|
|
|
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<cocos2d::CCCallFunc>(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(proxy->obj));
|
2012-10-23 02:07:23 +08:00
|
|
|
|
|
|
|
JS_SetReservedSlot(proxy->obj, 0, argv[0]);
|
2012-11-03 01:23:23 +08:00
|
|
|
if(argc > 1) {
|
|
|
|
JS_SetReservedSlot(proxy->obj, 1, argv[1]);
|
|
|
|
}
|
2012-10-27 06:31:52 +08:00
|
|
|
// if(argc == 3) {
|
|
|
|
// JS_SetReservedSlot(proxy->obj, 2, argv[2]);
|
|
|
|
// }
|
2012-10-23 02:07:23 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
// test->execute();
|
2012-11-05 18:44:28 +08:00
|
|
|
return JS_TRUE;
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_ReportError(cx, "Invalid number of arguments");
|
|
|
|
return JS_FALSE;
|
2012-08-28 10:22:36 +08:00
|
|
|
}
|
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
|
|
|
|
void JSScheduleWrapper::setTargetForSchedule(jsval sched, JSScheduleWrapper *target) {
|
2012-10-10 05:59:16 +08:00
|
|
|
do {
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
JSObject* jsfunc = JSVAL_TO_OBJECT(sched);
|
|
|
|
CCArray* targetArray = getTargetForSchedule(sched);
|
|
|
|
if (NULL == targetArray) {
|
|
|
|
targetArray = new CCArray();
|
|
|
|
schedFunc_proxy_t *p = (schedFunc_proxy_t *)malloc(sizeof(schedFunc_proxy_t));
|
|
|
|
assert(p);
|
|
|
|
p->jsfuncObj = jsfunc;
|
|
|
|
p->targets = targetArray;
|
|
|
|
HASH_ADD_PTR(_schedFunc_target_ht, jsfuncObj, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
CCAssert(!targetArray->containsObject(target), "The target was already added.");
|
|
|
|
|
|
|
|
targetArray->addObject(target);
|
2012-10-10 05:59:16 +08:00
|
|
|
} while(0);
|
|
|
|
}
|
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
CCArray * JSScheduleWrapper::getTargetForSchedule(jsval sched) {
|
|
|
|
schedFunc_proxy_t *t = NULL;
|
2012-10-10 05:59:16 +08:00
|
|
|
JSObject *o = JSVAL_TO_OBJECT(sched);
|
|
|
|
HASH_FIND_PTR(_schedFunc_target_ht, &o, t);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
return t != NULL ? t->targets : NULL;
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
void JSScheduleWrapper::setTargetForNativeNode(CCNode *pNode, JSScheduleWrapper *target)
|
|
|
|
{
|
|
|
|
CCArray* targetArray = getTargetForNativeNode(pNode);
|
|
|
|
if (NULL == targetArray) {
|
|
|
|
targetArray = new CCArray();
|
2012-10-10 05:59:16 +08:00
|
|
|
schedTarget_proxy_t *p = (schedTarget_proxy_t *)malloc(sizeof(schedTarget_proxy_t));
|
|
|
|
assert(p);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
p->nativeObj = pNode;
|
|
|
|
p->targets = targetArray;
|
|
|
|
HASH_ADD_PTR(_schedTarget_native_ht, nativeObj, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
CCAssert(!targetArray->containsObject(target), "The target was already added.");
|
|
|
|
targetArray->addObject(target);
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
CCArray * JSScheduleWrapper::getTargetForNativeNode(CCNode *pNode)
|
|
|
|
{
|
|
|
|
schedTarget_proxy_t *t = NULL;
|
2012-10-10 05:59:16 +08:00
|
|
|
HASH_FIND_PTR(_schedTarget_native_ht, &pNode, t);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
return t != NULL ? t->targets : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSScheduleWrapper::removeAllTargetsForNatiaveNode(CCNode* pNode)
|
|
|
|
{
|
2012-12-03 15:31:52 +08:00
|
|
|
CCLOGINFO("removeAllTargetsForNatiaveNode begin");
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
dump();
|
|
|
|
CCArray* removeNativeTargets = NULL;
|
|
|
|
schedTarget_proxy_t *t = NULL;
|
|
|
|
HASH_FIND_PTR(_schedTarget_native_ht, &pNode, t);
|
|
|
|
if (t != NULL) {
|
|
|
|
removeNativeTargets = t->targets;
|
|
|
|
HASH_DEL(_schedTarget_native_ht, t);
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
|
|
|
|
if (removeNativeTargets == NULL) return;
|
|
|
|
|
|
|
|
schedFunc_proxy_t *current, *tmp;
|
|
|
|
HASH_ITER(hh, _schedFunc_target_ht, current, tmp) {
|
|
|
|
std::vector<CCObject*> objectsNeedToBeReleased;
|
|
|
|
JSObject* key = current->jsfuncObj;
|
|
|
|
CCArray* targets = current->targets;
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(targets, pObj)
|
|
|
|
{
|
|
|
|
if (removeNativeTargets->containsObject(pObj))
|
|
|
|
{
|
|
|
|
objectsNeedToBeReleased.push_back(pObj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<CCObject*>::iterator iter = objectsNeedToBeReleased.begin();
|
|
|
|
for (; iter != objectsNeedToBeReleased.end(); ++iter)
|
|
|
|
{
|
|
|
|
targets->removeObject(*iter, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (targets->count() == 0)
|
|
|
|
{
|
|
|
|
HASH_DEL(_schedFunc_target_ht, current);
|
2012-12-03 22:17:53 +08:00
|
|
|
targets->release();
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
free(current);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
removeNativeTargets->removeAllObjects();
|
|
|
|
removeNativeTargets->release();
|
|
|
|
free(t);
|
|
|
|
dump();
|
2012-12-03 15:31:52 +08:00
|
|
|
CCLOGINFO("removeAllTargetsForNatiaveNode end");
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
}
|
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
void JSScheduleWrapper::removeTargetForNativeNode(CCNode* pNode, JSScheduleWrapper* target)
|
|
|
|
{
|
|
|
|
schedTarget_proxy_t *t = NULL;
|
|
|
|
HASH_FIND_PTR(_schedTarget_native_ht, &pNode, t);
|
|
|
|
if (t != NULL) {
|
|
|
|
t->targets->removeObject(target);
|
|
|
|
if (t->targets->count() == 0)
|
|
|
|
{
|
|
|
|
t->targets->release();
|
|
|
|
HASH_DEL(_schedTarget_native_ht, t);
|
|
|
|
free(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
schedFunc_proxy_t *current, *tmp, *removed=NULL;
|
|
|
|
|
|
|
|
HASH_ITER(hh, _schedFunc_target_ht, current, tmp) {
|
|
|
|
CCArray* targets = current->targets;
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
|
|
|
|
CCARRAY_FOREACH(targets, pObj)
|
|
|
|
{
|
|
|
|
JSScheduleWrapper* pOneTarget = (JSScheduleWrapper*)pObj;
|
|
|
|
if (pOneTarget == target)
|
|
|
|
{
|
|
|
|
removed = current;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (removed) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (removed)
|
|
|
|
{
|
|
|
|
removed->targets->removeObject(target);
|
|
|
|
if (removed->targets->count() == 0)
|
|
|
|
{
|
|
|
|
removed->targets->release();
|
|
|
|
HASH_DEL(_schedFunc_target_ht, removed);
|
|
|
|
free(removed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dump();
|
|
|
|
}
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
void JSScheduleWrapper::dump()
|
|
|
|
{
|
2012-12-03 15:31:52 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
CCLOG("\n---------JSScheduleWrapper dump begin--------------\n");
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
schedTarget_proxy_t *current, *tmp;
|
|
|
|
int nativeTargetsCount = 0;
|
|
|
|
HASH_ITER(hh, _schedTarget_native_ht, current, tmp) {
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(current->targets, pObj)
|
|
|
|
{
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
CCLOG("native %s ( %p ), target[%d]=( %p )", typeid(*current->nativeObj).name(), current->nativeObj, nativeTargetsCount, pObj);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
nativeTargetsCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CCLOG("\n-----------------------------\n");
|
|
|
|
|
|
|
|
schedFunc_proxy_t *current_func, *tmp_func;
|
|
|
|
int jsfuncTargetCount = 0;
|
|
|
|
HASH_ITER(hh, _schedFunc_target_ht, current_func, tmp_func) {
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(current_func->targets, pObj)
|
|
|
|
{
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
CCLOG("jsfunc ( %p ), target[%d]=( %p )", current_func->jsfuncObj, jsfuncTargetCount, pObj);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
jsfuncTargetCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CCAssert(nativeTargetsCount == jsfuncTargetCount, "");
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
CCLOG("\n---------JSScheduleWrapper dump end--------------\n");
|
2012-12-03 15:31:52 +08:00
|
|
|
#endif
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
2012-11-22 12:05:38 +08:00
|
|
|
void JSScheduleWrapper::scheduleFunc(float dt) const
|
|
|
|
{
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
jsval retval = JSVAL_NULL;
|
|
|
|
jsval data = DOUBLE_TO_JSVAL(dt);
|
|
|
|
|
2012-11-22 12:05:38 +08:00
|
|
|
JSContext *cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
|
|
|
|
|
|
JSBool ok = JS_AddValueRoot(cx, &data);
|
2013-01-11 18:57:33 +08:00
|
|
|
if (!ok) {
|
|
|
|
CCLOG("scheduleFunc: Root value fails.");
|
2012-11-22 12:05:38 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-11-29 07:56:57 +08:00
|
|
|
if(!jsCallback.isNullOrUndefined() || !jsThisObj.isNullOrUndefined()) {
|
|
|
|
JSAutoCompartment ac(cx, JSVAL_TO_OBJECT(jsThisObj));
|
|
|
|
|
2012-11-22 12:05:38 +08:00
|
|
|
JS_CallFunctionValue(cx, JSVAL_TO_OBJECT(jsThisObj), jsCallback, 1, &data, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_RemoveValueRoot(cx, &data);
|
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
CCObject* JSScheduleWrapper::getTarget()
|
|
|
|
{
|
|
|
|
return m_pTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSScheduleWrapper::setTarget(CCObject* pTarget)
|
|
|
|
{
|
|
|
|
m_pTarget = pTarget;
|
|
|
|
}
|
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
JSBool js_CCNode_unschedule(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc == 1) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCNode *node = (cocos2d::CCNode *)(proxy ? proxy->ptr : NULL);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(node, cx, JS_FALSE, "Invalid Native Object");
|
2012-10-10 05:59:16 +08:00
|
|
|
|
|
|
|
CCScheduler *sched = node->getScheduler();
|
|
|
|
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
CCArray* targetArray = JSScheduleWrapper::getTargetForSchedule(argv[0]);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
CCLOGINFO("unschedule target number: %d", targetArray->count());
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
CCObject* tmp = NULL;
|
|
|
|
CCARRAY_FOREACH(targetArray, tmp)
|
|
|
|
{
|
|
|
|
JSScheduleWrapper* target = (JSScheduleWrapper*)tmp;
|
|
|
|
if (node == target->getTarget())
|
|
|
|
{
|
|
|
|
sched->unscheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), target);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSScheduleWrapper::removeTargetForNativeNode(node, target);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
|
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
}
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
JSBool js_cocos2dx_CCNode_unscheduleAllSelectors(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCNode* cobj = (cocos2d::CCNode *)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
if (argc == 0)
|
|
|
|
{
|
|
|
|
cobj->unscheduleAllSelectors();
|
|
|
|
|
|
|
|
CCArray *arr = JSScheduleWrapper::getTargetForNativeNode(cobj);
|
|
|
|
// If there aren't any targets, just return true.
|
|
|
|
// Otherwise, the for loop will break immediately.
|
|
|
|
// It will lead to logic errors.
|
|
|
|
// For details to reproduce it, please refer to SchedulerTest/SchedulerUpdate.
|
|
|
|
if(! arr) return JS_TRUE;
|
|
|
|
for(unsigned int i = 0; i < arr->count(); ++i) {
|
|
|
|
if(arr->objectAtIndex(i)) {
|
|
|
|
cobj->getScheduler()->unscheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), arr->objectAtIndex(i));
|
|
|
|
}
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2012-11-03 01:23:23 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 0);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-07 03:04:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
JSBool js_cocos2dx_CCScheduler_unscheduleAllSelectorsForTarget(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCScheduler* cobj = (cocos2d::CCScheduler *)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCNode* arg0;
|
|
|
|
do {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
arg0 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg0)
|
2012-12-03 15:31:52 +08:00
|
|
|
cobj->unscheduleAllForTarget(arg0);
|
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
CCArray *arr = JSScheduleWrapper::getTargetForNativeNode(arg0);
|
2012-12-03 15:31:52 +08:00
|
|
|
// If there aren't any targets, just return true.
|
|
|
|
// Otherwise, the for loop will break immediately.
|
|
|
|
// It will lead to logic errors.
|
|
|
|
// For details to reproduce it, please refer to SchedulerTest/SchedulerUpdate.
|
|
|
|
if(! arr) return JS_TRUE;
|
2012-11-03 01:23:23 +08:00
|
|
|
for(unsigned int i = 0; i < arr->count(); ++i) {
|
|
|
|
if(arr->objectAtIndex(i)) {
|
2012-11-16 19:38:25 +08:00
|
|
|
arg0->getScheduler()->unscheduleAllForTarget(arr->objectAtIndex(i));
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
|
|
|
}
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JSScheduleWrapper::removeAllTargetsForNatiaveNode(arg0);
|
2012-11-03 01:23:23 +08:00
|
|
|
|
|
|
|
} while (0);
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
JSBool js_CCNode_scheduleOnce(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 1) {
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-10 05:59:16 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCNode *node = (cocos2d::CCNode *)(proxy ? proxy->ptr : NULL);
|
|
|
|
|
|
|
|
CCScheduler *sched = node->getScheduler();
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
JSScheduleWrapper *tmpCobj = NULL;
|
2012-11-02 18:15:25 +08:00
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
//
|
|
|
|
// delay
|
|
|
|
//
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
double delay;
|
2012-10-10 05:59:16 +08:00
|
|
|
if( argc >= 2 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &delay );
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
bool bFound = false;
|
|
|
|
CCArray* pTargetArr = JSScheduleWrapper::getTargetForNativeNode(node);
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(pTargetArr, pObj)
|
|
|
|
{
|
|
|
|
JSScheduleWrapper* pTarget = (JSScheduleWrapper*)pObj;
|
|
|
|
if (argv[0] == pTarget->getJSCallbackFunc())
|
|
|
|
{
|
|
|
|
tmpCobj = pTarget;
|
|
|
|
bFound = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bFound)
|
|
|
|
{
|
|
|
|
tmpCobj = new JSScheduleWrapper();
|
|
|
|
tmpCobj->autorelease();
|
|
|
|
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
|
|
|
tmpCobj->setJSCallbackFunc(argv[0]);
|
|
|
|
tmpCobj->setTarget(node);
|
|
|
|
|
|
|
|
JSScheduleWrapper::setTargetForSchedule(argv[0], tmpCobj);
|
|
|
|
JSScheduleWrapper::setTargetForNativeNode(node, tmpCobj);
|
|
|
|
}
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
if(argc == 1) {
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, 0, 0.0f, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
} else {
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, 0, delay, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
2012-11-30 22:00:30 +08:00
|
|
|
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
/* We shouldn't set the js callback function to reserved slot,
|
|
|
|
since the target object may execute more than one schedule.
|
|
|
|
Therefore, previous js callback function will be replaced
|
|
|
|
by the current one. For example:
|
|
|
|
this.scheduleOnce(function() { temporary function 1 }, 0.5);
|
|
|
|
this.scheduleOnce(function() { temporary function 2 }, 0.5);
|
|
|
|
In this case, the temporary function 1 will be removed from reserved slot 0.
|
|
|
|
And temporary function 2 will be set to reserved slot 0 of this object.
|
|
|
|
If gc is triggered before the 'JSScheduleWrapper::scheduleFunc' is invoked,
|
|
|
|
crash will happen. You could simply reproduce it by adding '__jsc__.garbageCollect();' after scheduleOnce.
|
|
|
|
|
|
|
|
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
|
|
|
|
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
|
|
|
|
at the destructor of JSScheduleWrapper.
|
|
|
|
*/
|
|
|
|
//jsb_set_reserved_slot(proxy->obj, 0, argv[0]);
|
2012-11-30 22:00:30 +08:00
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2013-01-11 18:57:33 +08:00
|
|
|
return JS_TRUE;
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
|
|
|
return JS_FALSE;
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_CCNode_schedule(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc >= 1) {
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-10 05:59:16 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCNode *node = (cocos2d::CCNode *)(proxy ? proxy->ptr : NULL);
|
|
|
|
CCScheduler *sched = node->getScheduler();
|
2012-12-03 15:31:52 +08:00
|
|
|
|
2012-10-27 06:31:52 +08:00
|
|
|
js_proxy_t *p = js_get_or_create_proxy<cocos2d::CCScheduler>(cx, sched);
|
2012-10-10 05:59:16 +08:00
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
JSScheduleWrapper *tmpCobj = NULL;
|
2012-11-01 16:06:05 +08:00
|
|
|
|
2012-11-22 12:05:38 +08:00
|
|
|
double interval = 0.0;
|
2012-10-10 05:59:16 +08:00
|
|
|
if( argc >= 2 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &interval );
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// repeat
|
|
|
|
//
|
2012-11-22 12:05:38 +08:00
|
|
|
double repeat = 0.0;
|
2012-10-10 05:59:16 +08:00
|
|
|
if( argc >= 3 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[2], &repeat );
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// delay
|
|
|
|
//
|
2012-11-22 12:05:38 +08:00
|
|
|
double delay = 0.0;
|
2012-10-10 05:59:16 +08:00
|
|
|
if( argc >= 4 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[3], &delay );
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
2013-01-11 18:57:33 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
bool bFound = false;
|
|
|
|
CCArray* pTargetArr = JSScheduleWrapper::getTargetForNativeNode(node);
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(pTargetArr, pObj)
|
|
|
|
{
|
|
|
|
JSScheduleWrapper* pTarget = (JSScheduleWrapper*)pObj;
|
|
|
|
if (argv[0] == pTarget->getJSCallbackFunc())
|
|
|
|
{
|
|
|
|
tmpCobj = pTarget;
|
|
|
|
bFound = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
if (!bFound)
|
|
|
|
{
|
|
|
|
tmpCobj = new JSScheduleWrapper();
|
|
|
|
tmpCobj->autorelease();
|
|
|
|
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(obj));
|
|
|
|
tmpCobj->setJSCallbackFunc(argv[0]);
|
|
|
|
tmpCobj->setTarget(node);
|
|
|
|
JSScheduleWrapper::setTargetForSchedule(argv[0], tmpCobj);
|
|
|
|
JSScheduleWrapper::setTargetForNativeNode(node, tmpCobj);
|
|
|
|
}
|
2012-10-10 05:59:16 +08:00
|
|
|
|
|
|
|
if(argc == 1) {
|
2012-11-07 03:04:02 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
} if(argc == 2) {
|
2012-11-07 03:04:02 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
} if(argc == 3) {
|
2012-11-22 12:05:38 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, (unsigned int)repeat, 0, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
} if (argc == 4) {
|
2012-11-22 12:05:38 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, (unsigned int)repeat, delay, !node->isRunning());
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
|
|
|
|
// I comment next line with the same reason in the js_CCNode_scheduleOnce.
|
|
|
|
//jsb_set_reserved_slot(proxy->obj, 0, argv[0]);
|
2012-10-27 06:31:52 +08:00
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2013-01-11 18:57:33 +08:00
|
|
|
return JS_TRUE;
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
|
|
|
return JS_FALSE;
|
2012-10-10 05:59:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
JSBool js_CCScheduler_schedule(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (argc >= 2) {
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-03 01:23:23 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCScheduler *sched = (cocos2d::CCScheduler *)(proxy ? proxy->ptr : NULL);
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
JSScheduleWrapper *tmpCobj = NULL;
|
2012-11-03 01:23:23 +08:00
|
|
|
|
|
|
|
cocos2d::CCNode* node;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
node = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, node)
|
|
|
|
|
2012-11-07 03:04:02 +08:00
|
|
|
double interval = 0;
|
2012-11-03 01:23:23 +08:00
|
|
|
if( argc >= 3 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[2], &interval );
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// repeat
|
|
|
|
//
|
2012-11-07 03:04:02 +08:00
|
|
|
double repeat = -1;
|
2012-11-03 01:23:23 +08:00
|
|
|
if( argc >= 4 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[3], &repeat );
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// delay
|
|
|
|
//
|
2012-11-07 03:04:02 +08:00
|
|
|
double delay = 0;
|
2012-11-03 01:23:23 +08:00
|
|
|
if( argc >= 5 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[4], &delay );
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
2012-11-07 03:04:02 +08:00
|
|
|
|
|
|
|
JSBool paused = !node->isRunning();
|
|
|
|
|
|
|
|
if( argc >= 6 ) {
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToBoolean(cx, argv[4], &paused);
|
2012-11-07 03:04:02 +08:00
|
|
|
}
|
|
|
|
|
2013-01-11 18:57:33 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
bool bFound = false;
|
|
|
|
CCArray* pTargetArr = JSScheduleWrapper::getTargetForNativeNode(node);
|
|
|
|
CCObject* pObj = NULL;
|
|
|
|
CCARRAY_FOREACH(pTargetArr, pObj)
|
|
|
|
{
|
|
|
|
JSScheduleWrapper* pTarget = (JSScheduleWrapper*)pObj;
|
2012-12-03 15:59:29 +08:00
|
|
|
if (argv[1] == pTarget->getJSCallbackFunc())
|
2012-12-03 15:31:52 +08:00
|
|
|
{
|
|
|
|
tmpCobj = pTarget;
|
|
|
|
bFound = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bFound)
|
|
|
|
{
|
|
|
|
tmpCobj = new JSScheduleWrapper();
|
|
|
|
tmpCobj->autorelease();
|
|
|
|
tmpCobj->setJSCallbackThis(OBJECT_TO_JSVAL(proxy->obj));
|
|
|
|
tmpCobj->setJSCallbackFunc(argv[1]);
|
|
|
|
|
|
|
|
JSScheduleWrapper::setTargetForSchedule(argv[1], tmpCobj);
|
|
|
|
JSScheduleWrapper::setTargetForNativeNode(node, tmpCobj);
|
|
|
|
}
|
2012-11-03 01:23:23 +08:00
|
|
|
|
2012-12-03 15:31:52 +08:00
|
|
|
sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, repeat, delay, paused);
|
2012-11-07 03:04:02 +08:00
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2013-01-11 18:57:33 +08:00
|
|
|
return JS_TRUE;
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
|
|
|
return JS_FALSE;
|
2012-11-03 01:23:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCScheduler_pauseTarget(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCNode* arg0;
|
|
|
|
do {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
arg0 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg0)
|
|
|
|
CCArray *arr = JSScheduleWrapper::getTargetForNativeNode(arg0);
|
2012-12-03 15:31:52 +08:00
|
|
|
if(! arr) return JS_TRUE;
|
2012-11-03 01:23:23 +08:00
|
|
|
for(unsigned int i = 0; i < arr->count(); ++i) {
|
|
|
|
if(arr->objectAtIndex(i)) {
|
|
|
|
arg0->getScheduler()->pauseTarget(arr->objectAtIndex(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (0);
|
|
|
|
//cobj->pauseTarget(arg0);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCScheduler_resumeTarget(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCNode* arg0;
|
|
|
|
do {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JSObject *tmpObj = JSVAL_TO_OBJECT(argv[0]);
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, tmpObj);
|
|
|
|
arg0 = (cocos2d::CCNode*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, arg0)
|
|
|
|
CCArray *arr = JSScheduleWrapper::getTargetForNativeNode(arg0);
|
2012-12-03 15:31:52 +08:00
|
|
|
if(! arr) return JS_TRUE;
|
2012-11-03 01:23:23 +08:00
|
|
|
for(unsigned int i = 0; i < arr->count(); ++i) {
|
|
|
|
if(arr->objectAtIndex(i)) {
|
|
|
|
arg0->getScheduler()->resumeTarget(arr->objectAtIndex(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (0);
|
|
|
|
//cobj->pauseTarget(arg0);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
|
|
|
|
JSBool js_doNothing(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JSBool js_forceGC(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
JSRuntime *rt = JS_GetRuntime(cx);
|
|
|
|
JS_GC(rt);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_retain(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *thisObj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
if (thisObj) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, thisObj);
|
|
|
|
if (proxy) {
|
|
|
|
((CCObject *)proxy->ptr)->retain();
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "Invalid Native Object.");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_release(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *thisObj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
if (thisObj) {
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_GET_NATIVE_PROXY(proxy, thisObj);
|
|
|
|
if (proxy) {
|
|
|
|
((CCObject *)proxy->ptr)->release();
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "Invalid Native Object.");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSet_constructor(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *obj;
|
|
|
|
cocos2d::CCSet* cobj;
|
|
|
|
|
|
|
|
if (argc == 0) {
|
|
|
|
cobj = new cocos2d::CCSet();
|
|
|
|
cobj->autorelease();
|
|
|
|
TypeTest<cocos2d::CCSet> t;
|
|
|
|
js_type_class_t *typeClass;
|
|
|
|
uint32_t typeId = t.s_id();
|
|
|
|
HASH_FIND_INT(_js_global_type_ht, &typeId, typeClass);
|
|
|
|
assert(typeClass);
|
|
|
|
obj = JS_NewObject(cx, typeClass->jsclass, typeClass->proto, typeClass->parentProto);
|
|
|
|
js_proxy_t *proxy;
|
|
|
|
JS_NEW_PROXY(proxy, cobj, obj);
|
|
|
|
JS_AddNamedObjectRoot(cx, &proxy->obj, typeid(cobj).name());
|
|
|
|
}
|
|
|
|
if (cobj) {
|
|
|
|
JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
|
|
|
|
JS_ReportError(cx, "Error in js_cocos2dx_CCSet_constructor");
|
2012-08-28 10:22:36 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
JSBool js_cocos2dx_CCNode_setPosition(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCNode* cobj = (cocos2d::CCNode *)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
cobj->setPosition(arg0);
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
} if (argc == 2) {
|
|
|
|
double x;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[0], &x );
|
2012-10-12 09:43:40 +08:00
|
|
|
double y;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &y );
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
|
|
|
cobj->setPosition(ccp(x,y));
|
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSprite_setPosition(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCSprite* cobj = (cocos2d::CCSprite *)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-12 09:43:40 +08:00
|
|
|
cobj->setPosition(arg0);
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
} if (argc == 2) {
|
|
|
|
double x;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[0], &x );
|
2012-10-12 09:43:40 +08:00
|
|
|
double y;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &y );
|
|
|
|
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-12 09:43:40 +08:00
|
|
|
cobj->setPosition(CCPoint(x,y));
|
2013-01-11 18:57:33 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-07 03:04:02 +08:00
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCTMXLayer_tileFlagsAt(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj;
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-07 03:04:02 +08:00
|
|
|
cocos2d::CCTMXLayer* cobj;
|
|
|
|
obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cobj = (cocos2d::CCTMXLayer *)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-11-07 03:04:02 +08:00
|
|
|
cocos2d::ccTMXTileFlags flags;
|
|
|
|
unsigned int ret = cobj->tileGIDAt(arg0, &flags);
|
|
|
|
jsval jsret;
|
|
|
|
jsret = UINT_TO_JSVAL((uint32_t)flags);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments");
|
2012-11-07 03:04:02 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
template<class T>
|
|
|
|
JSBool js_BezierActions_create(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
if (argc == 2) {
|
|
|
|
double t;
|
|
|
|
if( ! JS_ValueToNumber(cx, argv[0], &t) ) {
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
int num;
|
|
|
|
CCPoint *arr;
|
|
|
|
jsval_to_ccarray_of_CCPoint(cx, argv[1], &arr, &num);
|
|
|
|
|
|
|
|
ccBezierConfig config;
|
|
|
|
config.controlPoint_1 = arr[0];
|
|
|
|
config.controlPoint_2 = arr[1];
|
|
|
|
config.endPosition = arr[2];
|
|
|
|
|
2012-11-22 15:19:16 +08:00
|
|
|
T* ret = T::create(t, config);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
free(arr);
|
|
|
|
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
2012-11-22 15:19:16 +08:00
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<T>(cx, ret);
|
2012-10-12 09:43:40 +08:00
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
JSBool js_CardinalSplineActions_create(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
if (argc == 3) {
|
|
|
|
double dur;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[0], &dur);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
int num;
|
|
|
|
CCPoint *arr;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= jsval_to_ccarray_of_CCPoint(cx, argv[1], &arr, &num);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
double ten;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[2], &ten);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
2013-01-11 18:57:33 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
CCPointArray *points = CCPointArray::create(num);
|
|
|
|
|
|
|
|
for( int i=0; i < num;i++) {
|
|
|
|
points->addControlPoint(arr[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
T *ret = T::create(dur, points, ten);
|
|
|
|
|
|
|
|
free(arr);
|
|
|
|
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<T>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
JSBool js_CatmullRomActions_create(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 18:57:33 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
if (argc == 2) {
|
|
|
|
double dur;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[0], &dur);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
int num;
|
|
|
|
CCPoint *arr;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= jsval_to_ccarray_of_CCPoint(cx, argv[1], &arr, &num);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
2013-01-11 18:57:33 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
CCPointArray *points = CCPointArray::create(num);
|
|
|
|
|
|
|
|
for( int i=0; i < num;i++) {
|
|
|
|
points->addControlPoint(arr[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
T *ret = T::create(dur, points);
|
|
|
|
|
|
|
|
free(arr);
|
|
|
|
|
|
|
|
jsval jsret;
|
|
|
|
do {
|
|
|
|
if (ret) {
|
|
|
|
js_proxy_t *p;
|
|
|
|
JS_GET_PROXY(p, ret);
|
|
|
|
if (p) {
|
|
|
|
jsret = OBJECT_TO_JSVAL(p->obj);
|
|
|
|
} else {
|
|
|
|
// create a new js obj of that class
|
|
|
|
js_proxy_t *proxy = js_get_or_create_proxy<T>(cx, ret);
|
|
|
|
jsret = OBJECT_TO_JSVAL(proxy->obj);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
jsret = JSVAL_NULL;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool JSB_CCBezierBy_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
2012-10-12 15:41:45 +08:00
|
|
|
return js_BezierActions_create<cocos2d::CCBezierBy>(cx, argc, vp);
|
2012-10-12 09:43:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
JSBool JSB_CCBezierTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
2012-10-12 15:41:45 +08:00
|
|
|
return js_BezierActions_create<cocos2d::CCBezierTo>(cx, argc, vp);
|
2012-10-12 09:43:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool JSB_CCCardinalSplineBy_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return js_CardinalSplineActions_create<cocos2d::CCCardinalSplineBy>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool JSB_CCCardinalSplineTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return js_CardinalSplineActions_create<cocos2d::CCCardinalSplineTo>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool JSB_CCCatmullRomBy_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return js_CatmullRomActions_create<cocos2d::CCCatmullRomBy>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool JSB_CCCatmullRomTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) {
|
|
|
|
return js_CatmullRomActions_create<cocos2d::CCCatmullRomTo>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpAdd(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpAdd(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-01 02:08:37 +08:00
|
|
|
JSBool js_cocos2dx_ccpDistance(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-01 02:08:37 +08:00
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
|
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-11-01 02:08:37 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2012-11-01 02:08:37 +08:00
|
|
|
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-11-01 02:08:37 +08:00
|
|
|
float ret = ccpDistance(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = DOUBLE_TO_JSVAL(ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-10-27 06:31:52 +08:00
|
|
|
JSBool js_cocos2dx_ccpClamp(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-27 06:31:52 +08:00
|
|
|
if (argc == 3) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-27 06:31:52 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2012-10-27 06:31:52 +08:00
|
|
|
cocos2d::CCPoint arg2;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[2], &arg2);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2013-01-11 13:54:57 +08:00
|
|
|
|
2012-10-27 06:31:52 +08:00
|
|
|
CCPoint ret = ccpClamp(arg0, arg1, arg2);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JSBool js_cocos2dx_ccpLengthSQ(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-05 23:22:59 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-11-05 23:22:59 +08:00
|
|
|
float ret = ccpLengthSQ(arg0);
|
|
|
|
|
|
|
|
jsval jsret = DOUBLE_TO_JSVAL(ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpLength(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-05 23:22:59 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-11-05 23:22:59 +08:00
|
|
|
|
|
|
|
float ret = ccpLength(arg0);
|
|
|
|
|
|
|
|
jsval jsret = DOUBLE_TO_JSVAL(ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
2012-10-27 06:31:52 +08:00
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
JSBool js_cocos2dx_ccpNeg(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2013-01-11 13:54:57 +08:00
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpNeg(arg0);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpSub(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpSub(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpMult(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
2013-01-11 13:54:57 +08:00
|
|
|
cocos2d::CCPoint arg0;
|
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
double arg1;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_ValueToNumber(cx, argv[1], &arg1);
|
|
|
|
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-10-12 09:43:40 +08:00
|
|
|
|
|
|
|
CCPoint ret = ccpMult(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpMidpoint(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpMidpoint(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpDot(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
float ret = ccpDot(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = DOUBLE_TO_JSVAL(ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpCross(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
float ret = ccpCross(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = DOUBLE_TO_JSVAL(ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpPerp(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpPerp(arg0);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpRPerp(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpRPerp(arg0);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpProject(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpProject(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpRotate(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 2) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2012-10-12 09:43:40 +08:00
|
|
|
cocos2d::CCPoint arg1;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[1], &arg1);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpRotate(arg0, arg1);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
2012-10-12 09:43:40 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_ccpNormalize(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-10-12 09:43:40 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
cocos2d::CCPoint arg0;
|
2013-01-11 13:54:57 +08:00
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
CCPoint ret = ccpNormalize(arg0);
|
|
|
|
|
|
|
|
jsval jsret = ccpoint_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-07 03:04:02 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCNode_prototype;
|
2012-10-12 09:43:40 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCLayerColor_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCSprite_prototype;
|
2012-11-02 15:21:44 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCTMXLayer_prototype;
|
2012-08-28 10:22:36 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCAction_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCAnimation_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCMenuItem_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCSpriteFrame_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCSet_prototype;
|
2012-10-10 15:25:27 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCSprite_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCSpriteBatchNode_prototype;
|
|
|
|
//extern JSObject* js_cocos2dx_CCMotionStreak_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCAtlasNode_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCParticleBatchNode_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCParticleSystem_prototype;
|
2012-10-12 09:43:40 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCCatmullRomBy_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCCatmullRomTo_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCCardinalSplineTo_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCCardinalSplineBy_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCBezierTo_prototype;
|
|
|
|
extern JSObject* js_cocos2dx_CCBezierBy_prototype;
|
2012-11-03 01:23:23 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCScheduler_prototype;
|
2012-11-22 15:19:16 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCDrawNode_prototype;
|
2012-11-30 18:15:29 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCTexture2D_prototype;
|
2012-11-30 21:15:01 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCMenu_prototype;
|
2013-01-26 22:31:57 +08:00
|
|
|
extern JSObject* js_cocos2dx_CCFileUtils_prototype;
|
2012-10-10 15:25:27 +08:00
|
|
|
|
2012-10-25 05:05:29 +08:00
|
|
|
// setBlendFunc
|
|
|
|
template<class T>
|
|
|
|
JSBool js_cocos2dx_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSObject *obj;
|
|
|
|
T* cobj;
|
|
|
|
obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cobj = (T*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
2012-10-10 15:25:27 +08:00
|
|
|
if (argc == 2)
|
2012-10-25 05:05:29 +08:00
|
|
|
{
|
|
|
|
GLenum src, dst;
|
2013-01-11 17:44:39 +08:00
|
|
|
jsval_to_int32(cx, argv[0], (int32_t*)&src);
|
|
|
|
jsval_to_int32(cx, argv[1], (int32_t*)&dst);
|
2012-10-10 15:25:27 +08:00
|
|
|
ccBlendFunc blendFunc = {src, dst};
|
|
|
|
cobj->setBlendFunc(blendFunc);
|
|
|
|
return JS_TRUE;
|
2012-10-25 05:05:29 +08:00
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSprite_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCSprite>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCSpriteBatchNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCSpriteBatchNode>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
2012-10-10 15:25:27 +08:00
|
|
|
// JSBool js_cocos2dx_CCMotionStreak_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
// {
|
|
|
|
// return js_cocos2dx_setBlendFunc<CCMotionStreak>(cx, argc, vp);
|
|
|
|
// }
|
2012-10-25 05:05:29 +08:00
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCAtlasNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCAtlasNode>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCParticleBatchNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCParticleBatchNode>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCLayerColor_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCLayerColor>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCParticleSystem_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCParticleSystem>(cx, argc, vp);
|
2012-10-10 15:25:27 +08:00
|
|
|
}
|
2012-08-28 10:22:36 +08:00
|
|
|
|
2012-11-22 15:19:16 +08:00
|
|
|
JSBool js_cocos2dx_CCDrawNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
return js_cocos2dx_setBlendFunc<CCDrawNode>(cx, argc, vp);
|
|
|
|
}
|
|
|
|
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool js_cocos2dx_CCTexture2D_setTexParameters(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject* obj = (JSObject *)JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
CCTexture2D* cobj = (CCTexture2D*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
2012-11-30 18:15:29 +08:00
|
|
|
if (argc == 4)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
jsval *argvp = JS_ARGV(cx,vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
|
|
|
|
GLint arg0, arg1, arg2, arg3;
|
|
|
|
|
2013-01-11 17:44:39 +08:00
|
|
|
ok &= jsval_to_int32(cx, *argvp++, &arg0);
|
|
|
|
ok &= jsval_to_int32(cx, *argvp++, &arg1);
|
|
|
|
ok &= jsval_to_int32(cx, *argvp++, &arg2);
|
|
|
|
ok &= jsval_to_int32(cx, *argvp++, &arg3);
|
2013-01-11 13:54:57 +08:00
|
|
|
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2013-01-11 13:54:57 +08:00
|
|
|
|
|
|
|
ccTexParams param = { arg0, arg1, arg2, arg3 };
|
|
|
|
|
|
|
|
cobj->setTexParameters(¶m);
|
|
|
|
|
2012-11-30 18:15:29 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
2013-01-11 13:54:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 4);
|
|
|
|
return JS_FALSE;
|
2012-11-30 18:15:29 +08:00
|
|
|
}
|
|
|
|
|
2012-11-30 21:15:01 +08:00
|
|
|
JSBool js_cocos2dx_CCMenu_alignItemsInRows(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
JSObject* jsthis = (JSObject *)JS_THIS_OBJECT(cx, vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, jsthis);
|
|
|
|
CCMenu* cobj = (CCMenu*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
jsval *argvp = JS_ARGV(cx,vp);
|
|
|
|
|
|
|
|
CCArray* pArray = NULL;
|
|
|
|
ok &= jsvals_variadic_to_ccarray(cx, argvp, argc, &pArray);
|
|
|
|
if (ok && pArray)
|
|
|
|
{
|
|
|
|
cobj->alignItemsInRowsWithArray(pArray);
|
2012-11-30 21:15:01 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
2013-01-11 13:54:57 +08:00
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "Error in js_cocos2dx_CCMenu_alignItemsInRows");
|
2012-11-30 21:15:01 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCMenu_alignItemsInColumns(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2013-01-11 13:54:57 +08:00
|
|
|
JSObject* jsthis = (JSObject *)JS_THIS_OBJECT(cx, vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, jsthis);
|
|
|
|
CCMenu* cobj = (CCMenu*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
jsval *argvp = JS_ARGV(cx,vp);
|
|
|
|
CCArray* pArray = NULL;
|
|
|
|
ok &= jsvals_variadic_to_ccarray(cx, argvp, argc, &pArray);
|
|
|
|
if (ok && pArray)
|
|
|
|
{
|
|
|
|
cobj->alignItemsInColumnsWithArray(pArray);
|
2012-11-30 21:15:01 +08:00
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
2013-01-11 13:54:57 +08:00
|
|
|
}
|
2013-01-11 17:44:39 +08:00
|
|
|
JS_ReportError(cx, "Error in js_cocos2dx_CCMenu_alignItemsInColumns");
|
2012-11-30 21:15:01 +08:00
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-11-02 15:21:44 +08:00
|
|
|
// CCTMXLayer
|
|
|
|
JSBool js_cocos2dx_CCTMXLayer_getTileFlagsAt(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
2013-01-11 13:54:57 +08:00
|
|
|
JSBool ok = JS_TRUE;
|
2012-11-02 15:21:44 +08:00
|
|
|
JSObject *obj;
|
|
|
|
CCTMXLayer* cobj;
|
|
|
|
obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cobj = (CCTMXLayer*)(proxy ? proxy->ptr : NULL);
|
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
if (argc == 1)
|
|
|
|
{
|
|
|
|
ccTMXTileFlags flags;
|
2013-01-11 13:54:57 +08:00
|
|
|
CCPoint arg0;
|
|
|
|
ok &= jsval_to_ccpoint(cx, argv[0], &arg0);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2012-11-02 15:21:44 +08:00
|
|
|
cobj->tileGIDAt(arg0, &flags);
|
|
|
|
|
|
|
|
JS_SET_RVAL(cx, vp, UINT_TO_JSVAL((uint32_t)flags));
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 2);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
2013-01-11 13:54:57 +08:00
|
|
|
//#pragma mark - CCDrawNode
|
|
|
|
|
|
|
|
// Arguments: Array of points, fill color (ccc4f), width(float), border color (ccc4f)
|
|
|
|
// Ret value: void
|
|
|
|
JSBool js_cocos2dx_CCDrawNode_drawPolygon(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject* obj = (JSObject *)JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
2012-11-22 15:19:16 +08:00
|
|
|
CCDrawNode* cobj = (CCDrawNode*)(proxy ? proxy->ptr : NULL);
|
2013-01-11 13:54:57 +08:00
|
|
|
TEST_NATIVE_OBJECT(cx, cobj)
|
|
|
|
|
|
|
|
if ( argc == 4) {
|
|
|
|
jsval *argvp = JS_ARGV(cx,vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
JSObject *argArray = NULL;
|
|
|
|
ccColor4F argFillColor = ccc4f(0.0f, 0.0f, 0.0f, 0.0f);
|
|
|
|
double argWidth = 0.0;
|
|
|
|
ccColor4F argBorderColor = ccc4f(0.0f, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
|
|
// Points
|
|
|
|
ok &= JS_ValueToObject(cx, *argvp++, &argArray);
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2( (argArray && JS_IsArrayObject(cx, argArray)) , cx, JS_FALSE, "Vertex should be anArray object");
|
2013-01-11 13:54:57 +08:00
|
|
|
|
|
|
|
// Color 4F
|
|
|
|
ok &= jsval_to_cccolor4f(cx, *argvp++, &argFillColor);
|
|
|
|
|
|
|
|
// Width
|
|
|
|
ok &= JS_ValueToNumber( cx, *argvp++, &argWidth );
|
|
|
|
|
|
|
|
// Color Border (4F)
|
|
|
|
ok &= jsval_to_cccolor4f(cx, *argvp++, &argBorderColor);
|
|
|
|
|
2013-01-11 17:44:39 +08:00
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error parsing arguments");
|
2013-01-11 13:54:57 +08:00
|
|
|
|
|
|
|
{
|
|
|
|
uint32_t l;
|
|
|
|
if( ! JS_GetArrayLength(cx, argArray, &l) )
|
|
|
|
return JS_FALSE;
|
|
|
|
|
|
|
|
CCPoint* verts = new CCPoint[ l ];
|
|
|
|
CCPoint p;
|
|
|
|
|
|
|
|
for( uint32_t i=0; i<l; i++ ) {
|
|
|
|
jsval pointvp;
|
2013-01-11 18:57:33 +08:00
|
|
|
ok &= JS_GetElement(cx, argArray, i, &pointvp);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "JS_GetElement fails.");
|
|
|
|
|
|
|
|
ok &= jsval_to_ccpoint(cx, pointvp, &p);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
2013-01-11 13:54:57 +08:00
|
|
|
verts[i] = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
cobj->drawPolygon(verts, l, argFillColor, argWidth, argBorderColor);
|
|
|
|
CC_SAFE_DELETE_ARRAY(verts);
|
|
|
|
}
|
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 4);
|
2013-01-11 17:44:39 +08:00
|
|
|
return JS_FALSE;
|
2012-11-22 15:19:16 +08:00
|
|
|
}
|
2012-11-02 15:21:44 +08:00
|
|
|
|
2013-01-26 22:31:57 +08:00
|
|
|
static JSBool jsval_to_string_vector(JSContext* cx, jsval v, std::vector<std::string>& ret) {
|
|
|
|
JSObject *jsobj;
|
|
|
|
JSBool ok = JS_ValueToObject( cx, v, &jsobj );
|
|
|
|
JSB_PRECONDITION2( ok, cx, JS_FALSE, "Error converting value to object");
|
|
|
|
JSB_PRECONDITION2( jsobj && JS_IsArrayObject( cx, jsobj), cx, JS_FALSE, "Object must be an array");
|
|
|
|
|
|
|
|
uint32_t len = 0;
|
|
|
|
JS_GetArrayLength(cx, jsobj, &len);
|
|
|
|
|
|
|
|
for (uint32_t i=0; i < len; i++) {
|
|
|
|
jsval elt;
|
|
|
|
if (JS_GetElement(cx, jsobj, i, &elt)) {
|
|
|
|
|
|
|
|
if (JSVAL_IS_STRING(elt))
|
|
|
|
{
|
|
|
|
JSStringWrapper str(JSVAL_TO_STRING(elt));
|
|
|
|
ret.push_back(str.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static jsval string_vector_to_jsval(JSContext* cx, const std::vector<std::string>& arr) {
|
|
|
|
|
|
|
|
JSObject *jsretArr = JS_NewArrayObject(cx, 0, NULL);
|
|
|
|
|
|
|
|
int i = 0;
|
|
|
|
for(std::vector<std::string>::const_iterator iter = arr.begin(); iter != arr.end(); ++iter, ++i) {
|
|
|
|
jsval arrElement = c_string_to_jsval(cx, iter->c_str());
|
|
|
|
if(!JS_SetElement(cx, jsretArr, i, &arrElement)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return OBJECT_TO_JSVAL(jsretArr);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCFileUtils_setSearchResolutionsOrder(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCFileUtils* cobj = (cocos2d::CCFileUtils *)(proxy ? proxy->ptr : NULL);
|
|
|
|
JSB_PRECONDITION2( cobj, cx, JS_FALSE, "Invalid Native Object");
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
std::vector<std::string> arg0;
|
|
|
|
ok &= jsval_to_string_vector(cx, argv[0], arg0);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
cobj->setSearchResolutionsOrder(arg0);
|
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSBool js_cocos2dx_CCFileUtils_setSearchPath(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
jsval *argv = JS_ARGV(cx, vp);
|
|
|
|
JSBool ok = JS_TRUE;
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCFileUtils* cobj = (cocos2d::CCFileUtils *)(proxy ? proxy->ptr : NULL);
|
|
|
|
JSB_PRECONDITION2( cobj, cx, JS_FALSE, "Invalid Native Object");
|
|
|
|
|
|
|
|
if (argc == 1) {
|
|
|
|
std::vector<std::string> arg0;
|
|
|
|
ok &= jsval_to_string_vector(cx, argv[0], arg0);
|
|
|
|
JSB_PRECONDITION2(ok, cx, JS_FALSE, "Error processing arguments");
|
|
|
|
cobj->setSearchPath(arg0);
|
|
|
|
JS_SET_RVAL(cx, vp, JSVAL_VOID);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
JSBool js_cocos2dx_CCFileUtils_getSearchPath(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCFileUtils* cobj = (cocos2d::CCFileUtils *)(proxy ? proxy->ptr : NULL);
|
|
|
|
JSB_PRECONDITION2( cobj, cx, JS_FALSE, "Invalid Native Object");
|
|
|
|
|
|
|
|
if (argc == 0) {
|
|
|
|
std::vector<std::string> ret = cobj->getSearchPath();
|
|
|
|
jsval jsret;
|
|
|
|
jsret = string_vector_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 0);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
JSBool js_cocos2dx_CCFileUtils_getSearchResolutionsOrder(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSObject *obj = JS_THIS_OBJECT(cx, vp);
|
|
|
|
js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj);
|
|
|
|
cocos2d::CCFileUtils* cobj = (cocos2d::CCFileUtils *)(proxy ? proxy->ptr : NULL);
|
|
|
|
JSB_PRECONDITION2( cobj, cx, JS_FALSE, "Invalid Native Object");
|
|
|
|
|
|
|
|
if (argc == 0) {
|
|
|
|
std::vector<std::string> ret = cobj->getSearchResolutionsOrder();
|
|
|
|
jsval jsret;
|
|
|
|
jsret = string_vector_to_jsval(cx, ret);
|
|
|
|
JS_SET_RVAL(cx, vp, jsret);
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 0);
|
|
|
|
return JS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
void register_cocos2dx_js_extensions(JSContext* cx, JSObject* global)
|
|
|
|
{
|
|
|
|
// first, try to get the ns
|
|
|
|
jsval nsval;
|
|
|
|
JSObject *ns;
|
|
|
|
JS_GetProperty(cx, global, "cc", &nsval);
|
|
|
|
if (nsval == JSVAL_VOID) {
|
|
|
|
ns = JS_NewObject(cx, NULL, NULL, NULL);
|
|
|
|
nsval = OBJECT_TO_JSVAL(ns);
|
|
|
|
JS_SetProperty(cx, global, "cc", &nsval);
|
|
|
|
} else {
|
|
|
|
JS_ValueToObject(cx, nsval, &ns);
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_DefineFunction(cx, global, "__associateObjWithNative", js_cocos2dx_swap_native_object, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "__getPlatform", js_platform, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
JSObject *tmpObj;
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "getChildren", js_cocos2dx_CCNode_getChildren, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "copy", js_cocos2dx_CCNode_copy, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-10 05:59:16 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "onExit", js_doNothing, 1, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "onEnter", js_doNothing, 1, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
2012-11-05 18:44:28 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "onEnterTransitionDidFinish", js_doNothing, 0, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "onExitTransitionDidStart", js_doNothing, 0, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "init", js_doNothing, 0, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
|
|
|
|
2012-11-07 03:04:02 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCScheduler_prototype, "resumeTarget", js_cocos2dx_CCScheduler_resumeTarget, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCScheduler_prototype, "pauseTarget", js_cocos2dx_CCScheduler_pauseTarget, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-10-10 05:59:16 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "schedule", js_CCNode_schedule, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "scheduleOnce", js_CCNode_scheduleOnce, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "unschedule", js_CCNode_unschedule, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCScheduler_prototype, "scheduleCallbackForTarget", js_CCScheduler_schedule, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCScheduler_prototype, "unscheduleAllCallbacksForTarget", js_cocos2dx_CCScheduler_unscheduleAllSelectorsForTarget, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-07 03:04:02 +08:00
|
|
|
|
2012-11-03 01:23:23 +08:00
|
|
|
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "setPosition", js_cocos2dx_CCNode_setPosition, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCSprite_prototype, "setPosition", js_cocos2dx_CCSprite_setPosition, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-02 15:21:44 +08:00
|
|
|
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCTMXLayer_prototype, "getTileFlagsAt", js_cocos2dx_CCTMXLayer_getTileFlagsAt, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
2012-11-22 15:19:16 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCDrawNode_prototype, "drawPoly", js_cocos2dx_CCDrawNode_drawPolygon, 4, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCDrawNode_prototype, "setBlendFunc", js_cocos2dx_CCDrawNode_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-11-30 18:15:29 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCTexture2D_prototype, "setTexParameters", js_cocos2dx_CCTexture2D_setTexParameters, 4, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
2012-11-30 21:15:01 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCMenu_prototype, "alignItemsInRows", js_cocos2dx_CCMenu_alignItemsInRows, 1, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCMenu_prototype, "alignItemsInColumns", js_cocos2dx_CCMenu_alignItemsInColumns, 1, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT);
|
2012-11-30 18:15:29 +08:00
|
|
|
|
2013-01-26 22:31:57 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCFileUtils_prototype, "setSearchResolutionsOrder", js_cocos2dx_CCFileUtils_setSearchResolutionsOrder, 1, JSPROP_PERMANENT | JSPROP_SHARED);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCFileUtils_prototype, "setSearchPath", js_cocos2dx_CCFileUtils_setSearchPath, 1, JSPROP_PERMANENT | JSPROP_SHARED);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCFileUtils_prototype, "getSearchPath", js_cocos2dx_CCFileUtils_getSearchPath, 0, JSPROP_PERMANENT | JSPROP_SHARED);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCFileUtils_prototype, "getSearchResolutionsOrder", js_cocos2dx_CCFileUtils_getSearchResolutionsOrder, 0, JSPROP_PERMANENT | JSPROP_SHARED);
|
|
|
|
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.BezierBy; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCBezierBy_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.BezierTo; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCBezierTo_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.CardinalSplineBy; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCCardinalSplineBy_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.CardinalSplineTo; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCCardinalSplineTo_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.CatmullRomBy; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCCatmullRomBy_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.CatmullRomTo; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", JSB_CCCatmullRomTo_actionWithDuration, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "retain", js_cocos2dx_retain, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCNode_prototype, "release", js_cocos2dx_release, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-10 15:25:27 +08:00
|
|
|
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCSprite_prototype, "setBlendFunc", js_cocos2dx_CCSprite_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCSpriteBatchNode_prototype, "setBlendFunc", js_cocos2dx_CCSpriteBatchNode_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
//JS_DefineFunction(cx, js_cocos2dx_CCMotionStreak_prototype, "setBlendFunc", js_cocos2dx_CCMotionStreak_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAtlasNode_prototype, "setBlendFunc", js_cocos2dx_CCAtlasNode_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCParticleBatchNode_prototype, "setBlendFunc", js_cocos2dx_CCParticleBatchNode_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCLayerColor_prototype, "setBlendFunc", js_cocos2dx_CCLayerColor_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCParticleSystem_prototype, "setBlendFunc", js_cocos2dx_CCParticleSystem_setBlendFunc, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAction_prototype, "copy", js_cocos2dx_CCNode_copy, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAction_prototype, "retain", js_cocos2dx_retain, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAction_prototype, "release", js_cocos2dx_release, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAnimation_prototype, "copy", js_cocos2dx_CCNode_copy, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAnimation_prototype, "retain", js_cocos2dx_retain, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCAnimation_prototype, "release", js_cocos2dx_release, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCSpriteFrame_prototype, "retain", js_cocos2dx_retain, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCSpriteFrame_prototype, "release", js_cocos2dx_release, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-23 02:07:23 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCMenuItem_prototype, "setCallback", js_cocos2dx_CCMenuItem_setCallback, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-07 03:04:02 +08:00
|
|
|
JS_DefineFunction(cx, js_cocos2dx_CCTMXLayer_prototype, "getTileFlagsAt", js_cocos2dx_CCTMXLayer_tileFlagsAt, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-03 01:23:23 +08:00
|
|
|
|
|
|
|
|
2012-11-30 21:15:01 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Node.prototype; })()"));
|
2012-11-03 01:23:23 +08:00
|
|
|
JS_DefineFunction(cx, tmpObj, "unscheduleAllCallbacks", js_cocos2dx_CCNode_unscheduleAllSelectors, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_DefineFunction(cx, tmpObj, "copy", js_cocos2dx_CCNode_copy, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
issue #1581: JSBinding bug fixes.
Some fixes of JSBinding codes:
[1] Check whether the proxy was already added in JS_NEW_PROXY
[2] In struct schedFunc_proxy_t, JSScheduleWrapper* --> CCArray*
Reason: One js function may correspond to many targets. To debug this, you could refer to JSScheduleWrapper::dump function. It will prove that i'm right. :)
[3] In ScriptingCore::cleanupSchedulesAndActions function, we must invoke unschedule for all targets and remove the proxy both in _schedFunc_target_ht and _schedTarget_native_ht, otherwise the hash tables will grow bigger and bigger, so I added a new function JSScheduleWrapper::removeAllTargetsForNatiaveNode to make this things easier.
[4] To easily find out the bugs of binding codes, I add JS_SetGCZeal in ScriptingCore::createGlobalContext, it only works in DEBUG mode.
[5] In js_cocos2dx_CCNode_getChildren, we should add the generated array to root to avoid gc happen when invoking JS_SetElement.
[6] The JSCallFuncWrapper isn't needed since an action will be run by a cc.Node and it will be released at the CCNode::cleanup.
[7] Some improvements of JSScheduleWrapper class.
[8] Added a new function JSScheduleWrapper::setTarget, it's for js_CCNode_unschedule to find out which target need to be unscheduled.
[9] Commented JS_SetReservedSlot in js_CCNode_scheduleOnce and js_CCNode_schedule.
Reason:
For js_CCNode_scheduleOnce: Don't add the callback function to the reserved slot of this js object.Since the class of js object may be inherited from cocos class(e.g. cc.Sprite). The subclass will not contain reserved slots. It will crash if invoking this.
For js_CCNode_schedule: Don't add js callback function to the reserved slot of scheduler js object. Since the scheduler is an object always rooted.
So the callback function might not be released when gc comes.I looked inside the implementation of cc.Node.schedule, and it doesn't use JS_SetReservedSlot there.
2012-11-28 22:04:55 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Menu; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenu_create, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItem; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItem_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemSprite; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItemSprite_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemImage; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItemImage_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemLabel; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItemLabel_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemAtlasFont; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItemAtlasFont_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemFont; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCMenuItemFont_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.MenuItemToggle; })()"));
|
2012-10-12 09:43:40 +08:00
|
|
|
JS_DefineFunction(cx, tmpObj, "_create", js_cocos2dx_CCMenuItemToggle_create, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-08-28 10:22:36 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Sequence; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCSequence_create, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Spawn; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCSpawn_create, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Animation; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCAnimation_create, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-30 21:15:01 +08:00
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.LayerMultiplex; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_cocos2dx_CCLayerMultiplex_create, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
JS_DefineFunction(cx, ns, "registerTargettedDelegate", js_cocos2dx_JSTouchDelegate_registerTargettedDelegate, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "registerStandardDelegate", js_cocos2dx_JSTouchDelegate_registerStandardDelegate, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
fixed #1617: Some improvements for JS Bindings.
1) Changed cc.REPEAT_FOREVER = - 1 to cc.REPEAT_FOREVER = 0xffffffff
[Reason]: If cc.REPEAT_FOREVER = -1, it will be a very big double value after converting it to double by JS_ValueToNumber on android.
Then cast it to unsigned int, the value will be 0. The schedule will not be able to work.
I don't know why this occurs only on android.
[Solution]: Instead of passing -1 to it, I assign it with max value of unsigned int in c++.
2) Added two helper function, cc.ArrayGetIndexOfObject and cc.ArrayContainsObject.
3) Added JSScheduleWrapper::removeTargetForNativeNode to avoid memory leaks.
4) Improvments for JSTouchDelegate. Added four functions as follows:
// Set the touch delegate to map by using the key (pJSObj).
static void setDelegateForJSObject(JSObject* pJSObj, JSTouchDelegate* pDelegate);
// Get the touch delegate by the key (pJSObj).
static JSTouchDelegate* getDelegateForJSObject(JSObject* pJSObj);
// Remove the delegate by the key (pJSObj).
static void removeDelegateForJSObject(JSObject* pJSObj);
void unregisterTouchDelegate();
And exported cc.unregisterTouchDelegate(); to js.
Fix a memory leak for JSTouchDelegate by making it as an autorelease object.
5) Don't add js callback function to the reserved slot of object.
[Reason]: The target object may execute more than one schedule.
Therefore, previous js callback function will be replaced
by the current one. For example:
this.scheduleOnce(function() { temporary function 1 }, 0.5);
this.scheduleOnce(function() { temporary function 2 }, 0.5);
In this case, the temporary function 1 will be removed from reserved slot 0.
And temporary function 2 will be set to reserved slot 0 of this object.
If gc is triggered before the JSScheduleWrapper::scheduleFunc is invoked,
crash will happen. You could simply reproduce it by adding jsc.garbageCollect(); after scheduleOnce.
[Solution] Because one schedule corresponds to one JSScheduleWrapper, we root
the js callback function in JSScheduleWrapper::setJSCallbackFunc and unroot it
at the destructor of JSScheduleWrapper.
2012-12-18 11:56:44 +08:00
|
|
|
JS_DefineFunction(cx, ns, "unregisterTouchDelegate", js_cocos2dx_JSTouchDelegate_unregisterTouchDelegate, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-08-28 10:22:36 +08:00
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.CallFunc; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "create", js_callFunc, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
|
|
|
tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return this; })()"));
|
|
|
|
JS_DefineFunction(cx, tmpObj, "garbageCollect", js_forceGC, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
|
2012-10-12 09:43:40 +08:00
|
|
|
JS_DefineFunction(cx, ns, "pAdd", js_cocos2dx_ccpAdd, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-01 02:08:37 +08:00
|
|
|
JS_DefineFunction(cx, ns, "pDistance", js_cocos2dx_ccpDistance, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-12 09:43:40 +08:00
|
|
|
JS_DefineFunction(cx, ns, "pSub", js_cocos2dx_ccpSub, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pNeg", js_cocos2dx_ccpNeg, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pMult", js_cocos2dx_ccpMult, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pMidpoint", js_cocos2dx_ccpMidpoint, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pDot", js_cocos2dx_ccpDot, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pCross", js_cocos2dx_ccpCross, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pPerp", js_cocos2dx_ccpPerp, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pRPerp", js_cocos2dx_ccpRPerp, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pProject", js_cocos2dx_ccpProject, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pRotate", js_cocos2dx_ccpRotate, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pNormalize", js_cocos2dx_ccpNormalize, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-27 06:31:52 +08:00
|
|
|
JS_DefineFunction(cx, ns, "pClamp", js_cocos2dx_ccpClamp, 2, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-11-05 23:22:59 +08:00
|
|
|
JS_DefineFunction(cx, ns, "pLengthSQ", js_cocos2dx_ccpLengthSQ, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, ns, "pLength", js_cocos2dx_ccpLength, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2012-10-12 09:43:40 +08:00
|
|
|
|
2012-08-28 10:22:36 +08:00
|
|
|
// add constructor for CCSet
|
|
|
|
JSFunction *ccSetConstructor = JS_NewFunction(cx, js_cocos2dx_CCSet_constructor, 0, JSPROP_READONLY | JSPROP_PERMANENT, NULL, "constructor");
|
|
|
|
JSObject *ctor = JS_GetFunctionObject(ccSetConstructor);
|
|
|
|
JS_LinkConstructorAndPrototype(cx, ctor, js_cocos2dx_CCSet_prototype);
|
|
|
|
}
|