在cocos2dx的ui编辑器cocostudio有一套相关的api:
@H_301_3@1、表示各个UI控件的类,如LoadingBar,ListView等等,其都继承自Widget
@H_301_3@2、有相关各个UI控件解析的类,由于cocos提供UI编辑器,方便了开发者快速制作UI,动画等,载入游戏需要相关解析支持。
从cocos2dx的实现来看,通过ObjectFactory实现了,自动注册,将类型与创建UI接口对应,在UI解析方面,将类型与解析类别对应,都是
通过ObjectFactory来实现的,下面我看看相关分析:
@H_301_3@1、首先我们要将类型与相应的UI创建方法建立对应,那么需要一个数据结构,在ObjectFactory里有一个叫TInfo,
typedef cocos2d::Ref* (*Instance)(void); struct TInfo { TInfo(void); TInfo(const std::string& type,Instance ins = NULL); TInfo(const TInfo &t); ~TInfo(void); TInfo& operator= (const TInfo &t); std::string _class; Instance _fun; };
而Instance是一个指针类型,其定义:
typedef cocos2d::Ref* (*Instance)(void);
在TInfo的构造函数里,有如下实现:
ObjectFactory::TInfo::TInfo(const std::string& type,Instance ins) :_class(type),_fun(ins) { ObjectFactory::getInstance()->registerType(*this); }
由于是ObjectFactory是一个单例,只要是通过该构造函数,通过ObjectFactory就可以方便的访问。那么又是这么实现自动注册的。我们
知道,要想达到自动的目的,其中一个办法就是全局对象或者静态对象的构造函数里
在GUIDefine.h下有3个宏
#define DECLARE_CLASS_GUI_INFO \ public: \ static cocos2d::ObjectFactory::TInfo __Type; \ static cocos2d::Ref* createInstance(void); \ #define IMPLEMENT_CLASS_GUI_INFO(className) \ cocos2d::Ref* className::createInstance(void) \ { \ return className::create(); \ } \ cocos2d::ObjectFactory::TInfo className::__Type(#className,&className::createInstance); \ #define CREATE_CLASS_GUI_INFO(className) \ cocos2d::ObjectFactory::TInfo(#className,&className::createInstance) \
这是一个声明宏以及对于的实现宏,熟悉MFC的同学肯定感到脸熟。从中看到,一个变量,一个方法都声明成静态,都是静态成员,我们
知道静态成员都是属于类的。
在该文件的下面,还有专门给UIReader的宏,差别就是createInstance里的实现,不一样,UI是通过create,而UIReader是通过getInstance
#define DECLARE_CLASS_WIDGET_READER_INFO \ public: \ static cocos2d::ObjectFactory::TInfo __Type; \ static cocos2d::Ref* createInstance(void); \ #define IMPLEMENT_CLASS_WIDGET_READER_INFO(className) \ cocos2d::Ref* className::createInstance(void) \ { \ return className::getInstance(); \ } \ cocos2d::ObjectFactory::TInfo className::__Type(#className,&className::createInstance); \ #define CREATE_CLASS_WIDGET_READER_INFO(className) \ cocos2d::ObjectFactory::TInfo(#className,&className::createInstance) \
看看是如何自动注册的,我们看看这些宏都在那些地方使用了。
在所有ui扩展里的ui类实现,都会有
class UIClass : public Widget { // 注意这里 DECLARE_CLASS_GUI_INFO ...
在相应的cpp文件开头都有
IMPLEMENT_CLASS_GUI_INFO(CheckBox)
所以说每个UI类,都有一个静态成员__Type,他保存着该UI类型和相应的创建方法,并注册到ObjectFactory单例中。
同上,UIReader也用了类似的方式:
class LayoutReader : public WidgetReader { public: DECLARE_CLASS_WIDGET_READER_INFO
这样的话,当程序启动之后,所有的注册工作都自动处理好了。
关于最后一个CREATE宏,发现其调用的地方,是GUIReader初始化 的时候,但是从上面分析,这些注册过程都是自动化的,这里主动调用是为何?
GUIReader::GUIReader(): m_strFilePath("") { ObjectFactory* factoryCreate = ObjectFactory::getInstance(); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(ButtonReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(CheckBoxReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(SliderReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(ImageViewReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(LoadingBarReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(TextAtlasReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(TextReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(TextBMFontReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(TextFieldReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(LayoutReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(PageViewReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(ScrollViewReader)); factoryCreate->registerType(CREATE_CLASS_WIDGET_READER_INFO(ListViewReader)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(Button)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(CheckBox)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(ImageView)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(Text)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(TextAtlas)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(TextBMFont)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(LoadingBar)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(Slider)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(TextField)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(Layout)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(ListView)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(PageView)); factoryCreate->registerType(CREATE_CLASS_GUI_INFO(ScrollView)); }通过调试程序,跟踪流程,的确类型注册进入了多次,一个BUG? 还是有其他原因?