本文介绍Cocos2dx事件(以下简称Event)处理机制中的事件分发模块,在Event发生后,进过一系列处理,最后将会分发Event;

1、dispatchEvent& dispatchTouchEvent方法

voidEventDispatcher::dispatchEvent(Event* event)
{
if (!_isEnabled) return;
updateDirtyFlagForSceneGraph();
DispatchGuard guard(_inDispatch);
if (event->getType() ==Event::Type::TOUCH) {
dispatchTouchEvent(static_cast<EventTouch*>(event));
return;
}
auto listenerID = __getListenerID(event);
sortEventListeners(listenerID);
auto iter = _listenerMap.find(listenerID);
if (iter != _listenerMap.end()) {
auto listeners = iter->second;
auto onEvent =[&event](EventListener* listener) -> bool{
event->setCurrentTarget(listener->getAssociatedNode());
listener->_onEvent(event);
return event->isStopped();
};
dispatchEventToListeners(listeners,onEvent);
}
updateListeners(event);
}

在dispatchEvent方法中:

(1)  推断分发Event机制是否使能

(2)  更新脏数据标志

(3)  分发触摸Event

(4)  分发其它类型Event

voidEventDispatcher::dispatchTouchEvent(EventTouch* event)
{
sortEventListeners(EventListenerTouchOneByOne::LISTENER_ID);
sortEventListeners(EventListenerTouchAllAtOnce::LISTENER_ID);
auto oneByOneListeners =getListeners(EventListenerTouchOneByOne::LISTENER_ID);
auto allAtOnceListeners =getListeners(EventListenerTouchAllAtOnce::LISTENER_ID);
if (nullptr == oneByOneListeners &&nullptr == allAtOnceListeners)
return;
bool isNeedsMutableSet = (oneByOneListeners&& allAtOnceListeners);
const std::vector<Touch*>&originalTouches = event->getTouches();
std::vector<Touch*>mutableTouches(originalTouches.size());
std::copy(originalTouches.begin(),originalTouches.end(), mutableTouches.begin());
if (oneByOneListeners)
{
auto mutableTouchesIter =mutableTouches.begin();
auto touchesIter =originalTouches.begin();
for (; touchesIter !=originalTouches.end(); ++touchesIter) {
bool isSwallowed = false;
auto onTouchEvent =[&](EventListener* l) -> bool { ....
};
dispatchEventToListeners(oneByOneListeners, onTouchEvent);
if (event->isStopped()){
return;
}
if (!isSwallowed)
++mutableTouchesIter;
}
}
if (allAtOnceListeners &&mutableTouches.size() > 0) {
auto onTouchesEvent =[&](EventListener* l) -> bool{
....
};
dispatchEventToListeners(allAtOnceListeners, onTouchesEvent);
if (event->isStopped()){
return;
}
}
updateListeners(event);
}

在dispatchTouchEvent方法中:

(1) 对单指点击&多指点击EventListener列表进行排序;当然排序算法中首先推断眼下EventListener列表是否为脏数据,假设是脏数据,则进行排序;排序的具体细节以下会具体讲述

(2) 获取单指点击&多指点击EventListener列表,并推断EventListener是否为空

(3) 获取Event信息

(4) 若单指点击EventListener列表不为空,则分发单指点击Event

(5) 若多指点击EventListener列表不为空,则分发多指点击Event

(6) 更新EventListener列表状态

2、dispatchTouchEvent 方法中EvnetListener排序sortEventListeners

在dispatchTouchEvent方法中使用了sortEventListeners方法对EventListener列表进行排序,以下将具体解说该方法;

voidEventDispatcher::sortEventListeners(const EventListener::ListenerID&listenerID) {
DirtyFlag dirtyFlag = DirtyFlag::NONE;
auto dirtyIter =_priorityDirtyFlagMap.find(listenerID);
if (dirtyIter !=_priorityDirtyFlagMap.end()){
dirtyFlag = dirtyIter->second;
}
if (dirtyFlag != DirtyFlag::NONE) {
dirtyIter->second = DirtyFlag::NONE;
if ((int)dirtyFlag &(int)DirtyFlag::FIXED_PRIORITY) {
sortEventListenersOfFixedPriority(listenerID);
}
if ((int)dirtyFlag &(int)DirtyFlag::SCENE_GRAPH_PRIORITY) {
auto rootNode =Director::getInstance()->getRunningScene();
if (rootNode) {
sortEventListenersOfSceneGraphPriority(listenerID, rootNode);
}else{
dirtyIter->second =DirtyFlag::SCENE_GRAPH_PRIORITY;
}
}
}
}

在sortEventListeners方法中:

(1)  在脏数据列表中查找该listenerID的EventListener是否存在脏数据,若不存在脏数据则不须要排序,退出该方法;若存在脏数据,则进行排序

(2)  针对优先级不等于0的EventListener列表进行排序

(3)  针对优先级等于0的EventListener列表进行排序

以下为优先级不等于0的EventListener列表排序方法:

voidEventDispatcher::sortEventListenersOfFixedPriority(constEventListener::ListenerID& listenerID) {
auto listeners = getListeners(listenerID);
if (listeners == nullptr) return;
auto fixedListeners =listeners->getFixedPriorityListeners();
if (fixedListeners == nullptr) return;
std::sort(fixedListeners->begin(),fixedListeners->end(), [](const EventListener* l1, const EventListener* l2){
return l1->getFixedPriority() <l2->getFixedPriority();
});
intindex = 0;
for (auto& listener : *fixedListeners){
if (listener->getFixedPriority()>= 0)
break;
++index;
}
listeners->setGt0Index(index);
}

在sortEventListenersOfFixedPriority方法中:

(1) 依据ID获取EventListener列表,并推断列表是否为空

(2) 获取EventListener列表中优先级不等于0的EventListener列表_fixedListeners

(3) 使用STL中sort方法对_fixedListeners方法从小到大排序

(4) 统计fixedListeners类表中优先级数值小于0的EventListener的个数

从排序的方法能够得知,高优先级(数值越小优先级越高)EventListener先运行;若优先级同样,先注冊的EventListener先运行

以下为优先级等于0的EventListener列表排序方法:

voidEventDispatcher::sortEventListenersOfSceneGraphPriority(constEventListener::ListenerID& listenerID, Node* rootNode) {
auto listeners = getListeners(listenerID);
if (listeners == nullptr) return;
auto sceneGraphListeners =listeners->getSceneGraphPriorityListeners();
if (sceneGraphListeners == nullptr) return;
_nodePriorityIndex = 0;
_nodePriorityMap.clear();
visitTarget(rootNode, true);
std::sort(sceneGraphListeners->begin(),sceneGraphListeners->end(), [this](const EventListener* l1, constEventListener* l2) {
return_nodePriorityMap[l1->getAssociatedNode()] > _nodePriorityMap[l2->getAssociatedNode()];
});
}

在sortEventListenersOfSceneGraphPriority方法中:

(1) 依据ID获取EventListener列表,并推断列表是否为空

(2) 获取EventListener列表中优先级等于0的EventListener列表_sceneGraphListeners

(3) 使用_globalZOrder值对该Scene下的Node排序

(4) 依据EventListener相应Node的_globalZOrder值从大到小将_sceneGraphListeners列表排序

3、dispatchTouchEvent 方法中dispatchEventToListeners方法

voidEventDispatcher::dispatchEventToListeners(EventListenerVector* listeners, conststd::function<bool(EventListener*)>& onEvent) {
bool shouldStopPropagation = false;
auto fixedPriorityListeners =listeners->getFixedPriorityListeners();
auto sceneGraphPriorityListeners =listeners->getSceneGraphPriorityListeners();
ssize_t i = 0;
if (fixedPriorityListeners) {
if(!fixedPriorityListeners->empty()){
for (; i <listeners->getGt0Index(); ++i) {
auto l =fixedPriorityListeners->at(i);
if (l->isEnabled()&& !l->isPaused() && l->isRegistered() &&onEvent(l)) {
shouldStopPropagation =true;
break;
}
}
}
}
if (sceneGraphPriorityListeners) {
if (!shouldStopPropagation) {
for (auto& l :*sceneGraphPriorityListeners) {
if (l->isEnabled()&& !l->isPaused() && l->isRegistered() &&onEvent(l)) {
shouldStopPropagation =true;
break;
}
}
}
}
if (fixedPriorityListeners) {
if (!shouldStopPropagation) {
ssize_t size =fixedPriorityListeners->size();
for (; i < size; ++i) {
auto l =fixedPriorityListeners->at(i);
if (l->isEnabled()&& !l->isPaused() && l->isRegistered() &&onEvent(l)) {
shouldStopPropagation = true;
break;
}
}
}
}
}

在dispatchEventToListeners函数中:

(1) 获取_fixedListeners&_sceneGraphListeners列表

(2) 当_fixedListeners不为空时;运行_fixedListeners类表中EventToListener处理

(3) 当_sceneGraphListeners不为空时;运行_sceneGraphListeners类表中EventToListener处理

在_fixedListeners类表中EventToListener处理中,优先级小于0的是不运行的处理方法的;EventToListener的处理方法是通过參数传递过来的匿名函数;该匿名函数的实现以下会继续讲述

4、dispatchTouchEvent 方法中单点匿名方法onTouchEvent

auto onTouchEvent =[&](EventListener* l) -> bool {
EventListenerTouchOneByOne* listener =static_cast<EventListenerTouchOneByOne*>(l);
if (!listener->_isRegistered) returnfalse;
event->setCurrentTarget(listener->_node);
bool isClaimed = false;
std::vector<Touch*>::iteratorremovedIter;
EventTouch::EventCode eventCode =event->getEventCode();
if (eventCode ==EventTouch::EventCode::BEGAN) {
if (listener->onTouchBegan) {
isClaimed =listener->onTouchBegan(*touchesIter, event);
if (isClaimed &&listener->_isRegistered) {
listener->_claimedTouches.push_back(*touchesIter);
}
}
}
else if (listener->_claimedTouches.size()> 0
&& ((removedIter =std::find(listener->_claimedTouches.begin(),listener->_claimedTouches.end(), *touchesIter)) != listener->_claimedTouches.end())){
isClaimed = true;
switch (eventCode) {
case EventTouch::EventCode::MOVED:
if (listener->onTouchMoved) {
listener->onTouchMoved(*touchesIter,event);
}
break;
case EventTouch::EventCode::ENDED:
if (listener->onTouchEnded) {
listener->onTouchEnded(*touchesIter,event);
}
if (listener->_isRegistered) {
listener->_claimedTouches.erase(removedIter);
}
break;
caseEventTouch::EventCode::CANCELLED:
if (listener->onTouchCancelled){
listener->onTouchCancelled(*touchesIter,event);
}
if (listener->_isRegistered) {
listener->_claimedTouches.erase(removedIter);
}
break;
default:
CCASSERT(false, "Theeventcode is invalid.");
break;
}
}
if (event->isStopped()){
updateListeners(event);
return true;
}
if (isClaimed &&listener->_isRegistered && listener->_needSwallow) {
if (isNeedsMutableSet) {
mutableTouchesIter =mutableTouches.erase(mutableTouchesIter);
isSwallowed = true;
}
return true;
}
return false;
};

在匿名方法onTouchEvent中:

(1) 将传递參数强制转换成EventListenerTouchOneByOne类型,并推断是否为空

(2) 获取触摸(Win32下为鼠标点击\拖动)Event类型

(3) 推断Event类型是BEGAN时

a)  调用EventListener在注冊时指定的onTouchBegan,并获取返回值

b)  若返回值是true,将该Event的Touch信息放入_claimedTouches中

(4) 推断Event类型不是BEGAN时

a)  _claimedTouches的内容不为空,在_claimedTouches中有该Event的Touch信息

b)  若Event类型是MOVED,调用EventListener在注冊时指定的onTouchMoved

c)  若Event类型是ENDED,调用EventListener在注冊时指定的onTouchEnded

d)  若Event类型是CANCELLED,调用EventListener在注冊时指定的onTouchCancelled

e)  将该Event的Touch信息从_claimedTouches中移除

(5) 若该Event被停止,更新EventListener列表

(6) 若在onTouchBegan返回值为true,而且_needSwallow被设置为true时,将当前Event从多点触摸Event列表中移除

在该匿名方法中,onTouchBegan的返回值非常重要,他关注着兴许其它触摸操作(onTouchMoved\onTouchEnded\onTouchCancelled)是否运行,关注则_needSwallow标志是否生效;

5、dispatchTouchEvent 方法中多点的匿名方法onTouchEvent

auto onTouchesEvent= [&](EventListener* l) -> bool{
EventListenerTouchAllAtOnce* listener =static_cast<EventListenerTouchAllAtOnce*>(l);
if (!listener->_isRegistered) returnfalse;
event->setCurrentTarget(listener->_node);
switch (event->getEventCode())
{
case EventTouch::EventCode::BEGAN:
if (listener->onTouchesBegan) {
listener->onTouchesBegan(mutableTouches,event);
}
break;
case EventTouch::EventCode::MOVED:
if (listener->onTouchesMoved) {
listener->onTouchesMoved(mutableTouches,event);
}
break;
case EventTouch::EventCode::ENDED:
if (listener->onTouchesEnded) {
listener->onTouchesEnded(mutableTouches,event);
}
break;
case EventTouch::EventCode::CANCELLED:
if (listener->onTouchesCancelled){
listener->onTouchesCancelled(mutableTouches,event);
}
break;
default:
CCASSERT(false, "The eventcodeis invalid.");
break;
}
if (event->isStopped()){
updateListeners(event);
return true;
} return false;
};

在匿名方法onTouchEvent中:

(1)  将传递參数强制转换成EventListenerTouchAllAtOnce类型,并推断是否为空

(2)  获取触摸(Win32下为鼠标点击\拖动)Event类型

(3)  若Event类型为BEGAN时,调用EventListener在注冊时指定的onTouchBegan方法

(4)  若Event类型为MOVED时,调用EventListener在注冊时指定的onTouchesMoved方法

(5)  若Event类型为ENDED时,调用EventListener在注冊时指定的onTouchesEnded方法

(6)  若Event类型为CANCELLED时,调用EventListener在注冊时指定的onTouchesCancelled方法

(7)  若该Event被停止,更新EventListener列表

Cocos2dx引擎10-事件派发的更多相关文章

  1. cocos2d-x 3.10 PageView BUG

    cocos2d-x 3.10 PageView 拖动滚动到下一个单元,没事件,3.11有修复.

  2. cocos2dx+lua注册事件函数详解

    coocs2dx 版本 3.1.1 registerScriptTouchHandler 注册触屏事件 registerScriptTapHandler 注册点击事件 registerScriptHa ...

  3. cocos2dx+lua注册事件函数详解 事件

    coocs2dx 版本 3.1.1 registerScriptTouchHandler             注册触屏事件 registerScriptTapHandler             ...

  4. 48、[源码]-Spring容器创建-初始化事件派发器、监听器等

    48.[源码]-Spring容器创建-初始化事件派发器.监听器等 8.initApplicationEventMulticaster();初始化事件派发器: 获取BeanFactory 从BeanFa ...

  5. Java多线程开发系列之番外篇:事件派发线程---EventDispatchThread

    事件派发线程是java Swing开发中重要的知识点,在安卓app开发中,也是非常重要的一点.今天我们在多线程开发中,穿插进来这个线程.分别从线程的来由.原理和使用方法三个方面来学习事件派发线程. 一 ...

  6. wex5 实战 框架拓展之2 事件派发与data刷新

    一 前言 讲完公共data,相信大家对框架级的data组件级绑定有了更新的认识,接下来我们继续深入,以求研究明白wex5的框架能力. 在一个web项目中,其实有一个data, 是基础框架必须的data ...

  7. 使用lua实现一个简单的事件派发器

    设计一个简单的事件派发器,个人觉得最重要的一点就是如何保证事件派发过程中,添加或删除同类事件,不影响事件迭代顺序和结果,只要解决这一点,其它都好办. 为了使用pairs遍历函数,重写了pairs(lu ...

  8. cocos2d-x lua 触摸事件

    cocos2d-x lua 触摸事件 version: cocos2d-x 3.6 1.监听 function GameLayer:onEnter() local eventDispatcher = ...

  9. [置顶] Android源码分析-点击事件派发机制

    转载请注明出处:http://blog.csdn.net/singwhatiwanna/article/details/17339857 概述 一直想写篇关于Android事件派发机制的文章,却一直没 ...

随机推荐

  1. django-model-utils

    一个普通例子: todos = Todo.objects.filter(owner=request.user).filter(is_done=False).filter(priority=1) 弊端: ...

  2. 显示器中关于HS,VS,HBP,VBP参数浅析

    先来解释几个缩写的含义 HSYNC : 水平同步信号(horizontal synchronization signal) VSYNC :  垂直同步(Vertical Sync):场同步 FR : ...

  3. MVC-01 概述

    一.何谓MVC 1.MVC是开发时所使用的一种架构(框架). 2.目的在于简化软件开发的复杂度,以一种概念简单却又权责分明的架构,贯穿整个软件开发流程,通过“商业逻辑层”与“数据表现层”的切割,让这两 ...

  4. 基于Sql Server 2008的分布式数据库的实践(五)

    原文 基于Sql Server 2008的分布式数据库的实践(五) 程序设计 ------------------------------------------------------------- ...

  5. block 解析 - block变量

    block变量 上一篇 讲的是block静态变量的特性,这里我们来看一下_block变量.引用官方: You can specify that an imported variable be muta ...

  6. 2^x mod n = 1 【杭电-HDOJ-1395】 附题

    /* 2^x mod n = 1 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

  7. vector的成员函数解析

    vector是线性容器,它的元素严格的依照线性序列排序,和动态数组非常相似,和数组一样,它的元素存储在一块连续的存储空间中,这也意味着我们不仅能够使用迭代器(iterator)訪问元素,还能够使用指针 ...

  8. 80端口的烦恼:[3]清除NT Kernel占用80端口

    链接地址:http://jingyan.baidu.com/article/f96699bbca15a1894e3c1bc4.html 当一台电脑安装了vs又安装了xampp时,可以能发生80端口号冲 ...

  9. 【转】[Mysql] Linux Mysql 日志专题

    原文链接:http://blog.csdn.net/xiaoxu0123/article/details/6258538 1, 设置存放的目录: [root@Linux etc]# more /etc ...

  10. WCF技术剖析之三十:一个很有用的WCF调用编程技巧[上篇]

    原文:WCF技术剖析之三十:一个很有用的WCF调用编程技巧[上篇] 在进行基于会话信道的WCF服务调用中,由于受到并发信道数量的限制,我们需要及时的关闭信道:当遇到某些异常,我们需要强行中止(Abor ...