触摸事件分发中几个代码解读:

怎么说呢,感觉cocos2dx中的消息分发机制,相对于android中触摸事件分发机制要简单的多。因为android中要做区域判断,过滤器,以及父子组件分发给谁等等的逻辑..
cocos2dx 中相对就要简单多了。

如果有一个组件如果想要接收触摸事件,会通过一个继承一个CCTouchDelegate接口注册给CCTouchDispatcher
CCTouchDispatcher 中维护了一个CCTouchHandler的队列。CCTouchHandler 是CCTouchDelegate两个派生类的包装类。
在接到触摸事件之后,遍历 所维护的CCTouchHandler 队列,并按触摸事件类型,调用对应的方法,CCTouchDelegate 接到回调后,再来进行逻辑处理

而 CCTouchDispatcher 实现了一个 EGLTouchDelegate接口。CCDirector会把这个接口以CCEGLView::setTouchDelegate(CCTouchDispatcher)方式注册到CCEGLViewProtocol里,而这个类针对支持的平台都有适配,然后平台会把相应的事件分发下来。

先来看CCTouchDelegate注册事件解除注册两个方法(Standard方式和Target方式类似,只说一种)

注册回调接口

void CCTouchDispatcher::addStandardDelegate(CCTouchDelegate *pDelegate, int nPriority)
{
CCTouchHandler *pHandler = CCStandardTouchHandler::handlerWithDelegate(pDelegate, nPriority);
if (! m_bLocked)
{
forceAddHandler(pHandler, m_pStandardHandlers);
}
else
{
if (ccCArrayContainsValue(m_pHandlersToRemove, pDelegate))
{
ccCArrayRemoveValue(m_pHandlersToRemove, pDelegate);
return;
} m_pHandlersToAdd->addObject(pHandler);
m_bToAdd = true;
}
}

移除回调接口

void CCTouchDispatcher::removeDelegate(CCTouchDelegate *pDelegate)
{
...
if (! m_bLocked)
{
forceRemoveDelegate(pDelegate);
}
else
{
CCTouchHandler *pHandler = findHandler(m_pHandlersToAdd, pDelegate);
if (pHandler)
{
m_pHandlersToAdd->removeObject(pHandler);
return;
}
ccCArrayAppendValue(m_pHandlersToRemove, pDelegate);
m_bToRemove = true;
}
}

先说为什么要加入一个m_bLocked, m_pHandlersToRemove,m_pHandlersToAdd 而不是直接放入m_pStandardHandlers中。

触摸消息的分发是在CCTouchDispatcher::touches方法中实现的

touches方法中,消息的分发是通过 CCARRAY_FOREACH(m_pStandardHandlers,pObj)这样的方式来遍历m_pStandardHandlers的,来依次判断是否需要把触摸事件分发到对应的胡回调接口中。

假设没有做阻塞标识符m_blocked,如果在移除接口的方法调用的时候,恰好正在遍历这个队列,这个时候直接从m_pStandardHandlers移除对象,
很可能会破坏队列结构,很可能会导致脚标越界的异常(就是常说的队列安全)。

所以就需要一个缓存队列缓存add和remove操作,在遍历结束后,把缓存队列中的接口移除或者添加进m_pStandardHandlers中。

另外forceAddHandler(),forceRemoveDelegate 这些方法顾名思义,就是什么也不用管了,直接添加或移除。这个两个方法调用之前,都做了 !m_blocked 判断。

touches 方法解读

首先将阻塞标识位m_bLocked设置为true,防止外部直接改变 m_pTargetedHandlers 队列

m_bLocked = true;

之后 判断了这次分发触摸事件是否是两中处理方式都要做。

unsigned int uTargetedHandlersCount = m_pTargetedHandlers->count();
unsigned int uStandardHandlersCount = m_pStandardHandlers->count();
bool bNeedsMutableSet = (uTargetedHandlersCount && uStandardHandlersCount);
pMutableTouches = (bNeedsMutableSet ? pTouches->mutableCopy() : pTouches);//这里为什么要复制一个队列出来,会下边解释

然后进入Target处理方式的逻辑。这个顺序表明Target的处理优先级要比Standard要高

Target处理循环(代码又删减)

for (setIter = pTouches->begin(); setIter != pTouches->end(); ++setIter)
{
....
CCARRAY_FOREACH(m_pTargetedHandlers, pObj)//遍历m_pTargetedHandlers,
{
pHandler = (CCTargetedTouchHandler *)(pObj);
bool bClaimed = false;
if (uIndex == CCTOUCHBEGAN)
{
bClaimed = pHandler->getDelegate()->ccTouchBegan(pTouch, pEvent);// 标记1
if (bClaimed)
{
pHandler->getClaimedTouches()->addObject(pTouch);
}
} else
if (pHandler->getClaimedTouches()->containsObject(pTouch))//标记2
{
// moved ended canceled
bClaimed = true;
.....
pHandler->getClaimedTouches()->removeObject(pTouch);
break;
} if (bClaimed && pHandler->isSwallowsTouches())//标记3
{
if (bNeedsMutableSet)
{
pMutableTouches->removeObject(pTouch);
}
break;//标记4
}
}
}
}

对应标记解释

标记1:

如果继承的是CCTargetedTouchDelegate 接口,began回调的返回值至关重要,只有返回true的时候,会把当前CCTouch放入CCTargetedTouchDelegate对应包装类的触摸事件集合     中。
标记2

当只有在标记1中触摸事件返回值为true 的情况,才会分发后续事件。

估计有人看到这里会疑惑,当CCTouch 是began时候放入包装类中。如果触摸事件已经是变化成了MOVE 或者 END时候已经是第二次调用touchs方法了,为什么却使用pHandler->getClaimedTouches()->containsObject(pTouch)这样的方式判断的?

事实上,经过测试发现,当一个触摸事件生成之后,从began,到end和cancle,都是同一个对象。

另外,经过测试,这些CCTouch对象是通过多个固定对象来缓存的,而不是每次有触摸就创建一个新对象。多次触摸事件测试结果显示,单点触摸情况下大约有四个左右()的缓存对象


标记3

这个地方的代码非常的巧妙..

如果这个CCTouch已经有一个CCTargetedTouchDelegate的对他处理,且这个CCTargetedTouchDelegate的状态又是isSwallowsTouches==true,即注册监听的时候调用方法addTargetedDelegate(CCTouchDelegate *pDelegate, int nPriority, bool bSwallowsTouches)时,第三个参数为true,则不再把这个CCTouch分发给这个组件之后的其他组件。当然,优先级比他高的就没办法屏蔽了。

其中这个“不分发给其他组件”有两个逻辑处理
     1、标记4处的break ,就是当这bSwallowsTouches为true情况。会退出这个CCTouch处理逻辑中的CCARRAY_FOREACH(m_pTargetedHandlers, pObj) 循环,保证了这组件之后的CCTargetedTouchDelegate 都不会接受到这个CCTouch事件

2、如果是有CCStandardTouchDelegate要处理(bNeedsMutableSet==true)的情况,则把这个CCTouch从事件集合中移除,来保证Standard方式的组件不会接收到这个CCTouch事件。同样的原因,因为对CCTargetedTouchDelegate的处理是在对CCSet*pTouches遍历中做的,直接从pTouches中移除又会破坏队列结构,所以有了上边说到的pMutableTouches=pTouches->mutableCopy() 的copy操作

再来看这么写的原因

bool bNeedsMutableSet = (uTargetedHandlersCount && uStandardHandlersCount);
pMutableTouches = (bNeedsMutableSet ? pTouches->mutableCopy() : pTouches);

bNeedsMutableSet = uTargetedHandlersCount && uStandardHandlersCount

bNeedsMutableSet = true&&true  就是标记3解释的地方说到的情况是

bNeedsMutableSet = true&&false 的情况,没有Standard 类型处理方式 需要处理,也就没必要屏蔽这个事件,所以没有copy
bNeedsMutableSet = false&&true 这种情况没有Target的处理,则也不会出现Swallows 为true需要屏蔽的情况,也就不会有移除操作,不用copy
bNeedsMutableSet = false&&false 这种情况什么处理也不做,也不需要copy

这就是为什么 bNeedsMutableSet==true 的情况需要copy操作了

standard的处理循环

  CCARRAY_FOREACH(m_pStandardHandlers, pObj)
{
pHandler = (CCStandardTouchHandler*)(pObj);
switch (sHelper.m_type)
{
case CCTOUCHBEGAN:
pHandler->getDelegate()->ccTouchesBegan(pMutableTouches, pEvent);
break;
........
}
}

从代码可看出,Standard 消息处理方式更简单,什么也不管,直接遍历所有注册过的组件,然后事件集合全部发下去,让组件自己处理。

处理完Standard之后,会将阻塞标记m_blocked设置为false,然后组件开始检查add和remove队列。
并且可以发现,Target处理方式并不是不接受多点触摸,只不过是把多点触摸拆开分发下去的。而Standard是把多点触摸的消息一起发现去的。且分发过程中,是没有做区域判断的。并且,Standard处理方式ccTouchesBegan,move,end等方法的返回值是true或false都没影响

//待证明的假设
1、这边就假设一个场景,一个精灵注册的是Target的方式,并且随着触摸位置来更新精灵的位置。
如果只是拿到了CCTouch 就直接更新精灵的位置,可能会导致两个手指一起按住精灵拖动,因为CCSet *pTouches多次分发下去的,可能会导致精灵在两个手指之间跳动。但是因为这个分发过程是在一个循环中直接分发完成,两个指头触摸事件顺序是不变的,界面都没有来的及重绘,而直接到了最后那个触摸事件的位置上。

2、还是1中的设定,如果在注册监听的时候,所有精灵都没有设置m_bSwallowsTouches属性,可能会导致重叠精灵会被一起拖动--已证明

cocos2dx 中触摸事件分发一些解读的更多相关文章

  1. Cocos2d-x中触摸事件

    理解一个触摸事件可以从时间和空间两方面考虑. 1.触摸事件的时间方面 触摸事件的在时间方面,如下图所示,可以有不同的“按下”.“移动”和“抬起”等阶段,表示触摸是否刚刚开始.是否正在移动或处于静止状态 ...

  2. 一个demo让你彻底理解Android中触摸事件的分发

    注:本文涉及的demo的地址:https://github.com/absfree/TouchDispatch 1. 触摸动作及事件序列 (1)触摸事件的动作 触摸动作一共有三种:ACTION_DOW ...

  3. cocos2d-x lua 触摸事件

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

  4. 安卓中的事件分发机制之View控件

    前言:Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev).onInterceptTouchEvent(MotionEvent e ...

  5. Android中的事件分发机制

    Android中的事件分发机制 作者:丁明祥 邮箱:2780087178@qq.com 这篇文章这周之内尽量写完 参考资料: Android事件分发机制完全解析,带你从源码的角度彻底理解(上) And ...

  6. 图解Android触摸事件分发

    Android中触摸事件传递过程中最重要的是dispatchTouchEvent().onInterceptTouchEvent()和onTouchEvent()方法. View和Activity有d ...

  7. iOS中—触摸事件详解及使用

    iOS中--触摸事件详解及使用 (一)初识 要想学好触摸事件,这第一部分的基础理论是必须要学会的,希望大家可以耐心看完. 1.基本概念: 触摸事件 是iOS事件中的一种事件类型,在iOS中按照事件划分 ...

  8. 【Unity游戏开发】用C#和Lua实现Unity中的事件分发机制EventDispatcher

    一.简介 最近马三换了一家大公司工作,公司制度规范了一些,因此平时的业余时间多了不少.但是人却懒了下来,最近这一个月都没怎么研究新技术,博客写得也是拖拖拉拉,周六周天就躺尸在家看帖子.看小说,要么就是 ...

  9. Cocos2d-x 3.X 事件分发机制

    介绍 Cocos2d-X 3.X 引入了一种新的响应用户事件的机制. 涉及三个基本的方面: Event listeners 封装你的事件处理代码 Event dispatcher 向 listener ...

随机推荐

  1. [js高手之路]原型对象(prototype)与原型链相关属性与方法详解

    一,instanceof: instanceof检测左侧的__proto__原型链上,是否存在右侧的prototype原型. 我在之前的两篇文章 [js高手之路]构造函数的基本特性与优缺点 [js高手 ...

  2. 基于.NET CORE微服务框架 -谈谈surging API网关

    1.前言 对于最近surging更新的API 网关大家也有所关注,也收到了不少反馈提出是否能介绍下Api网关,那么我们将在此篇文章中剥析下surging的Api 网关 开源地址:https://git ...

  3. 菜鸟之路Vue----一

    Vue api 学习笔记之 全局配置 1.Vue全局配置 Vue.config是一个对象,它包含了Vue的全局变量配置. #silent  用来取消 Vue 所有的日志与警告,其值值类型为布尔值(Bo ...

  4. VisualSVN安装图解

    VisualSVN安装教程... ----------------------------------- 参考网址:https://www.visualsvn.com/server/download/ ...

  5. 深入了解mysql数据传输编码原理

    一.基本概念(这里引用http://www.laruence.com/2008/01/05/12.html) 1. 给定一系列字符,对每个字符赋予一个数值,用数值来代表对应的字符,这一数值就是字符的编 ...

  6. Spring(四)-- JdbcTemplate、声明式事务

    1.Spring提供的一个操作数据库的技术JdbcTemplate,是对Jdbc的封装.语法风格非常接近DBUtils.   JdbcTemplate可以直接操作数据库,加快效率,而且学这个JdbcT ...

  7. CODE大全给你推荐几个免费的leapftp 注册码

    leapftp 2.7.6 注册码, Name: Kmos/CiA in 1999 s/n: MOD1-MO2D-M3OD-NOPQ LeapFTP2.7.5 注册名:swzn 注册码:214065- ...

  8. ssh的相关实验

    author:JevonWei 版权声明:原创作品 跨主机ssh连接 主机A想连接主机C,但是主机C防火墙等原因禁止主机A连接,而主机A可以连接主机B,主机B也可连接主机C,即主机A就可通过主机B做跳 ...

  9. Java学习记录 : 画板的实现

    接触java不满一个月,看厚厚的java入门简直要醉,故利用实例来巩固所学知识. 画板的实现其实从原理来说超级简单,可能一会儿就完成了. 但作为一名强迫症患者,要实现和win下面的画板一样的功能还是需 ...

  10. 详解 mpls vpn 的实现

    MPLS VPN的实现 一.实验目的 该实验通过MPLS VPN的数据配置,使学生掌握路由器相关接口的IP地址设置.路由协议的配置以及MPLS VPN的完整的创建过程, 从而加深对IP网络的IP编址. ...