CocosCreator JSB 绑定

小白教程, 如有错误,欢迎指正

原文地址

版本信息

Mac 10.13.5
Xcode 9.4.1
Cocos Creator 1.9.2
Python 2.7

步骤

新建 Hello World 工程,项目名称demo

使用Xcode新建两个.cpp文件, 分别取名jsToCPPAoneClient

jsToCPP.h 内容如下

#ifndef jsToCPP_h
#define jsToCPP_h

#include <stdio.h>
#include <string>
#include <map>
#include <vector>
#include <functional>

using namespace std;
class jsToCPP
{
public:
    //  单例
    static jsToCPP* getInstance();
    
    // 静态方法
    static void static_func();
    
    // 数组
    void setArray(string str);
    vector<string> getArray();
    
    // map
    void setMap(map<string, string> param);
    map<string, string> getMap();
    
    // 回调函数
   void initAsync(const function<void(int, string)>& cb);
    
    template<typename TypeOne, typename TypeTwo>
    string mapToString(map<TypeOne, TypeTwo>& m);
public:
    int age;
    function<void(int, string)> _initCb;
    
private:
    string getString(const int32_t& a);
    string getString(const int64_t& a);
    string getString(const string& s);
    
private:
    vector<string> _mArray;
    map<string, string> _mMap;
};
#endif /* jsToCPP_h */

jsToCpp.cpp 内容如下

#include "jsToCPP.h"
#include "AoneClient.h"

jsToCPP* jsToCPP::getInstance()
{
    static jsToCPP _instance;
    return &_instance;
}

void jsToCPP::static_func()
{
    printf("--- <%s : %d>\n", __func__, __LINE__);
}

void jsToCPP::setArray(string str)
{
    printf("--- <%s : %d> array add = %s\n", __func__, __LINE__, str.c_str());
    _mArray.push_back(str);
}
vector<string> jsToCPP::getArray()
{
    return _mArray;
}


void jsToCPP::setMap(map<string, string> param)
{
    map<string, string>::iterator iter;
    for(iter = param.begin(); iter != param.end(); iter++) {
        printf("--- <%s : %d> key = %s\n", __func__, __LINE__, iter -> first.c_str());
        printf("--- <%s : %d> valut = %s\n", __func__, __LINE__, iter -> second.c_str());
        _mMap.insert(pair<string, string>(iter -> first, iter -> second));
    }
}
map<string, string> jsToCPP::getMap()
{
    return _mMap;
}


void _initCallback(int code, std::map<string, string> dataMap)
{
   jsToCPP* obj = jsToCPP::getInstance();

   string mapStr = obj -> mapToString(dataMap);

   obj -> _initCb(code, mapStr);
}

void jsToCPP::initAsync(const function<void(int, string)>& cb)
{
   _initCb = cb;

   AoneClient *client = new AoneClient();

   client -> initAsync(_initCallback);
}

template<typename TypeOne, typename TypeTwo>
string jsToCPP::mapToString(map<TypeOne, TypeTwo>& m)
{
    if(m.empty()) return "";
    string str = "{";
    typename map<TypeOne, TypeTwo>::iterator it;
    for (it = m.begin(); it != m.end(); it++)
    {
        str += "\"" + getString(it -> first) + "\":\"" + getString(it -> second) + "\"";
        str += ",";
    }
    // delete the last ","
    str = str.substr(0, str.length()-1);
    str += "}";
    return str;
}

// =====   Private Method

string jsToCPP::getString(const int32_t& a)
{
    char c[1024] = {0};
    snprintf(c, sizeof(c), "%d", a);
    return c;
}

string jsToCPP::getString(const int64_t& a)
{
    char c[1024] = {0};
    snprintf(c, sizeof(c), "%lld", a);
    return c;
}

string jsToCPP::getString(const string& s)
{
    return s;
}

AoneClient.h 内容如下

#ifndef AoneClient_h
#define AoneClient_h

#include <stdio.h>
#include <string>
#include <map>

using namespace std;

typedef void(*AONESDK_CB)(int result, map<string, string> dataMap);

class AoneClient
{
public:
    AoneClient();
    ~AoneClient();

    void initAsync(AONESDK_CB cb);

private:
    AONESDK_CB _initCb;
};


#endif /* AoneClient_h */

AoneClient.cpp 内容如下

#include "AoneClient.h"


AoneClient::AoneClient()
{
    printf("--- <%s : %d>\n", __func__, __LINE__);
}

AoneClient::~AoneClient()
{
    printf("--- <%s : %d>\n", __func__, __LINE__);
}

void AoneClient::initAsync(AONESDK_CB cb)
{
    _initCb = cb;
    assert(_initCb);
    printf("--- <%s : %d>\n", __func__, __LINE__);
    // 模拟初始化操作
    map<std::string, std::string> dataMap;
    dataMap.insert(pair<string, string>("name", "chenxi"));
    dataMap.insert(pair<string, string>("sex", "man"));
    dataMap.insert(pair<string, string>("skill", "Object-C"));
    dataMap.insert(pair<string, string>("type", "facebook"));
    int code = 1;
    
    _initCb(code, dataMap);
}

进入CocosCreator软件目录下,在cocos2d-x目录下,创建jsToCPP文件夹, 该文件夹的绝对路径为:/Applications/CocosCreator.app/Contents/Resources/cocos2d-x/jsToCPP

拷贝创建好的jsToCPP.hjsTocPP.cppAoneClient.hAoneClient.cpp 文件到 jsToCPP 目录

jsToCPP-files

cocos2d-x/tools/tojs/目录下,新建jsbToCPP.ini文件,内容如下

# 模块名称
[jsToCPP]

# 绑定回调函数的前缀,也是生成的自动绑定文件的前缀
prefix = jsToCPP

# 绑定的类挂载在 JS 中的哪个对象中,类似命名空间
target_namespace = 

# 自动绑定工具基于 Android 编译环境,此处配置 Android 头文件搜索路径
android_headers = -I%(androidndkdir)s/platforms/android-14/arch-arm/usr/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.8/libs/armeabi-v7a/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.8/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.9/libs/armeabi-v7a/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.9/include

# 配置 Android 编译参数
android_flags = -D_SIZE_T_DEFINED_ 

# 配置 clang 头文件搜索路径
clang_headers = -I%(clangllvmdir)s/%(clang_include)s 

# 配置 clang 编译参数
clang_flags = -nostdinc -x c++ -std=c++11 -U __SSE__

# 配置引擎的头文件搜索路径
cocos_headers = -I%(cocosdir)s -I%(cocosdir)s/cocos/editor-support -I%(cocosdir)s/cocos -I%(cocosdir)s/cocos/platform/android -I%(cocosdir)s/external/sources

# 配置引擎编译参数
cocos_flags = -DANDROID

cxxgenerator_headers = 

# extra arguments for clang
extra_arguments = %(android_headers)s %(clang_headers)s %(cxxgenerator_headers)s %(cocos_headers)s %(android_flags)s %(clang_flags)s %(cocos_flags)s %(extra_flags)s 

# 需要自动绑定工具解析哪些头文件
headers = %(cocosdir)s/jsToCPP/jsToCPP.h

# 在生成的绑定代码中,重命名头文件
replace_headers = 

# 需要绑定哪些类,可以使用正则表达式,以空格为间隔
classes = jsToCPP

# what should we skip? in the format ClassName::[function function]
# ClassName is a regular expression, but will be used like this: "^ClassName$" functions are also
# regular expressions, they will not be surrounded by "^$". If you want to skip a whole class, just
# add a single "*" as functions. See bellow for several examples. A special class name is "*", which
# will apply to all class names. This is a convenience wildcard to be able to skip similar named
# functions from all classes.

skip = 

rename_functions =

rename_classes = 

# for all class names, should we remove something when registering in the target VM?
remove_prefix = 

# classes for which there will be no "parent" lookup
classes_have_no_parents = 

# base classes which will be skipped when their sub-classes found them.
base_classes_to_skip = 

# classes that create no constructor
# Set is special and we will use a hand-written constructor
abstract_classes = 

classes_need_extend =

cocos2d-x目录下,找到/tools/tojs/genbindings.py文件, 在其中添加如下字段

        cmd_args = {
                    'cocos2dx.ini' : ('cocos2d-x', 'jsb_cocos2dx_auto'), \
                    'cocos2dx_audioengine.ini' : ('cocos2dx_audioengine', 'jsb_cocos2dx_audioengine_auto'), \
                    'cocos2dx_network.ini' : ('cocos2dx_network', 'jsb_cocos2dx_network_auto'), \
                    'cocos2dx_extension.ini' : ('cocos2dx_extension', 'jsb_cocos2dx_extension_auto'), \
                    'cocos2dx_ui.ini' : ('cocos2dx_ui', 'jsb_cocos2dx_ui_auto'), \
                    'cocos2dx_spine.ini' : ('cocos2dx_spine', 'jsb_cocos2dx_spine_auto'), \
                    'cocos2dx_dragonbones.ini' : ('cocos2dx_dragonbones', 'jsb_cocos2dx_dragonbones_auto'), \
                    'cocos2dx_experimental_webView.ini' : ('cocos2dx_experimental_webView', 'jsb_cocos2dx_experimental_webView_auto'), \
                    'cocos2dx_experimental_video.ini' : ('cocos2dx_experimental_video', 'jsb_cocos2dx_experimental_video_auto'), \
                    'creator.ini': ('creator', 'jsb_creator_auto'),
                    'creator_camera.ini': ('creator', 'jsb_creator_camera_auto'),
                    'creator_graphics.ini': ('creator', 'jsb_creator_graphics_auto'),
                    'creator_physics.ini': ('creator', 'jsb_creator_physics_auto'),
                    'box2d.ini' : ('box2d', 'jsb_box2d_auto'),
                    'anysdk-common.ini': ('protocols', 'jsb_anysdk_protocols_auto'),
                    'anysdk-appstore.ini': ('protocols', 'jsb_anysdk_protocols_auto'),
                    }

        # just generate custom file
        cmd_args.clear()
        cmd_args["jsToCPP.ini"] = ('jsToCPP', 'jsb_jsToCPP_auto')

其作用是,为节省运行时间,只需要生成我们自定义的代码

进入 genbindings.py 文件所在目录

运行该Python脚本: python genbindings.py

运行该脚本可能遇到如下错误

1. NDK 错误
NDK_ROOT not defined. Please define NDK_ROOT in your environment.

解决方法:

1. open .bash_profile
2. 添加NDK路径
2. Python yaml 模块错误
Traceback (most recent call last):
  File "/Applications/CocosCreator.app/Contents/Resources/cocos2d-x/tools/bindings-generator/generator.py", line 11, in <module>
    import yaml
ImportError: No module named yaml
-------------------------------------
Generating javascript bindings fails.
-------------------------------------

解决方法:

sudo easy_install  pyyaml
3. Python Cheetah
Traceback (most recent call last):
  File "/Applications/CocosCreator.app/Contents/Resources/cocos2d-x/tools/bindings-generator/generator.py", line 16, in <module>
    from Cheetah.Template import Template
ImportError: No module named Cheetah.Template
-------------------------------------
Generating javascript bindings fails.
-------------------------------------

解决方式:

方法一:
1. [下载cheetah](http://pythonhosted.org//Cheetah/)
2. 解压
3. 进入根目录运行setup.py文件
   sudo python setup.py install
   
方法二:
sudo pip install Cheetah

运行之后,出现如下所示提示, 表示绑定成功:

>>> parse_header: /Applications/CocosCreator.app/Contents/Resources/cocos2d-x/jsToCPP/jsToCPP.h
----------------------------------------
Generating javascript bindings succeeds.
----------------------------------------

找到/Applications/CocosCreator.app/Contents/Resources/cocos2d-x/cocos/scripting/js-bindings/auto 目录,发现会生成jsb_jsToCPP_auto.hppjsb_jsToCPP_auto.cpp 文件, 该文件,后续需要添加到cocos2ds_js_bindings.xcodeproj, 后面再说。

找到 /Applications/CocosCreator.app/Contents/Resources/cocos2d-x/cocos/scripting/js-bindings/auto/api/ 目录, 发现会生成 jsb_jsToCPP_auto_api.js 文件,拷贝该文件到/Applications/CocosCreator.app/Contents/Resources/cocos2d-x/cocos/scripting/js-bindings/auto/script/ 目录下

打开demo工程,找到Helloworld.js文件,在该文件里面调用CPP文件, 参考代码

// use this for initialization
    onLoad: function () {
        if (cc.sys.platform !== cc.sys.IPHONE) {
          console.log("\n--- on load yet, not run on iPhone");
          return;
        }

        console.log("\n\n--- begin test static func");
        jsToCPP.static_func();

        var cppObj = jsToCPP.getInstance();
        console.log("old jsToCPP.age: " + cppObj.age);
        cppObj.age = 12;
        console.log("new jsToCPP.age: " + cppObj.age);


        // test array
        console.log("\n\n--- begin test array");
        cppObj.setArray("ont");
        cppObj.setArray("two");
        cppObj.setArray("three");
        var array = cppObj.getArray();
        if (array instanceof Array) {
          console.log("--- array data = [" + array.join(", ") + "]");
        }

        // test map
        console.log("\n\n--- begin test map");
        var mapObj = {
          "name": "chen xi",
          "result": "success",
          "msg": "this is a test message"
        };
        cppObj.setMap(mapObj);
        var map = cppObj.getMap();
        console.log("--- type ", map);
        // convert to json string
        var jsonData = JSON.stringify(map);
        console.log("--- map data = " + jsonData);

        // test callback
        console.log("\n\n--- begin test callback");
        var delegateObj = {
          onInitCallback: function (code, msg) {
            console.log("delegate obj, onCallback: " + code + ", msg = " + msg);
          },
        }
        cppObj.initAsync(delegateObj.onInitCallback);
    }

构建项目: CocosCreator -> 目 -> 构建发布, 选择发布到iOS平台,构建完成之后,在该demo工程的根目录下会生成一个build文件夹, 找到编译生成的Xcode工程( build/jsb-link/frameworks/runtime-src/proj.ios_mac/hello_world.xcodeproj) , 打开该Xcode工程, 在Classes目录下, 找到 jsb_module_register.cpp 文件, 进行如下操作

1.引入生成的绑定文件: #include "cocos/scripting/js-bindings/auto/jsb_jsToCPP_auto.hpp"

2.在bool jsb_register_all_modules() 方法中,注册CPP se->addRegisterCallback(register_all_jsToCPP);

bool jsb_register_all_modules()
{
    se::ScriptEngine* se = se::ScriptEngine::getInstance();

    se->addBeforeInitHook([](){
        JSBClassType::init();
    });

    se->addBeforeCleanupHook([se](){
        se->garbageCollect();
        PoolManager::getInstance()->getCurrentPool()->clear();
        se->garbageCollect();
        PoolManager::getInstance()->getCurrentPool()->clear();
    });

    se->addRegisterCallback(jsb_register_global_variables);

    se->addRegisterCallback(run_prepare_script);

    se->addRegisterCallback(register_all_cocos2dx);
    se->addRegisterCallback(jsb_register_Node_manual);
    se->addRegisterCallback(register_all_cocos2dx_manual);
    se->addRegisterCallback(JSB_register_opengl);
    se->addRegisterCallback(register_all_jsToCPP);

   // ..... 
}

展开coco2d_js_bindings.xcodeproj, 找到 auto 目录,添加生成的jsb_jsToCPP_auto.hpp js_jsToCPP_auto.cpp 文件到该工程

添加 jsToCPP 文件夹下的所有文件到该工程, 并添加正确的搜索路径

add-file-to-js_bindings-project

真机运行项目,出现如下信息:

--- begin test static func
--- <static_func : 19>
JS: old jsToCPP.age: undefined
JS: new jsToCPP.age: 12
JS: 

--- begin test array
--- <setArray : 24> array add = ont
--- <setArray : 24> array add = two
--- <setArray : 24> array add = three
JS: --- array data = [ont, two, three]
JS: 

--- begin test map
--- <setMap : 37> key = msg
--- <setMap : 38> valut = this is a test message
--- <setMap : 37> key = name
--- <setMap : 38> valut = chen xi
--- <setMap : 37> key = result
--- <setMap : 38> valut = success
JS: --- type  [object Object]
JS: --- map data = {"msg":"this is a test message","name":"chen xi","result":"success"}
JS: 

--- begin test callback
--- <AoneClient : 6>
--- <initAsync : 18>
JS: delegate obj, onCallback: 1, msg = {"name":"chenxi","sex":"man","skill":"Object-C","type":"facebook"}

完成!!!

参考链接
  1. mac下运行CocosCreator的genbindings.py出现的错误
  2. JSB 2.0 绑定教程
  3. No type named 'function' in namespace std
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容