cocos2d-x 3.0rc0 范例的 simple_test 的环境太单纯了,并且没有用到 cocos2d 内的任何物件,故!在撰写自己的 .ini 档时会遇上一狗票问题,于是要回头参考 tools/tolua 底下 cocos2d-x 本身的 lua binding 设定,这个说明能详细点吗? ~___~
把遇到的几个问题列一下,供大夥儿参考一下。
问题1:
于 Win7 或 Win8 环境下,如果自订的类别有参考到 cocos2d-x 类,写完自个儿的 .ini 档后俐落的直接执行,会在执行 .bat 后看到类似的错误:
Errors in parsing headers:
1. <severity = Warning,
location = <SourceLocation file None, line 178, column 9>,
details = "'__WCHAR_MAX__' macro redefined">
2. <severity = Fatal,
location = <SourceLocation file '../../../Evo/include/TalkWidgety.h', line 38, column 10>,
details = "'cocos2d.h' file not found">
好,这个很直觉,请在 .ini 设定档里加上 cocos2d-x headers 的路径。
于是,就近搬移 cocos2dx.ini 里对于 cocos2d-x 内路径的设定。当然,如果还有用到其他的如 ui 或是 cocostudio 的话,要一并移进来。
待会儿会在下头将完成的范例贴一下就会明白这儿在说些啥玩意。
问题2:
搬移完 cocos2d-x 等相关的 header 路径后心想这应该就搞定了吧?
BUT!天底下哪有这么便宜的事!
执行后赫然喵到下面的错误…… TAT
Errors in parsing headers:
1. <severity = Warning,
location = <SourceLocation file None, line 179, column 9>, details = "'__WCHAR_MAX__' macro redefined">
2. <severity = Warning, location = <SourceLocation file '../../../Evo/include/TalkWidgety.h', line 27, column 9>,
details = '#pragma once in main file'>
3. <severity = Fatal,
location = <SourceLocation file 'D:../../../cocos/base\\CCConsole.h', line 30, column 10>,
details = "'BaseTsd.h' file not found">
惨!这下该怎么办……
花了点时间看了下产生 cocos2d-x 的 genbindings.py 才恍然大悟,原来要加上 -U__MINGW32__ 的 undef 来避免问题。
原因?自己去看 CCConsole.h 的错误点就知道问题在哪了!这里不想多说,伤心啊!
解决以上两个问题的 .ini 如下,仅供参考:
--------------------------------------------------------------------------------------------
[evo]
prefix = evo_bindings
target_namespace = evo
cpp_namespace = Evo
# 自定义的变数,偷懒用
evo_root = ../../../../Evo
evo_header = %(evo_root)s/include
cocosdir = ../../../
android_headers = -I%(androidndkdir)s/platforms/android-14/arch-arm/usr/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.7/libs/armeabi-v7a/include -I%(androidndkdir)s/sources/cxx-stl/gnu-libstdc++/4.7/include
android_flags = -D_SIZE_T_DEFINED_
clang_headers = -I%(clangllvmdir)s/lib/clang/3.3/include
clang_flags = -nostdinc -x c++ -std=c++11
# 从 cocos2dx.ini 等搬过来的
cocos_headers = -I%(cocosdir)s/cocos/2d -I%(cocosdir)s/cocos/ui -I%(cocosdir)s/cocos/base -I%(cocosdir)s/cocos/physics -I%(cocosdir)s/cocos/2d/platform -I%(cocosdir)s/cocos/2d/platform/android -I%(cocosdir)s/cocos/math/kazmath -I%(cocosdir)s/extensions -I%(cocosdir)s/external -I%(cocosdir)s/cocos/editor-support -I%(cocosdir)s
cocos_flags = -DANDROID -U__MINGW32__
evo_headers = -I%(evo_root)s/include
extra_arguments = %(android_headers)s %(clang_headers)s %(cocos_headers)s %(evo_headers)s %(android_flags)s %(clang_flags)s %(cocos_flags)s %(extra_flags)s
headers = %(evo_header)s/TalkWidget.h
classes = TalkWidget
remove_prefix =
skip =
base_objects =
abstract_classes =
classes_have_type_info = no
rename =
rename_functions =
rename_classes =
# 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 =
# Determining whether to use script object(js object) to control the lifecycle of native(cpp) object or the other way around. Supported values are 'yes' or 'no'.
script_control_cpp = no
---------------------------------------------------------------------------------------
问题3:
如果自定义的类别里有自行定义的 namespace 时,跑 .bat 后看 output log 的最后一行会发现这样的错误:
Exception: The namespace (Evo::TalkWidget) conversion wasn't set in 'ns_map' section of the conversions.yaml
好吧,看错误内容……来去找一下 conversions.yaml 这个档,位置应该在 /tools/bindings-generator/targets/lua 底下,直接用文字编辑器打开来看。
我勒了个去!果然要把自订的 namespace 加到 ns_map 的变数中,那个 .ini 里的 cpp_namespace 作用果然只是用来卖萌的吧....=A=
要加的内容如下:
---------------------------------------------------------------------------------------
ns_map:
"cocos2d::extension::": "cc."
"cocos2d::ui::": "ccui."
"cocos2d::": "cc."
"spine::": "sp."
"cocostudio::": "ccs."
"cocosbuilder::": "cc."
"CocosDenshion::": "cc."
"Evo::": "evo."
---------------------------------------------------------------------------------------
问题4:
解决以上三个问题后,我的老天爷啊!很尴尬的是居然又遇上一个鸟问题,错误如下:
File ../../tools/bindings-generator/evo/../generator.py", line 220, in from_type
(ret_type, params) = r.groups()
AttributeError: 'NoneType' object has no attribute 'groups'
……无言以对。
花了点时间看了下 generator.py 才发现,原来在 generator.py 的 line 219 :
r = re.compile('function<(\S+) \((.*)\)>').search(cdecl.displayname)
这里并没有考虑到会有人在类别里写 std::function<Object*()> 或 std::function<Object&()> 这样回传物件 pointer 或是 reference 的 functor,所以 regex 解析到坏掉。
也就是说,如果你的 code 里写了像这样的,请换个写法吧!
class TalkWidget : public cocos2d::ui::Widget
{
public:
....
bool setParseMethod(std::function<std::string&(const std::string&, const int)> &func);
....
};
如上面的 setParseMethod, func 是一个会回传 std::string reference ( std::string& ) 的 functor,这样子写就不行!
把整个 cocos2d-x 3.0rc0 找了遍……赫!果然整个 cocos2d-x 3.0rc0 里的 function functor 都是回传标准型别或干脆就是不回传任何玩意的 void 。唉!真他喵的,踩到雷了……
由于不想多花时间去看整个 generator.py ,所以,山不转路转,换个写法就好了。 =___=
[2014-5-2 注:到目前为止 3.0 正式版,完全没有解决 c++11 functional 与 lambda function 参数自动绑定到 lua 中的任何解决方法,所以,问题四就直接无视吧!]
但是,麻烦有看到的大大请帮忙提醒一下要改正这个问题啊!这个是明显的 bug 而不是说明不详细了!
花了整整一下午才搞定,好累……看到这里,希望多少能提供点帮助给遇上一样问题的人。
以上!