#include "cocos2d.h" #include "cocos2d_specifics.hpp" #include schedFunc_proxy_t *_schedFunc_target_ht = NULL; schedTarget_proxy_t *_schedTarget_native_ht = NULL; callfuncTarget_proxy_t *_callfuncTarget_native_ht = NULL; 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); } 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); } 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); } 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 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(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); JS_AddNamedObjectRoot(cx, &proxy->obj, typeid(*nativeObj).name()); 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(); unsigned int count = children ? children->count() : 0; JSObject *jsarr = JS_NewArrayObject(cx, count, NULL); for (unsigned int i=0; i < count; i++) { cocos2d::CCNode *child = (cocos2d::CCNode*)children->objectAtIndex(i); js_proxy_t *childProxy = js_get_or_create_proxy(cx, child); jsval childVal = OBJECT_TO_JSVAL(childProxy->obj); JS_SetElement(cx, jsarr, i, &childVal); } JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(jsarr)); } return JS_TRUE; } 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(); uint32_t i = 0; 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(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(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } } else { jsret = JSVAL_NULL; } } while (0); JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } 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(); uint32_t i = 0; 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(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } } else { jsret = JSVAL_NULL; } } while (0); JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } 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(); uint32_t i = 0; 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(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } } else { jsret = JSVAL_NULL; } } while (0); JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } 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(); JSObject *obj = bind_menu_item(cx, ret, argv[0], argc == 2? argv[1] : JSVAL_VOID); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } return JS_FALSE; } // "create" in JS // cc.MenuItemSprite.create( normalSprite, selectedSprite, [disabledSprite], [callback_fn], [this] JSBool js_cocos2dx_CCMenuItemSprite_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 2 && argc <= 5) { 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; JSBool thirdArgIsCallback = JS_FALSE; jsval jsCallback = JSVAL_VOID; jsval jsThis = JSVAL_VOID; cocos2d::CCNode* arg2 = NULL; if (argc >= 3) { tmpObj = JSVAL_TO_OBJECT(argv[2]); 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; } } cocos2d::CCMenuItemSprite* ret = cocos2d::CCMenuItemSprite::create(arg0, arg1, arg2); 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]; } } } } JSObject *obj = bind_menu_item(cx, ret, jsCallback, jsThis); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } JS_ReportError(cx, "Invalid number of arguments. Expecting: 2 <= args <= 5"); return JS_FALSE; } // "create" in JS // cc.MenuItemImage.create( normalImage, selectedImage, [disabledImage], callback_fn, [this] JSBool js_cocos2dx_CCMenuItemImage_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 2 && argc <= 5) { jsval *argv = JS_ARGV(cx, vp); JSStringWrapper arg0(argv[0]); JSStringWrapper arg1(argv[1]); JSStringWrapper arg2; bool thirdArgIsString = true; jsval jsCallback = JSVAL_VOID; jsval jsThis = JSVAL_VOID; int last = 2; if (argc >= 3) { thirdArgIsString = argv[2].isString(); if (thirdArgIsString) { arg2.set(argv[2], cx); last = 3; } } cocos2d::CCMenuItemImage* ret = cocos2d::CCMenuItemImage::create(arg0, arg1, arg2); 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(cx, ret, jsCallback, jsThis); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } JS_ReportError(cx, "Invalid number of arguments. Expecting: 2 <= args <= 5"); return JS_FALSE; } // "create" in JS: // cc.MenuItemLabel.create( label, callback_fn, [this] ); JSBool js_cocos2dx_CCMenuItemLabel_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1 && argc <= 3) { 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); JSObject *obj = bind_menu_item(cx, ret, (argc >= 2 ? argv[1] : JSVAL_VOID), (argc == 3 ? argv[2] : JSVAL_VOID) ); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d or %d", argc, 1, 2, 3); return JS_FALSE; } JSBool js_cocos2dx_CCMenuItemAtlasFont_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 5) { jsval *argv = JS_ARGV(cx, vp); JSStringWrapper arg0(argv[0]); JSStringWrapper arg1(argv[1]); int arg2; if (!JS_ValueToInt32(cx, argv[2], &arg2)) return JS_FALSE; int arg3; if (!JS_ValueToInt32(cx, argv[3], &arg3)) return JS_FALSE; int arg4; if (!JS_ValueToInt32(cx, argv[4], &arg4)) return JS_FALSE; cocos2d::CCMenuItemAtlasFont* ret = cocos2d::CCMenuItemAtlasFont::create(arg0, arg1, arg2, arg3, arg4); JSObject *obj = bind_menu_item(cx, ret, (argc >= 6 ? argv[5] : JSVAL_VOID), (argc == 7 ? argv[6] : JSVAL_VOID)); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } return JS_FALSE; } // "create" in JS // cc.MenuItemFont.create( string, callback_fn, [this] ); JSBool js_cocos2dx_CCMenuItemFont_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1 && argc <= 3) { jsval *argv = JS_ARGV(cx, vp); JSStringWrapper arg0(argv[0]); cocos2d::CCMenuItemFont* ret = cocos2d::CCMenuItemFont::create(arg0); JSObject *obj = bind_menu_item(cx, ret, (argc >= 2 ? argv[1] : JSVAL_VOID), (argc == 3 ? argv[2] : JSVAL_VOID)); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(obj)); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d or %d", argc, 1, 2, 3); return JS_FALSE; } JSBool js_cocos2dx_CCMenuItemToggle_create(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1) { jsval *argv = JS_ARGV(cx, vp); cocos2d::CCMenuItemToggle* ret = cocos2d::CCMenuItemToggle::create(); for (uint32_t i=0; i < argc; i++) { 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(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } } else { jsret = JSVAL_NULL; } JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } return JS_FALSE; } // "setCallback" in JS // item.setCallback( callback_fn, [this]); template JSBool js_cocos2dx_setCallback(JSContext *cx, uint32_t argc, jsval *vp) { if(argc == 1 || argc == 2) { jsval *argv = JS_ARGV(cx, vp); JSObject *obj = JS_THIS_OBJECT(cx, vp); jsval jsThis = JSVAL_VOID; jsval jsFunc = argv[0]; if (argc == 2) { jsThis = argv[1]; } js_proxy_t *proxy; JS_GET_NATIVE_PROXY(proxy, obj); T* item = (T*)(proxy ? proxy->ptr : NULL); TEST_NATIVE_OBJECT(cx, item) bind_menu_item(cx, item, jsFunc, jsThis); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d or %d", argc, 1, 2); return JS_FALSE; } JSBool js_cocos2dx_CCMenuItem_setCallback(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setCallback(cx, argc, vp); } JSBool js_cocos2dx_CCAnimation_create(JSContext *cx, uint32_t argc, jsval *vp) { JSBool ok = JS_TRUE; jsval *argv = JS_ARGV(cx, vp); if (argc <= 3) { cocos2d::CCArray* arg0; if (argc > 0) { ok &= jsval_to_ccarray(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } } cocos2d::CCAnimation* ret; double arg1 = 0.0f; if (argc > 0 && argc == 2) { if (argc == 2) { JS_ValueToNumber(cx, argv[1], &arg1); } ret = cocos2d::CCAnimation::createWithSpriteFrames(arg0, arg1); } else if (argc == 3) { unsigned int loops; JS_ValueToNumber(cx, argv[1], &arg1); JS_ValueToECMAUint32(cx, argv[1], &loops); ret = cocos2d::CCAnimation::create(arg0, arg1, loops); } else if (argc == 1) { ret = cocos2d::CCAnimation::createWithSpriteFrames(arg0); } else if (argc == 0) { ret = cocos2d::CCAnimation::create(); } 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(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } } else { jsret = JSVAL_NULL; } JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } return JS_FALSE; } JSBool js_cocos2dx_CCLayerMultiplex_create(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); cocos2d::CCArray* arg0; JSBool ok = JS_TRUE; ok &= jsvals_variadic_to_ccarray(cx, argv, argc, &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } cocos2d::CCLayerMultiplex* ret = cocos2d::CCLayerMultiplex::createWithArray(arg0); jsval jsret; do { if (ret) { js_proxy_t *proxy = js_get_or_create_proxy(cx, ret); jsret = OBJECT_TO_JSVAL(proxy->obj); } else { jsret = JSVAL_NULL; } } while (0); JS_SET_RVAL(cx, vp, jsret); return JS_TRUE; } JSBool js_cocos2dx_JSTouchDelegate_registerStandardDelegate(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1) { jsval *argv = JS_ARGV(cx, vp); JSObject* jsobj = NULL; JSTouchDelegate *touch = new JSTouchDelegate(); touch->autorelease(); touch->registerStandardDelegate(); jsobj = (argc == 1 ? JSVAL_TO_OBJECT(argv[0]) : JSVAL_TO_OBJECT(JSVAL_VOID)); touch->setJSObject(jsobj); JSTouchDelegate::setDelegateForJSObject(jsobj, touch); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting >= 1", argc); return JS_FALSE; } JSBool js_cocos2dx_JSTouchDelegate_registerTargettedDelegate(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1) { jsval *argv = JS_ARGV(cx, vp); JSObject* jsobj = NULL; JSTouchDelegate *touch = new JSTouchDelegate(); touch->autorelease(); touch->registerTargettedDelegate((argc >= 1 ? JSVAL_TO_INT(argv[0]) : 0), (argc >= 2 ? JSVAL_TO_BOOLEAN(argv[1]) : true)); jsobj = (argc == 3 ? JSVAL_TO_OBJECT(argv[2]) : JSVAL_TO_OBJECT(JSVAL_VOID)); touch->setJSObject(jsobj); JSTouchDelegate::setDelegateForJSObject(jsobj, touch); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting >=1", argc); return JS_FALSE; } 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; } 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) { JS_RemoveObjectRoot(cx, &nproxy->obj); JS_REMOVE_PROXY(jsproxy, nproxy); JS_NEW_PROXY(nproxy, ptrTwo, one); JS_AddNamedObjectRoot(cx, &nproxy->obj, typeid(*((CCObject*)nproxy->ptr)).name()); } } } 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); cocos2d::CCObject *node = (cocos2d::CCObject *)(proxy ? proxy->ptr : NULL); TEST_NATIVE_OBJECT(cx, node) cocos2d::CCObject *ret = node->copy(); proxy = js_get_or_create_proxy(cx, ret); if (ret && proxy) { ret->autorelease(); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(proxy->obj)); return JS_TRUE; } } return JS_FALSE; } JSObject* getObjectFromNamespace(JSContext* cx, JSObject *ns, const char *name) { jsval out; if (JS_GetProperty(cx, ns, name, &out) == JS_TRUE) { JSObject *obj; if (JS_ValueToObject(cx, out, &obj) == JS_TRUE) { } } 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; } JSCallbackWrapper::JSCallbackWrapper() : jsCallback(JSVAL_VOID), jsThisObj(JSVAL_VOID), extraData(JSVAL_VOID) { } JSCallbackWrapper::~JSCallbackWrapper() { JSContext* cx = ScriptingCore::getInstance()->getGlobalContext(); JS_RemoveValueRoot(cx, &jsCallback); } void JSCallbackWrapper::setJSCallbackFunc(jsval func) { jsCallback = func; JSContext* cx = ScriptingCore::getInstance()->getGlobalContext(); // Root the callback function. JS_AddNamedValueRoot(cx, &jsCallback, "JSCallbackWrapper_callback_func"); } void JSCallbackWrapper::setJSCallbackThis(jsval thisObj) { jsThisObj = thisObj; } void JSCallbackWrapper::setJSExtraData(jsval data) { extraData = data; } const jsval& JSCallbackWrapper::getJSCallbackFunc() const { return jsCallback; } const jsval& JSCallbackWrapper::getJSCallbackThis() const { return jsThisObj; } const jsval& JSCallbackWrapper::getJSExtraData() const { return extraData; } void JSCallFuncWrapper::setTargetForNativeNode(CCNode *pNode, JSCallFuncWrapper *target) { 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); } CCArray * JSCallFuncWrapper::getTargetForNativeNode(CCNode *pNode) { callfuncTarget_proxy_t *t; HASH_FIND_PTR(_callfuncTarget_native_ht, &pNode, t); if(!t) { return NULL; } return t->obj; } void JSCallFuncWrapper::callbackFunc(CCNode *node) const { bool hasExtraData = !JSVAL_IS_VOID(extraData); JSObject* thisObj = JSVAL_IS_VOID(jsThisObj) ? NULL : JSVAL_TO_OBJECT(jsThisObj); JSContext *cx = ScriptingCore::getInstance()->getGlobalContext(); js_proxy_t *proxy = js_get_or_create_proxy(cx, node); jsval retval; 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); } else { jsval senderVal = OBJECT_TO_JSVAL(proxy->obj); JS_AddValueRoot(cx, &senderVal); JS_CallFunctionValue(cx, thisObj, jsCallback, 1, &senderVal, &retval); JS_RemoveValueRoot(cx, &senderVal); } } // 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); } // cc.CallFunc.create( func, this, [data]) // cc.CallFunc.create( func ) JSBool js_callFunc(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 1 && argc <= 3) { jsval *argv = JS_ARGV(cx, vp); JSCallFuncWrapper *tmpCobj = new JSCallFuncWrapper(); tmpCobj->autorelease(); tmpCobj->setJSCallbackFunc(argv[0]); if(argc >= 2) { tmpCobj->setJSCallbackThis(argv[1]); } if(argc == 3) { tmpCobj->setJSExtraData(argv[2]); } CCCallFunc *ret = (CCCallFunc *)CCCallFuncN::create((CCObject *)tmpCobj, callfuncN_selector(JSCallFuncWrapper::callbackFunc)); js_proxy_t *proxy = js_get_or_create_proxy(cx, ret); JS_SET_RVAL(cx, vp, OBJECT_TO_JSVAL(proxy->obj)); JS_SetReservedSlot(proxy->obj, 0, argv[0]); if(argc > 1) { JS_SetReservedSlot(proxy->obj, 1, argv[1]); } // if(argc == 3) { // JS_SetReservedSlot(proxy->obj, 2, argv[2]); // } // test->execute(); return JS_TRUE; } JS_ReportError(cx, "Invalid number of arguments"); return JS_FALSE; } void JSScheduleWrapper::setTargetForSchedule(jsval sched, JSScheduleWrapper *target) { do { 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); } while(0); } CCArray * JSScheduleWrapper::getTargetForSchedule(jsval sched) { schedFunc_proxy_t *t = NULL; JSObject *o = JSVAL_TO_OBJECT(sched); HASH_FIND_PTR(_schedFunc_target_ht, &o, t); return t != NULL ? t->targets : NULL; } void JSScheduleWrapper::setTargetForNativeNode(CCNode *pNode, JSScheduleWrapper *target) { CCArray* targetArray = getTargetForNativeNode(pNode); if (NULL == targetArray) { targetArray = new CCArray(); schedTarget_proxy_t *p = (schedTarget_proxy_t *)malloc(sizeof(schedTarget_proxy_t)); assert(p); 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); } CCArray * JSScheduleWrapper::getTargetForNativeNode(CCNode *pNode) { schedTarget_proxy_t *t = NULL; HASH_FIND_PTR(_schedTarget_native_ht, &pNode, t); return t != NULL ? t->targets : NULL; } void JSScheduleWrapper::removeAllTargetsForNatiaveNode(CCNode* pNode) { CCLOGINFO("removeAllTargetsForNatiaveNode begin"); 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); } if (removeNativeTargets == NULL) return; schedFunc_proxy_t *current, *tmp; HASH_ITER(hh, _schedFunc_target_ht, current, tmp) { std::vector 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::iterator iter = objectsNeedToBeReleased.begin(); for (; iter != objectsNeedToBeReleased.end(); ++iter) { targets->removeObject(*iter, true); } if (targets->count() == 0) { HASH_DEL(_schedFunc_target_ht, current); targets->release(); free(current); } } removeNativeTargets->removeAllObjects(); removeNativeTargets->release(); free(t); dump(); CCLOGINFO("removeAllTargetsForNatiaveNode end"); } 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(); } void JSScheduleWrapper::dump() { #if COCOS2D_DEBUG > 1 CCLOG("\n---------JSScheduleWrapper dump begin--------------\n"); schedTarget_proxy_t *current, *tmp; int nativeTargetsCount = 0; HASH_ITER(hh, _schedTarget_native_ht, current, tmp) { CCObject* pObj = NULL; CCARRAY_FOREACH(current->targets, pObj) { CCLOG("native %s ( %p ), target[%d]=( %p )", typeid(*current->nativeObj).name(), current->nativeObj, nativeTargetsCount, pObj); 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) { CCLOG("jsfunc ( %p ), target[%d]=( %p )", current_func->jsfuncObj, jsfuncTargetCount, pObj); jsfuncTargetCount++; } } CCAssert(nativeTargetsCount == jsfuncTargetCount, ""); CCLOG("\n---------JSScheduleWrapper dump end--------------\n"); #endif } void JSScheduleWrapper::scheduleFunc(float dt) const { jsval retval = JSVAL_NULL; jsval data = DOUBLE_TO_JSVAL(dt); JSContext *cx = ScriptingCore::getInstance()->getGlobalContext(); JSBool ok = JS_AddValueRoot(cx, &data); if(!ok) { return; } if(!jsCallback.isNullOrUndefined() || !jsThisObj.isNullOrUndefined()) { JSAutoCompartment ac(cx, JSVAL_TO_OBJECT(jsThisObj)); JS_CallFunctionValue(cx, JSVAL_TO_OBJECT(jsThisObj), jsCallback, 1, &data, &retval); } JS_RemoveValueRoot(cx, &data); } CCObject* JSScheduleWrapper::getTarget() { return m_pTarget; } void JSScheduleWrapper::setTarget(CCObject* pTarget) { m_pTarget = pTarget; } 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); if(!node) return JS_FALSE; CCScheduler *sched = node->getScheduler(); CCArray* targetArray = JSScheduleWrapper::getTargetForSchedule(argv[0]); CCLOGINFO("unschedule target number: %d", targetArray->count()); CCObject* tmp = NULL; CCARRAY_FOREACH(targetArray, tmp) { JSScheduleWrapper* target = (JSScheduleWrapper*)tmp; if (node == target->getTarget()) { sched->unscheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), target); JSScheduleWrapper::removeTargetForNativeNode(node, target); break; } } JS_SET_RVAL(cx, vp, JSVAL_VOID); } return JS_TRUE; } 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) 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)); } } return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 0); return JS_FALSE; } 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) cobj->unscheduleAllForTarget(arg0); CCArray *arr = JSScheduleWrapper::getTargetForNativeNode(arg0); // 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)) { arg0->getScheduler()->unscheduleAllForTarget(arr->objectAtIndex(i)); } } JSScheduleWrapper::removeAllTargetsForNatiaveNode(arg0); } while (0); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1); return JS_FALSE; } JSBool js_CCNode_scheduleOnce(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); CCScheduler *sched = node->getScheduler(); JSScheduleWrapper *tmpCobj = NULL; // // delay // double delay; if( argc >= 2 ) { if( ! JS_ValueToNumber(cx, argv[1], &delay ) ) return JS_FALSE; } 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); } if(argc == 1) { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, 0, 0.0f, !node->isRunning()); } else { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, 0, delay, !node->isRunning()); } /* 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]); JS_SET_RVAL(cx, vp, JSVAL_VOID); } return JS_TRUE; } JSBool js_CCNode_schedule(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); CCScheduler *sched = node->getScheduler(); js_proxy_t *p = js_get_or_create_proxy(cx, sched); JSScheduleWrapper *tmpCobj = NULL; double interval = 0.0; if( argc >= 2 ) { if( ! JS_ValueToNumber(cx, argv[1], &interval ) ) return JS_FALSE; } // // repeat // double repeat = 0.0; if( argc >= 3 ) { if( ! JS_ValueToNumber(cx, argv[2], &repeat ) ) return JS_FALSE; } // // delay // double delay = 0.0; if( argc >= 4 ) { if( ! JS_ValueToNumber(cx, argv[3], &delay ) ) return JS_FALSE; } 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); } if(argc == 1) { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, 0, !node->isRunning()); } if(argc == 2) { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, !node->isRunning()); } if(argc == 3) { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, (unsigned int)repeat, 0, !node->isRunning()); } if (argc == 4) { sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, (unsigned int)repeat, delay, !node->isRunning()); } // I comment next line with the same reason in the js_CCNode_scheduleOnce. //jsb_set_reserved_slot(proxy->obj, 0, argv[0]); JS_SET_RVAL(cx, vp, JSVAL_VOID); } return JS_TRUE; } JSBool js_CCScheduler_schedule(JSContext *cx, uint32_t argc, jsval *vp) { if (argc >= 2) { 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); JSScheduleWrapper *tmpCobj = NULL; 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) double interval = 0; if( argc >= 3 ) { if( ! JS_ValueToNumber(cx, argv[2], &interval ) ) return JS_FALSE; } // // repeat // double repeat = -1; if( argc >= 4 ) { if( ! JS_ValueToNumber(cx, argv[3], &repeat ) ) return JS_FALSE; } // // delay // double delay = 0; if( argc >= 5 ) { if( ! JS_ValueToNumber(cx, argv[4], &delay ) ) return JS_FALSE; } JSBool paused = !node->isRunning(); if( argc >= 6 ) { if( ! JS_ValueToBoolean(cx, argv[4], &paused)) return JS_FALSE; } bool bFound = false; CCArray* pTargetArr = JSScheduleWrapper::getTargetForNativeNode(node); CCObject* pObj = NULL; CCARRAY_FOREACH(pTargetArr, pObj) { JSScheduleWrapper* pTarget = (JSScheduleWrapper*)pObj; if (argv[1] == pTarget->getJSCallbackFunc()) { 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); } sched->scheduleSelector(schedule_selector(JSScheduleWrapper::scheduleFunc), tmpCobj, interval, repeat, delay, paused); JS_SET_RVAL(cx, vp, JSVAL_VOID); } return JS_TRUE; } 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); if(! arr) return JS_TRUE; 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); if(! arr) return JS_TRUE; 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; } JSBool js_doNothing(JSContext *cx, uint32_t argc, jsval *vp) { return JS_TRUE; } 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; } } 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; } } 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 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; } return JS_FALSE; } 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); JSBool ok = JS_TRUE; 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; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } cobj->setPosition(arg0); return JS_TRUE; } if (argc == 2) { double x; if( ! JS_ValueToNumber(cx, argv[0], &x ) ) { return JS_FALSE; } double y; if( ! JS_ValueToNumber(cx, argv[1], &y ) ) { return JS_FALSE; } cobj->setPosition(CCPoint(x,y)); 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); JSBool ok = JS_TRUE; 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; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } cobj->setPosition(arg0); return JS_TRUE; } if (argc == 2) { double x; if( ! JS_ValueToNumber(cx, argv[0], &x ) ) { return JS_FALSE; } double y; if( ! JS_ValueToNumber(cx, argv[1], &y ) ) { return JS_FALSE; } cobj->setPosition(CCPoint(x,y)); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 1); return JS_FALSE; } JSBool js_cocos2dx_CCTMXLayer_tileFlagsAt(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSObject *obj; JSBool ok = JS_TRUE; 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; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); 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; } return JS_FALSE; } template 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]; T* ret = T::create(t, config); 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(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 JSBool js_CardinalSplineActions_create(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); if (argc == 3) { double dur; if( ! JS_ValueToNumber(cx, argv[0], &dur) ) { return JS_FALSE; } int num; CCPoint *arr; jsval_to_ccarray_of_CCPoint(cx, argv[1], &arr, &num); double ten; if( ! JS_ValueToNumber(cx, argv[2], &ten) ) { return JS_FALSE; } 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(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 JSBool js_CatmullRomActions_create(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); if (argc == 2) { double dur; if( ! JS_ValueToNumber(cx, argv[0], &dur) ) { return JS_FALSE; } int num; CCPoint *arr; jsval_to_ccarray_of_CCPoint(cx, argv[1], &arr, &num); 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(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) { return js_BezierActions_create(cx, argc, vp); } JSBool JSB_CCBezierTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) { return js_BezierActions_create(cx, argc, vp); } JSBool JSB_CCCardinalSplineBy_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) { return js_CardinalSplineActions_create(cx, argc, vp); } JSBool JSB_CCCardinalSplineTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) { return js_CardinalSplineActions_create(cx, argc, vp); } JSBool JSB_CCCatmullRomBy_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) { return js_CatmullRomActions_create(cx, argc, vp); } JSBool JSB_CCCatmullRomTo_actionWithDuration(JSContext *cx, uint32_t argc, jsval *vp) { return js_CatmullRomActions_create(cx, argc, vp); } JSBool js_cocos2dx_ccpAdd(JSContext *cx, uint32_t argc, jsval *vp) { JSBool ok = JS_TRUE; jsval *argv = JS_ARGV(cx, vp); if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } JSBool js_cocos2dx_ccpDistance(JSContext *cx, uint32_t argc, jsval *vp) { JSBool ok = JS_TRUE; jsval *argv = JS_ARGV(cx, vp); if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } JSBool js_cocos2dx_ccpClamp(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 3) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); cocos2d::CCPoint arg2; ok &= jsval_to_ccpoint(cx, argv[2], &arg2); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } JSBool js_cocos2dx_ccpLengthSQ(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } JSBool js_cocos2dx_ccpNeg(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } double arg1; if( ! JS_ValueToNumber(cx, argv[1], &arg1) ) { return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } CCPoint ret = ccpMidpoint(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, 2); return JS_FALSE; } JSBool js_cocos2dx_ccpDot(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } float ret = ccpDot(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, 2); return JS_FALSE; } JSBool js_cocos2dx_ccpCross(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } float ret = ccpCross(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, 2); return JS_FALSE; } JSBool js_cocos2dx_ccpPerp(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } CCPoint ret = ccpProject(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, 2); return JS_FALSE; } JSBool js_cocos2dx_ccpRotate(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 2) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); cocos2d::CCPoint arg1; ok &= jsval_to_ccpoint(cx, argv[1], &arg1); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } CCPoint ret = ccpRotate(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, 2); return JS_FALSE; } JSBool js_cocos2dx_ccpNormalize(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; if (argc == 1) { cocos2d::CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } extern JSObject* js_cocos2dx_CCNode_prototype; extern JSObject* js_cocos2dx_CCLayerColor_prototype; extern JSObject* js_cocos2dx_CCSprite_prototype; extern JSObject* js_cocos2dx_CCTMXLayer_prototype; 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; 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; 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; extern JSObject* js_cocos2dx_CCScheduler_prototype; extern JSObject* js_cocos2dx_CCDrawNode_prototype; extern JSObject* js_cocos2dx_CCTexture2D_prototype; extern JSObject* js_cocos2dx_CCMenu_prototype; // setBlendFunc template 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) if (argc == 2) { GLenum src, dst; JS_ValueToInt32(cx, argv[0], (int32_t*)&src); JS_ValueToInt32(cx, argv[1], (int32_t*)&dst); ccBlendFunc blendFunc = {src, dst}; cobj->setBlendFunc(blendFunc); return JS_TRUE; } 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(cx, argc, vp); } JSBool js_cocos2dx_CCSpriteBatchNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } // JSBool js_cocos2dx_CCMotionStreak_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) // { // return js_cocos2dx_setBlendFunc(cx, argc, vp); // } JSBool js_cocos2dx_CCAtlasNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } JSBool js_cocos2dx_CCParticleBatchNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } JSBool js_cocos2dx_CCLayerColor_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } JSBool js_cocos2dx_CCParticleSystem_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } JSBool js_cocos2dx_CCDrawNode_setBlendFunc(JSContext *cx, uint32_t argc, jsval *vp) { return js_cocos2dx_setBlendFunc(cx, argc, vp); } 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) if (argc == 4) { jsval *argvp = JS_ARGV(cx,vp); JSBool ok = JS_TRUE; GLint arg0, arg1, arg2, arg3; ok &= JS_ValueToInt32(cx, *argvp++, &arg0); ok &= JS_ValueToInt32(cx, *argvp++, &arg1); ok &= JS_ValueToInt32(cx, *argvp++, &arg2); ok &= JS_ValueToInt32(cx, *argvp++, &arg3); if( ! ok ) return JS_FALSE; ccTexParams param = { arg0, arg1, arg2, arg3 }; cobj->setTexParameters(¶m); JS_SET_RVAL(cx, vp, JSVAL_VOID); return JS_TRUE; } JS_ReportError(cx, "wrong number of arguments: %d, was expecting %d", argc, 4); return JS_FALSE; } JSBool js_cocos2dx_CCMenu_alignItemsInRows(JSContext *cx, uint32_t argc, jsval *vp) { 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); JS_SET_RVAL(cx, vp, JSVAL_VOID); return JS_TRUE; } return JS_FALSE; } JSBool js_cocos2dx_CCMenu_alignItemsInColumns(JSContext *cx, uint32_t argc, jsval *vp) { 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); JS_SET_RVAL(cx, vp, JSVAL_VOID); return JS_TRUE; } return JS_FALSE; } // CCTMXLayer JSBool js_cocos2dx_CCTMXLayer_getTileFlagsAt(JSContext *cx, uint32_t argc, jsval *vp) { jsval *argv = JS_ARGV(cx, vp); JSBool ok = JS_TRUE; 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; CCPoint arg0; ok &= jsval_to_ccpoint(cx, argv[0], &arg0); if (!ok) { // TODO: Need to use JSB_PRECONDITION2 like which was done in cocos2d-iphone. return JS_FALSE; } 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; } //#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); CCDrawNode* cobj = (CCDrawNode*)(proxy ? proxy->ptr : NULL); 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); if( ! (argArray && JS_IsArrayObject(cx, argArray) ) ) { JS_ReportError(cx, "Vertex should be anArray object"); return JS_FALSE; } // 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); if( ! ok ) { return JS_FALSE; } { uint32_t l; if( ! JS_GetArrayLength(cx, argArray, &l) ) return JS_FALSE; CCPoint* verts = new CCPoint[ l ]; CCPoint p; for( uint32_t i=0; idrawPolygon(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); return JS_FALSE; } 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); 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); 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); 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); 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); 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); 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); JS_DefineFunction(cx, js_cocos2dx_CCTMXLayer_prototype, "getTileFlagsAt", js_cocos2dx_CCTMXLayer_getTileFlagsAt, 1, JSPROP_READONLY | JSPROP_PERMANENT); 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); JS_DefineFunction(cx, js_cocos2dx_CCTexture2D_prototype, "setTexParameters", js_cocos2dx_CCTexture2D_setTexParameters, 4, JSPROP_ENUMERATE | JSPROP_SHARED | JSPROP_PERMANENT); 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); 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); 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); 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); 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); JS_DefineFunction(cx, js_cocos2dx_CCMenuItem_prototype, "setCallback", js_cocos2dx_CCMenuItem_setCallback, 2, JSPROP_READONLY | JSPROP_PERMANENT); JS_DefineFunction(cx, js_cocos2dx_CCTMXLayer_prototype, "getTileFlagsAt", js_cocos2dx_CCTMXLayer_tileFlagsAt, 2, JSPROP_READONLY | JSPROP_PERMANENT); tmpObj = JSVAL_TO_OBJECT(anonEvaluate(cx, global, "(function () { return cc.Node.prototype; })()")); JS_DefineFunction(cx, tmpObj, "unscheduleAllCallbacks", js_cocos2dx_CCNode_unscheduleAllSelectors, 1, JSPROP_READONLY | JSPROP_PERMANENT); JS_DefineFunction(cx, tmpObj, "copy", js_cocos2dx_CCNode_copy, 1, JSPROP_READONLY | JSPROP_PERMANENT); 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; })()")); JS_DefineFunction(cx, tmpObj, "_create", js_cocos2dx_CCMenuItemToggle_create, 1, JSPROP_READONLY | JSPROP_PERMANENT); 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); 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); 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); JS_DefineFunction(cx, ns, "unregisterTouchDelegate", js_cocos2dx_JSTouchDelegate_unregisterTouchDelegate, 1, JSPROP_READONLY | JSPROP_PERMANENT); 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); JS_DefineFunction(cx, ns, "pAdd", js_cocos2dx_ccpAdd, 1, JSPROP_READONLY | JSPROP_PERMANENT); JS_DefineFunction(cx, ns, "pDistance", js_cocos2dx_ccpDistance, 1, JSPROP_READONLY | JSPROP_PERMANENT); 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); JS_DefineFunction(cx, ns, "pClamp", js_cocos2dx_ccpClamp, 2, JSPROP_READONLY | JSPROP_PERMANENT); 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); // 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); }