上篇触摸机制讲解中提到过存在两个按钮同时响应的问题。
问题描述
我们一般使用EventListenerTouchOneByOne注册触摸事件,但是这里的触摸消息是按顺序依次响应的,当你在屏幕中同时点击了两个按钮(A和B),彼此没有交集,触摸机制会依次触发两次触摸消息,一次按钮A的触摸消息,另一次按钮B的触摸消息。重要的是依次触发消息,每次触摸消息都会执行按钮A和按钮B的回调函数,根据回调函数里面的区域判断当前的触摸消息点击到了哪个按钮。这样按钮A和按钮B的回调函数会各执行两次,即使对每个按钮设置setSwallowTouches(true),也只对点击到按钮的那次触摸消息有作用。因为setSwallowTouches只在onTouchBegan返回true的时候才生效。假设按钮A的触摸优先级高,当处理按钮A的触摸消息时,区域判断返回true,触摸会被吞噬,按钮B不会被触发,但是处理按钮B的触摸消息时,按钮A的区域判断失败,setSwallowTouches不会执行,接着执行按钮B的回调函数。因此,两个按钮同时响应了。再提一点是,android上面是这样的,但是ios上,是不会同时响应两个按钮,这个可能跟操作系统有关吧。。
menu的实现方式
引擎中的Menu类已经对这种情况做了处理,可以参考一下如何下代码,具体思路是在触摸回调中加入一个状态的判定。开始处理一个触摸消息后,在执行onTouchBegan后,一直到onTouchEnd结束调用之前,禁用其他按钮的触摸。
bool Menu::onTouchBegan(Touch* touch,Event* event) { if (_state != Menu::State::WAITING || ! _visible || !_enabled) { return false; } for (Node *c = this->_parent; c != nullptr; c = c->getParent()) { if (c->isVisible() == false) { return false; } } _selectedItem = this->getItemForTouch(touch); if (_selectedItem) { _state = Menu::State::TRACKING_TOUCH; _selectedItem->selected(); return true; } return false; } void Menu::onTouchEnded(Touch* touch,Event* event) { CCASSERT(_state == Menu::State::TRACKING_TOUCH,"[Menu ccTouchEnded] -- invalid state"); this->retain(); if (_selectedItem) { _selectedItem->unselected(); _selectedItem->activate(); } _state = Menu::State::WAITING; this->release(); }_state 变量就是为了防止按钮同时响应的状态变量,如果当前有按钮不是处在Menu::State::WAITING状态时,就说明有按钮正在响应,不处理当前的触摸消息。但是,这同样会引入一个问题,当一个scene的不同layer中有多个menu呢,那也会导致处于不同的menu的按钮同时响应。规避办法就要看开发者如何设计了,又或者一个scene共同一个menu。。
自定义按钮类的实现方式
项目中也有很多自定义按钮类的时候,一般会使用起来会比menu类更方便,更直接。自己封装的按钮类一般都会导致上述问题,尤其是当封装的按钮的种类过多时,要支持.9的,又要支持两张图片合成的,等等。所以这里提供一个解决思路,也是参照menu的实现方式。
class ButtonManager; static ButtonManager* m_instance = nullptr; class ButtonManager { public: ButtonManager(); ~ButtonManager(); static ButtonManager* getInstance() { if (!m_instance) { m_instance = new ButtonManager(); } return m_instance; } static void release() { if (m_instance) { delete m_instance; m_instance = nullptr; } } private: CC_SYNTHESIZE(bool,m_bIsClickEnable,IsClickEnable); private: std::vector<WWButton*> m_allButtons; };很简单的buttonManager类,都不需要cpp文件,根据m_bIsClickEnable变量来判断当前按钮状态即可。
使用方法
bool WWButton::onTouchBegan(cocos2d::Touch *touch,cocos2d::Event *unused_event) { if (!m_bEnable) { return false; } if (!ButtonManager::getInstance()->getIsClickEnable()) { return false; } if (!containTouch(touch)) { return false; } ButtonManager::getInstance()->setIsClickEnable(false); return true; } void WWButton::onTouchEnded(cocos2d::Touch *touch,cocos2d::Event *unused_event) { ButtonManager::getInstance()->setIsClickEnable(true); }WWButton是一个按钮基类,只要继承此类,扩展的子类都不用担心按钮同时响应的问题。