Cocos2d-x 3.0final 终结者系列教程11-触摸机制

2023-11-08 04:32

本文主要是介绍Cocos2d-x 3.0final 终结者系列教程11-触摸机制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

刚刚打完篮球,快中暑了,大脑不转,转载一部分来源:百度

在cocos2dx 3.0版本中,废弃了以往2.x版本的写法,我们先来看一下Layer.h中的一段代码

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片

  1.     //单点触摸  
  2.     virtual bool onTouchBegan(Touch *touch, Event *unused_event);   
  3.     virtual void onTouchMoved(Touch *touch, Event *unused_event);   
  4.     virtual void onTouchEnded(Touch *touch, Event *unused_event);   
  5.     virtual void onTouchCancelled(Touch *touch, Event *unused_event);  
  6.     //多点触摸  
  7.     virtual void onTouchesBegan(const std::vector<Touch*>& touches, Event *unused_event);  
  8.     virtual void onTouchesMoved(const std::vector<Touch*>& touches, Event *unused_event);  
  9.     virtual void onTouchesEnded(const std::vector<Touch*>& touches, Event *unused_event);  
  10.     virtual void onTouchesCancelled(const std::vector<Touch*>&touches, Event *unused_event);  

单点触摸:(即只有注册的Layer才能接收触摸事件)

 onTouchBegan:     如果返回true:本层的后续Touch事件可以被触发,并阻挡向后层传递

                                 如果返回false,本层的后续Touch事件不能被触发,并向后传递,也就是不会调用onTouchMoved

简单点来说,如果

1.Layer 只有一层的情况:

virtual bool onTouchBegan(CCTouch *pTouch, CCEvent *pEvent);
a.返回false,则ccTouchMoved(),ccTouchEnded()不会再接收到消息
b.返回true,则ccTouchMoved(),ccTouchEnded()可以接收到消息
2.Layer 有多层的情况:
virtual bool onTouchBegan(CCTouch *pTouch, CCEvent *pEvent);
a.返回false,则本层的onTouchMoved(),onTouchEnded()不会再接收到消息,但是本层之下的其它层会接收到消息
b.返回true,则本层的onTouchMoved(),onTouchEnded()可以接收到消息,但是本层之下的其它层不能再接收到消息

单点触摸简单用法:

在Layer中添加如下代码,重写onTouchxxx函数

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1.        auto dispatcher = Director::getInstance()->getEventDispatcher();  
  2. auto listener = EventListenerTouchOneByOne::create();  
  3. listener->onTouchBegan = CC_CALLBACK_2(GameLayer::onTouchBegan,this);  
  4. listener->onTouchMoved = CC_CALLBACK_2(GameLayer::onTouchMoved,this);  
  5. listener->onTouchEnded = CC_CALLBACK_2(GameLayer::onTouchEnded,this);  
  6. listener->setSwallowTouches(true);//不向下传递触摸  
  7. dispatcher->addEventListenerWithSceneGraphPriority(listener,this);  

 listener->setSwallowTouches(true),不向下触摸,简单点来说,比如有两个sprite ,A 和 B,A在上B在下(位置重叠),触摸A的时候,B不会受到影响

  listener->setSwallowTouches(false)反之,向下传递触摸,触摸A也等于触摸了B

多点触摸点单用法(多个Layer获取屏幕事件):

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1.        auto dispatcher = Director::getInstance()->getEventDispatcher();  
  2. auto listener1 = EventListenerTouchAllAtOnce::create();  
  3. listener1->onTouchesBegan = CC_CALLBACK_2(GameLayer::onTouchesBegan,this);  
  4. listener1->onTouchesMoved = CC_CALLBACK_2(GameLayer::onTouchesMoved,this);  
  5. listener1->onTouchesEnded = CC_CALLBACK_2(GameLayer::onTouchesEnded,this);  
  6. dispatcher->addEventListenerWithSceneGraphPriority(listener1,this);  
或者setTouchEnabled(true),然后重写layer的onTouchsxxx函数

关于eventDispatcher:

  • 获取方法:

  • [cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
    1. auto dispatcher = Director::getInstance()->getEventDispatcher();  

事件监听器包含以下几种:

  • 触摸事件 (EventListenerTouch)
  • 键盘响应事件 (EventListenerKeyboard)
  • 加速记录事件 (EventListenerAcceleration)
  • 鼠标响应事件 (EventListenerMouse)
  • 自定义事件 (EventListenerCustom)

    以上事件监听器统一由 _eventDispatcher 来进行管理。

优先权:
1.优先级越低,越先响应事件
2.如果优先级相同,则上层的(z轴)先接收触摸事件

有两种方式将 事件监听器 listener1 添加到 事件调度器_eventDispatcher 中:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void EventDispatcher::addEventListenerWithSceneGraphPriority(EventListener* listener, Node* node)  
  2.     void EventDispatcher::addEventListenerWithFixedPriority(EventListener* listener, int fixedPriority)  

代码展开一下:
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void EventDispatcher::addEventListenerWithSceneGraphPriority(EventListener* listener, Node* node)  
  2. {  
  3.     CCASSERT(listener && node, "Invalid parameters.");  
  4.     CCASSERT(!listener->isRegistered(), "The listener has been registered.");  
  5.       
  6.     if (!listener->checkAvailable())  
  7.         return;  
  8.       
  9.     listener->setSceneGraphPriority(node);  
  10.     listener->setFixedPriority(0);  
  11.     listener->setRegistered(true);  
  12.       
  13.     addEventListener(listener);  
  14. }  

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void EventDispatcher::addEventListenerWithFixedPriority(EventListener* listener, int fixedPriority)  
  2. {  
  3.     CCASSERT(listener, "Invalid parameters.");  
  4.     CCASSERT(!listener->isRegistered(), "The listener has been registered.");  
  5.     CCASSERT(fixedPriority != 0, "0 priority is forbidden for fixed priority since it's used for scene graph based priority.");  
  6.       
  7.     if (!listener->checkAvailable())  
  8.         return;  
  9.       
  10.     listener->setSceneGraphPriority(nullptr);  
  11.     listener->setFixedPriority(fixedPriority);  
  12.     listener->setRegistered(true);  
  13.     listener->setPaused(false);  
  14.   
  15.     addEventListener(listener);  
  16. }  


(1)addEventListenerWithSceneGraphPriority 的事件监听器优先级是0,而且在 addEventListenerWithFixedPriority 中的事件监听器的优先级不可以设置为 0,因为这个是保留给 SceneGraphPriority 使用的。

(2)另外,有一点非常重要,FixedPriority listener添加完之后需要手动remove,而SceneGraphPriority listener是跟node绑定的,在node的析构函数中会被移除。

移除方法:
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. dispatcher->removeEventListener(listener);  

其实还可以这么用:

    //1加入用户触摸事件侦听

    auto listener=EventListenerTouchOneByOne::create();

    listener->setSwallowTouches(true);

    listener->onTouchBegan=[&](Touch * t,Event * e){

        //改变贪食蛇移动的方向

        int col=t->getLocation().x/32;

        int row=t->getLocation().y/32;

        int spHeadCol=spHead->getPositionX()/32;

        int spHeadRow=spHead->getPositionY()/32;

        if(abs(spHeadCol-col)>abs(spHeadRow-row))

        {

           if(spHeadCol<col)

            {

            spHead->m_dir=ENUM_DIR::DIR_RIGHT;

            }else

            {

                spHead->m_dir=ENUM_DIR::DIR_LEFT;

            }

        }

        else

            {if(spHeadRow<row)

                {

                spHead->m_dir=ENUM_DIR::DIR_UP;

                }else

                    {

                    spHead->m_dir=ENUM_DIR::DIR_DOWN;

                    }     

        }

        return true;

    }; //这条语句像不像Java中的匿名接口对象,像不像Objective-C中Block ,他有个很华丽的名字 lambda,读音为:腊母达,相信未来你会喜欢上这妹子的

//3注册这个侦听到消息分发器中   

_eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this);

   


这篇关于Cocos2d-x 3.0final 终结者系列教程11-触摸机制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/367882

相关文章

springboot使用Scheduling实现动态增删启停定时任务教程

《springboot使用Scheduling实现动态增删启停定时任务教程》:本文主要介绍springboot使用Scheduling实现动态增删启停定时任务教程,具有很好的参考价值,希望对大家有... 目录1、配置定时任务需要的线程池2、创建ScheduledFuture的包装类3、注册定时任务,增加、删

如何为Yarn配置国内源的详细教程

《如何为Yarn配置国内源的详细教程》在使用Yarn进行项目开发时,由于网络原因,直接使用官方源可能会导致下载速度慢或连接失败,配置国内源可以显著提高包的下载速度和稳定性,本文将详细介绍如何为Yarn... 目录一、查询当前使用的镜像源二、设置国内源1. 设置为淘宝镜像源2. 设置为其他国内源三、还原为官方

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Maven的使用和配置国内源的保姆级教程

《Maven的使用和配置国内源的保姆级教程》Maven是⼀个项目管理工具,基于POM(ProjectObjectModel,项目对象模型)的概念,Maven可以通过一小段描述信息来管理项目的构建,报告... 目录1. 什么是Maven?2.创建⼀个Maven项目3.Maven 核心功能4.使用Maven H

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注

SpringRetry重试机制之@Retryable注解与重试策略详解

《SpringRetry重试机制之@Retryable注解与重试策略详解》本文将详细介绍SpringRetry的重试机制,特别是@Retryable注解的使用及各种重试策略的配置,帮助开发者构建更加健... 目录引言一、SpringRetry基础知识二、启用SpringRetry三、@Retryable注解

Python虚拟环境终极(含PyCharm的使用教程)

《Python虚拟环境终极(含PyCharm的使用教程)》:本文主要介绍Python虚拟环境终极(含PyCharm的使用教程),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录一、为什么需要虚拟环境?二、虚拟环境创建方式对比三、命令行创建虚拟环境(venv)3.1 基础命令3

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

SpringKafka错误处理(重试机制与死信队列)

《SpringKafka错误处理(重试机制与死信队列)》SpringKafka提供了全面的错误处理机制,通过灵活的重试策略和死信队列处理,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、Spring Kafka错误处理基础二、配置重试机制三、死信队列实现四、特定异常的处理策略五

python连接本地SQL server详细图文教程

《python连接本地SQLserver详细图文教程》在数据分析领域,经常需要从数据库中获取数据进行分析和处理,下面:本文主要介绍python连接本地SQLserver的相关资料,文中通过代码... 目录一.设置本地账号1.新建用户2.开启双重验证3,开启TCP/IP本地服务二js.python连接实例1.