cocos2dx 中触摸事件分发一些解读
触摸事件分发中几个代码解读:
怎么说呢,感觉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 中触摸事件分发一些解读的更多相关文章
- Cocos2d-x中触摸事件
理解一个触摸事件可以从时间和空间两方面考虑. 1.触摸事件的时间方面 触摸事件的在时间方面,如下图所示,可以有不同的“按下”.“移动”和“抬起”等阶段,表示触摸是否刚刚开始.是否正在移动或处于静止状态 ...
- 一个demo让你彻底理解Android中触摸事件的分发
注:本文涉及的demo的地址:https://github.com/absfree/TouchDispatch 1. 触摸动作及事件序列 (1)触摸事件的动作 触摸动作一共有三种:ACTION_DOW ...
- cocos2d-x lua 触摸事件
cocos2d-x lua 触摸事件 version: cocos2d-x 3.6 1.监听 function GameLayer:onEnter() local eventDispatcher = ...
- 安卓中的事件分发机制之View控件
前言:Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev).onInterceptTouchEvent(MotionEvent e ...
- Android中的事件分发机制
Android中的事件分发机制 作者:丁明祥 邮箱:2780087178@qq.com 这篇文章这周之内尽量写完 参考资料: Android事件分发机制完全解析,带你从源码的角度彻底理解(上) And ...
- 图解Android触摸事件分发
Android中触摸事件传递过程中最重要的是dispatchTouchEvent().onInterceptTouchEvent()和onTouchEvent()方法. View和Activity有d ...
- iOS中—触摸事件详解及使用
iOS中--触摸事件详解及使用 (一)初识 要想学好触摸事件,这第一部分的基础理论是必须要学会的,希望大家可以耐心看完. 1.基本概念: 触摸事件 是iOS事件中的一种事件类型,在iOS中按照事件划分 ...
- 【Unity游戏开发】用C#和Lua实现Unity中的事件分发机制EventDispatcher
一.简介 最近马三换了一家大公司工作,公司制度规范了一些,因此平时的业余时间多了不少.但是人却懒了下来,最近这一个月都没怎么研究新技术,博客写得也是拖拖拉拉,周六周天就躺尸在家看帖子.看小说,要么就是 ...
- Cocos2d-x 3.X 事件分发机制
介绍 Cocos2d-X 3.X 引入了一种新的响应用户事件的机制. 涉及三个基本的方面: Event listeners 封装你的事件处理代码 Event dispatcher 向 listener ...
随机推荐
- [js高手之路]原型对象(prototype)与原型链相关属性与方法详解
一,instanceof: instanceof检测左侧的__proto__原型链上,是否存在右侧的prototype原型. 我在之前的两篇文章 [js高手之路]构造函数的基本特性与优缺点 [js高手 ...
- 基于.NET CORE微服务框架 -谈谈surging API网关
1.前言 对于最近surging更新的API 网关大家也有所关注,也收到了不少反馈提出是否能介绍下Api网关,那么我们将在此篇文章中剥析下surging的Api 网关 开源地址:https://git ...
- 菜鸟之路Vue----一
Vue api 学习笔记之 全局配置 1.Vue全局配置 Vue.config是一个对象,它包含了Vue的全局变量配置. #silent 用来取消 Vue 所有的日志与警告,其值值类型为布尔值(Bo ...
- VisualSVN安装图解
VisualSVN安装教程... ----------------------------------- 参考网址:https://www.visualsvn.com/server/download/ ...
- 深入了解mysql数据传输编码原理
一.基本概念(这里引用http://www.laruence.com/2008/01/05/12.html) 1. 给定一系列字符,对每个字符赋予一个数值,用数值来代表对应的字符,这一数值就是字符的编 ...
- Spring(四)-- JdbcTemplate、声明式事务
1.Spring提供的一个操作数据库的技术JdbcTemplate,是对Jdbc的封装.语法风格非常接近DBUtils. JdbcTemplate可以直接操作数据库,加快效率,而且学这个JdbcT ...
- CODE大全给你推荐几个免费的leapftp 注册码
leapftp 2.7.6 注册码, Name: Kmos/CiA in 1999 s/n: MOD1-MO2D-M3OD-NOPQ LeapFTP2.7.5 注册名:swzn 注册码:214065- ...
- ssh的相关实验
author:JevonWei 版权声明:原创作品 跨主机ssh连接 主机A想连接主机C,但是主机C防火墙等原因禁止主机A连接,而主机A可以连接主机B,主机B也可连接主机C,即主机A就可通过主机B做跳 ...
- Java学习记录 : 画板的实现
接触java不满一个月,看厚厚的java入门简直要醉,故利用实例来巩固所学知识. 画板的实现其实从原理来说超级简单,可能一会儿就完成了. 但作为一名强迫症患者,要实现和win下面的画板一样的功能还是需 ...
- 详解 mpls vpn 的实现
MPLS VPN的实现 一.实验目的 该实验通过MPLS VPN的数据配置,使学生掌握路由器相关接口的IP地址设置.路由协议的配置以及MPLS VPN的完整的创建过程, 从而加深对IP网络的IP编址. ...