浅析cocos2d-x 2.x到3.x事件侦听设计变化原因之一

DionysosLai 2015/2/28

对于事件侦听的设计,cocos2d-x从2.x到3.x发生了根本性的变化,一直以来对此,只是单纯的使用考虑如何构建自己的游戏代码,并未对其二者设计孰优孰劣进行探究。只是前段时间在做一个新游戏时,关于2.x的触摸事件发现了一个设计不人性化问题,本想向cocos2dx官网反应,但测试3.x时,并未发现这个问题。对此,本文细述这个问题,分析二者设计的不同,同时希望抛砖引玉。

简单阐述2.x与3.x的事件侦听不同点:

在2.x中,需要注册一个事件侦听。

在3.x中,则需要创建一个侦听器的对象,然后定义回调方法,最后将侦听和事件分发器绑定。

注意,2.x是注册一个事件侦听,在3.x中则是创建一个侦听器对象。这二者有什么不同呢?这里分别在2.x中和3.x写一个测试demo,可以看到2.x中一个严重设计bug问题。

2.x代码如下:

.h
    DelegateTest();
    virtual ~DelegateTest();
    virtual void keyBackClicked();

    .cpp
    DelegateTest::DelegateTest(){}
.cpp
DelegateTest::~DelegateTest()
{
    CCLOG("DelegateTest release!!");
}

bool DelegateTest::init()
{
    if (!CCLayer::init())
    {
        return false;
    }

    this->setKeypadEnabled(true);

    CCDirector::sharedDirector()->getTouchDispatcher()->addTargetedDelegate(this, 0 ,true);
    return true;
}

void DelegateTest::keyBackClicked()
{
    CCScene* sceneGame = DelegateTest::scene();
    CCDirector::sharedDirector()->replaceScene(sceneGame);
    CCLog("keyBackClicked!!!");
}

这里,是一个简单的触摸侦听事件事件demo,在init()中添加注册触摸事件,同时使能返回键功能,同时按下返回键时,切换界面。

先理论分析调试信息,应该是如下所示:

按下返回键:

keyBackClicked!!!

DelegateTest release!!

再次按下返回键:

keyBackClicked!!!

DelegateTest release!!

……

事件呢,调试信息如下:

按下返回键:

keyBackClicked!!!

再次按下返回键:

keyBackClicked!!!

退出游戏:

DelegateTest release!!

DelegateTest release!!

看到区别了吗!出现这种现象的结果什么呢。就是当前场景并未移除掉,同样能接收触摸,当不止一个场景出现时,其逻辑就会出现紊乱!!!

现在代码移植到3.x中。

3.x代码如下:

.h
    DelegateTest();
    virtual ~DelegateTest();
    virtual void onKeyReleased(cocos2d::EventKeyboard::KeyCode keyCode, cocos2d::Event* event) override;
    cocos2d::EventListenerKeyboard* m_keyboardListener;     // 键盘listener
    cocos2d::EventListenerTouchOneByOne* m_touchListener    ;   // 触摸listener

.cpp
DelegateTest::DelegateTest(){}

DelegateTest::~DelegateTest()
{
    log("DelegateTest release!!");
}

bool DelegateTest::init()
{
    if (!CCLayer::init())
    {
        return false;
    }

    m_keyboardListener = EventListenerKeyboard::create();
    m_keyboardListener->retain();
    m_keyboardListener->onKeyReleased = CC_CALLBACK_2(DelegateTest::onKeyReleased, this);
    getEventDispatcher()->addEventListenerWithSceneGraphPriority(m_keyboardListener, this);

    //setKeypadEnabled(true); ----不再适用
    m_touchListener = EventListenerTouchOneByOne::create();
    m_touchListener->retain();
    m_touchListener->onTouchBegan = CC_CALLBACK_2(DelegateTest::onTouchBegan, this);
    m_touchListener->onTouchBegan = [&](cocos2d::Touch* touch, cocos2d::Event* event){
        return true;
    };
    getEventDispatcher()->addEventListenerWithSceneGraphPriority(m_touchListener, this);

    return true;
}

void DelegateTest::onKeyReleased( cocos2d::EventKeyboard::KeyCode keyCode, cocos2d::Event* event )
{
    Scene* sceneLayer = DelegateTest::scene();
    Director::getInstance()->replaceScene(sceneLayer);
    log("onKeyReleased!!!");
}

功能与前面描述的一模一样。调试信息如下所示:

onKeyReleased!!!

DelegateTest release!!

onKeyReleased!!!

DelegateTest release!!

与我们要的结果一致。为何会出现这两种不同的情况呢。这是由于2.x中,是注册侦听事件,因此我们必须对应的要注销侦听事件。而3.x是创建侦听器对象,这样析构时,会自动移除对象,从设计上来说,3.x比2.x更加的容易,且不容易出错。

那么对于2.x要如何更改,使之出现的结果跟我们理论上一致呢?这里更改函数keyBackClicked()代码即可:

void DelegateTest::keyBackClicked()
{
    CCScene* sceneGame = DelegateTest::scene();
    CCDirector::sharedDirector()->replaceScene(sceneGame);
    CCDirector::sharedDirector()->getTouchDispatcher()->removeDelegate(this);
    CCLog("keyBackClicked!!!");
}

当然,理论上3.x退出场景时,也要移除侦听器对象,函数是release(),这里省咯了而已。也许大家会说,不就是在2.x中添加一句话而已,而且忘记注销,本身就是自己的问题。这里首先要说一点,就是当layer比较多时,注销触摸并不是一件简单的事情,同时,退出场景可能在一个子类中调用;也可能在另一个layer中,要求关闭当前场景中所有layer,比如暂停界面。另一个面,当使用到popScene和pushScene时,这更加是一个灾难!!!对于一个游戏引擎,良好的设计应该能主动帮忙开发者避免以坑。这也是为什么在cocos2d-x中渲染时,为何不专门开一个线程,实现逻辑与展示分开。因为这样太难了,会导致大量细节问题,无形中加大开发者难度,当然,随着以后引擎开发,相信会解决这个问题。详细可以参考王哲回答:http://www.zhihu.com/question/27612727/answer/38078748

对于以上代码细节,可以访问笔者githup:https://github.com/DionysosLai/CocoSamples 中的“Delegate Test”内容。2015年第一篇文章,新年新兆头,祝大家新年更棒!!!—ps:第一次使用markdown编辑器,感觉萌萌哒!!!

时间: 2024-11-08 22:13:51

浅析cocos2d-x 2.x到3.x事件侦听设计变化原因之一的相关文章

Cocos2d-x 《雷电大战》-精灵随手指移动,你点哪我走哪!

林炳文Evankaka原创作品.转载请注明出处http://blog.csdn.net/evankaka 本文要实现飞机游戏中,人的手指按着飞机,就能拖着飞机走动,这里实现了当你手指按在手机的图片上,手指一直按着屏幕,飞机就会跟着你走.同时,还加入了边界判断条件,让飞机在你的视野内移动,实现的效果完全和我们手机上的飞机游戏一样. 效果: Cocos2d-x版本:3.4 工程环境:VS30213 一.代码编写 1.头文件GameMain.h /** *@作者 林炳文(邮箱:[email prote

cocos2dx基础篇(14)——文本框之一CCTextFieldTTF

[引言] 前面我们讲了精灵贴图.标签.菜单.按钮.感觉似乎少了点什么?UI控件里是不是应该还有一个很重要的控件--文本框.在手机网游中,启动游戏,过了开场动画后,基本上显示的第一个界面应该就是游戏的登录界面了吧.输入用户名.密码什么的,这些都是需要借助文本框来实现输入的.点击文本,弹出虚拟键盘,输入账号密码,点击登录. cocos2dx引擎为我们提供了两类文本框的控件: (1)CCTextFieldTTF(基于CCLabelTTF) (2)CCEditBox(基于CCControlButton)

Cocos2d-x 3.0 屏幕触摸及消息分发机制

***************************************转载请注明出处:http://blog.csdn.net/lttree******************************************** 题外话: 唉. 开学了!    好烦. 这就已经大三了, 两年前的这时候,我还是懵懂的大一小学弟, 两年后.就要奔上社会就业了. 光阴似箭.日月如梭呀~ 正文: 好久没做cocos2d-x了,这次练习一下.屏幕触摸及消息分发机制. 这里,我用的是cocos2d-

【转载】cocos2dx基础篇(16)——滚动视图CCScrollView

[唠叨] 本节要讲讲滚动视图CCScrollView,相信玩过手游的同学们应该对它不会陌生吧. 例如:愤怒的小鸟的游戏场景里大大的地图,手机的屏幕肯定无法完全显示的,所以需要通过触摸滚动才能显示大地图的其他区域:排行榜中上下滑动来查看其他玩家的排名:以及手机上主界面左右滑动来切换界面等等. 如下图为屏幕滚动,切换手机的界面. [致谢] http://blog.csdn.net/paea_gulang/article/details/10283601 [Demo下载] https://github

cocos2dx基础篇(16)——滚动视图CCScrollView

[唠叨] 本节要讲讲滚动视图CCScrollView,相信玩过手游的同学们应该对它不会陌生吧. 例如:愤怒的小鸟的游戏场景里大大的地图,手机的屏幕肯定无法完全显示的,所以需要通过触摸滚动才能显示大地图的其他区域:排行榜中上下滑动来查看其他玩家的排名:以及手机上主界面左右滑动来切换界面等等. 如下图为屏幕滚动,切换手机的界面. [致谢] http://blog.csdn.net/paea_gulang/article/details/10283601 http://bbs.firedragonpz

java代码筛选文件

快过节了,谢谢了屈原,我们爱你. 应该多几个向屈大人一样跳江的,这样我们就可以放假纪念啦. ---------------------------------快过节了,弄个案例,大家不妨假期做做, 运行效果展示: 全部代码和资源: http://download.csdn.net/detail/sdhjob/7424329 1.准备资源 背景图片menuback.png: 节点图片 greenstar.png   redstar.png      yellowstar.png  2.创建一个新项

[cocos2dx笔记010]用于UI的事件管理器

cocos2dx有一个编辑器:cocostudio,目前来说,已经是比较好用了,只要加载导出的资源,就可以用上了.省去手动搭建面的麻烦.但是,很多需要事件的地方,操作比较麻烦,所以这里提供一个事件管理器来集中和简化管理事件.对于C++事件委托方面,我这里使用了是FastDelegate(注:一个牛人写的).下面是具体实现的代理,不多. /* UI触摸事件管理器. 原有cocos2dx带的触摸事件,每次监听要操作的步骤比较多,为此增加了一个事件管理器,来集中和简化管理 */ #ifndef _X_

Cocos2d-x 3.0final 终结者系列教程13-贪食蛇游戏案例(全)

快过节了.谢谢了屈原,我们爱你. 应该多几个向屈大人一样跳江的,这样我们就能够放假纪念啦. ---------------------------------快过节了.弄个案例,大家最好还是假期做做, 执行效果展示: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvc2Roam9i/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" > 所有代码和资

Tapable 0.2.8 入门

[原文:Tapable 0.2.8 入门] tapable是webpack的核心框架(4.0以上版本的API已经发生了变化),是一个基于事件流的框架,或者叫做发布订阅模式,或观察者模式,webpack的整个生命周期及其开放的自定义插件系统都离不开tapable的支持,研究其运行原理是阅读webpack源代码的第一步. Tapable是一个小型库,允许你添加和应用插件到一个javascript模块.它可以被继承或混入其他模块.除可以定制事件发射和操作,还可以通过回调参数访问事件的“排放者”或“生产