事件处理步骤:

1.创建一个触摸事件监听器(单点触摸或多点触摸)

2.实现触摸事件的响应方法

3.添加事件监听器(场景优先或固定值优先)

4.当用户触摸时,事件分发器就会将事件分发给监听器进行响应

  首先,用户点击屏幕,这时候cocos会把用户点击的信息,包括点击的坐标,优先顺序的值...等等,包含在一个Touch类型的变量中,可以是t。然后把这个信息传递给事件分发器,再由事件分发器分发给相应的事件监听器。事件监听器根据事件的类型,就是一个Event类型的变量,可以是e。决定要响应什么样的动作,这个响应的动作可以用Lambda表达式为架构,里面放上一个回调函数。

  一个事件监听器可以有多个事件处理响应的Lambda表达式,但是只能响应一个实例对象,这个实例对象可以是this(其实是指layer),或者sprite,或者其他的Node类的子类。而且事件分发器只允许注册一次监听器,也就是在事件分发器里面不能有重名的监听器。所以,如果要让多个对象都可以被监听到相同的监听器,可以在注册监听器的时候用listener1->clone(),sprite2。

  listener1->setSwallowTouches(true)是设置是否吞并事件,我们平时点击或者触摸的时候,如果有多个对象重叠的时候,而且每个重叠的对象都有自己的监听器和响应的Lambda。如果我们把它设置为true,吞并触摸,那么事件响应的时候只响应最上层的那个对象,就是只处理最上层对象的事件传递和响应。而如果我们把它设置成false,那么事件响应的时候会按照对象zOrder的大小来依次处理各个对象的事件传递和响应。也就是不吞并触摸,让所有在这一点下的对象都被触摸到,而不是被最上层的对象吞掉。

实例代码:

bool HelloWorld::init()
{
if (!Layer::init())
{
return false;
}
auto visibleSize = Director::getInstance()->getVisibleSize();
Vec2 origin = Director::getInstance()->getVisibleOrigin();

//删除触摸监听器
auto removeAllTouchItem = MenuItemFont::create("shanchu suoyou de chumo jiantingqi", [this](Ref * sender){
  auto senderItem = static_cast<MenuItemFont*>(sender);
  senderItem->setString("chumo jiantingqi shanchu chenggong");
  _eventDispatcher->removeEventListenersForType(EventListener::Type::TOUCH_ONE_BY_ONE);
});
removeAllTouchItem->setPosition(visibleSize.width / 2, visibleSize.height*0.2);

//标题标签
auto title = Label::createWithSystemFont("dan dian chumo ceshi", "", 64);
title->setPosition(visibleSize.width / 2, visibleSize.height*0.9);
this->addChild(title);

//提示标签
auto subtitle = Label::createWithSystemFont("qing dan ji huozhe tuo dong huakuai", "", 32);
subtitle->setPosition(visibleSize.width / 2, visibleSize.height*0.8);
this->addChild(subtitle);

//创建三个精灵对象
auto sprite1 = Sprite::create("CyanSquare.png");
sprite1->setPosition(Vec2(visibleSize.width / 2, visibleSize.height / 2) + Vec2(-80, 80));
addChild(sprite1, 10);

auto sprite2 = Sprite::create("MagentaSquare.png");
sprite2->setPosition(Vec2(visibleSize.width / 2, visibleSize.height / 2));
addChild(sprite2, 20);

auto sprite3 = Sprite::create("YellowSquare.png");
sprite3->setPosition(Vec2(0, 0));
sprite2->addChild(sprite3, 1);

//创建一个单点触摸事件监听器,处理触摸事件逻辑
auto listener1 = EventListenerTouchOneByOne::create();
//设置是否向下传递触摸
listener1->setSwallowTouches(true);
//通过Lambda表达式直接实现触摸事件的响应方法
listener1->onTouchBegan = [](Touch * touch, Event * event){
auto target = static_cast<Sprite*>(event->getCurrentTarget());
Point locationInNode = target->convertToNodeSpace(touch->getLocation());

Size s = target->getContentSize();
Rect rect = Rect(0, 0, s.width, s.height);
if (rect.containsPoint(locationInNode))
{
  log("OnTouchBegan...x=%f,y=%f", locationInNode.x, locationInNode.y);
  target->setOpacity(180);
  return true;
}
return false;
};

listener1->onTouchMoved = [](Touch * touch, Event * event){
  log("OnTouchMoved...");
  auto target = static_cast<Sprite*>(event->getCurrentTarget());
  target->setPosition(target->getPosition() + touch->getDelta());
};

listener1->onTouchEnded = [=](Touch * touch, Event * event){
  auto target = static_cast<Sprite*>(event->getCurrentTarget());
  target->setOpacity(255);
  std::string name;
  if (target == sprite2)
  {
    name = "MagentaSquare.png";
    sprite1->setZOrder(100);
    subtitle->setString("xiang ying shijian de shi jiuhongse,qingse de Zorder zhi bei xiu gai wei100");

  }
  else if (target == sprite1)
  {
    name = "CyanSquare.png";
    sprite1->setZOrder(0);
    subtitle->setString("xiang ying shijian de shi qingse ,qingse de Zorder zhi bei xiu gai wei 0");
  }
  else
  {
    name = "YellowSquare.png";
  }
  log("OnTocuhEnded...your touch is%s", name.c_str());
};

//添加场景优先事件监听器
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1, sprite1);
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1->clone(), sprite2);
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1->clone(), sprite3);

return true;
}

7月25日

事件响应是写在HelloWorld类中的.cpp文件里面的init方法里面的

1.首先判断是什么事件类型

EventListenerPhysicsContact  //物理碰撞事件

EventListenerTouchOneByOne  //单点触摸事件

EventListenerTouchAllAtOnce  //多点触摸事件

EventListenerAcceleration  //加速度计事件

EventListenerKeyboard  //按键事件

EventListenerMouse  //鼠标事件

EventListenerCustom  //自定义事件

2.创建出相应的事件监听器

第一种:auto listener = EventListenerTouchOneByOne::create();

第二种:在.h文件的属性定义里面定义EventListenerTouchOneByOne * listener;

在.cpp文件的init方法里面写listener = EventListenerTouchOneByOne::create();

3.设置事件监听器,结构是Lambda表达式,里面有一个回调函数,当被监听的对象满足事件监听器的要求时,执行回调函数里面的内容,this指的是当前层,Touch * t是用户的触摸各种属性, Event * e是事件类型。

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

  for (auto it = this->gcs.begin(); it != this->gcs.end(); it++) {
    auto gc = *it;

    if (gc->getEdge()->getBoundingBox().containsPoint(t->getLocation())) {
    gc->onTouch();
    break;
    }
  }

  return false;

};

4.把事件监听器添加到事件分发器中,当用户触摸时,事件分发器就会将相应事件分发给相应的监听器进行响应,listener就是要添加的监听器,this指的是监听对象,就是当前层。

Director::getInstance()->getEventDispatcher()->addEventListenerWithSceneGraphPriority(listener, this);

注意:

MOVED类型的鼠标点击事件,是把MOVED所对应的事件函数执行一遍,然后停留在执行完后的状态,直到ENDED类型的鼠标点击事件的函数被调用。

case Widget::TouchEventType::MOVED:
gamecontroller->onCreep();
break;

case Widget::TouchEventType::ENDED:
gamecontroller->onNormal();
break;

关于Cocos2d-x事件处理机制的更多相关文章

  1. cocos2d触摸事件处理机制(2.x和3.x变化)

    2.x的触摸事件的版本号 触摸事件处理有2种子.以下单点触摸的样本.(另一种多点触摸屏). 创建cocos2d 该项目. 1. 重写下面虚函数. bool ccTouchBegan(cocos2d:: ...

  2. Cocos2d-x之事件处理机制

    |   版权声明:本文为博主原创文章,未经博主允许不得转载. 事件处理机制分为单点触屏,多点触屏,加速度事件,键盘事件和鼠标事件.在现在的智能手机中,触屏的应用比较的广泛,尤其是多点触屏事件的技术,使 ...

  3. java 事件处理机制:按下上下左右键控制小球的运动

    /** * 加深对事件处理机制的理解 * 通过上下左右键来控制一个小球的位置 */package com.test3;import java.awt.*;import javax.swing.*;im ...

  4. Android事件处理机制

    包括监听和回调两种机制. 1. 基于监听的事件处理: 事件监听包含三类对象,事件源,事件,事件监听器.Android的事件处理机制是一种委派式(Delegation)事件处理方式:普通组件(事件源)将 ...

  5. Android的两种事件处理机制

    UI编程通常都会伴随事件处理,Android也不例外,它提供了两种方式的事件处理:基于回调的事件处理和基于监听器的事件处理. 对于基于监听器的事件处理而言,主要就是为Android界面组件绑定特定的事 ...

  6. Android的Touch事件处理机制

    Android的Touch事件处理机制比较复杂,特别是在考虑了多点触摸以及事件拦截之后. Android的Touch事件处理分3个层面:Activity层,ViewGroup层,View层. 首先说一 ...

  7. IOS事件处理机制(关于触发者和响应者的确认)

    事件处理机制 在iOS中发生触摸后,事件会加入到UIApplication事件队列(在这个系列关于iOS开发的第一篇文章中我们分析iOS程序原理的时候就说过程序运行后UIApplication会循环监 ...

  8. Java Swing事件处理机制

    Java Swing的事件处理机制 Swing GUI启动后,Java虚拟机就启动三个线程,分别为主线程,事件派发线程(也是事件处理线程)和系统工具包线程. 主线程 :负责创建并显示该程序的初始界面: ...

  9. Qt事件处理机制

    研一的时候开始使用Qt,感觉用Qt开发图形界面比MFC的一套框架来方便的多.后来由于项目的需要,也没有再接触Qt了.现在要重新拾起来,于是要从基础学起. Now,开始学习Qt事件处理机制. 先给出原文 ...

  10. core java 8~9(GUI & AWT事件处理机制)

    MODULE 8 GUIs--------------------------------GUI中的包: java.awt.*; javax.swing.*; java.awt.event.*; 要求 ...

随机推荐

  1. POSIX 共享内存和 系列函数

    在前面介绍了system v 共享内存的相关知识,现在来稍微看看posix 共享内存 和系列函数. 共享内存简单来说就是一块真正的物理内存区域,可以使用一些函数将这块区域映射到进程的地址空间进行读写, ...

  2. nginx实战三

    nginx正向代理 https://coding.net/u/aminglinux/p/nginx/git/blob/master/proxy/z_proxy.md Nginx正向代理使用场景并不多见 ...

  3. [WARNING] mod_event_socket.c:2641 IP 172.18.1.112 Rejected by acl "loopback.auto"

    一.目标修改event_socket配置,使之能够建立远端ESL链接. 二.步骤 . vim ../autoload_configs/event_socket.conf.xml . 默认的监听地址配置 ...

  4. 查看mysql日志

    1.首先确认你日志是否启用了 MySQL>show variables like 'log_bin'; 2.如果启用了,即ON那日志文件就在MySQL的安装目录的data目录下 3.怎样知道当前 ...

  5. Generalized Linear Models

    作者:桂. 时间:2017-05-22  15:28:43 链接:http://www.cnblogs.com/xingshansi/p/6890048.html 前言 主要记录python工具包:s ...

  6. 如何发布Node模块到NPM社区

    “学骑自行车最快的方式就是先骑上去” 一.安装node和npm 1.一种是通过编译node源文件安装node(注意:需要Python 2.6或2.7已经安装) $ wget http://nodejs ...

  7. 怎么从docker中copy文件到 本机

    怎么从docker中copy文件到 本机 docker cp <containerId>:/file/path/within/container /host/path/target

  8. 如何在 block 中修改外部变量

    转自:http://www.cnblogs.com/easonoutlook/archive/2012/08/22/2650070.html block 的目的是为了支持并行编程,对于普通的 loca ...

  9. 轮询、长轮询、长连接、websocket

    Web端即时通讯技术:即时通讯技术简单的说就是实现这样一种功能:服务器端可以即时地将数据的更新或变化反应到客户端,例如消息即时推送等功能都是通过这种技术实现的.但是在Web中,由于浏览器的限制,实现即 ...

  10. AngularJS实现跨域请求

    跨域,前端开发中经常遇到的问题,AngularJS实现跨域方式类似于Ajax,使用CORS机制. 下面阐述一下AngularJS中使用$http实现跨域请求数据. AngularJS XMLHttpR ...