加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

Cocos2d-x 3.0 触摸机制

发布时间:2020-12-14 19:04:48 所属栏目:百科 来源:网络整理
导读:在Cocos2dx 3.0版本中,废弃了以往2.x版本的写法,我们先来看一下Layer.h中的一段代码: 1 2 3 4 5 6 7 8 9 10 11 //单点触摸 virtual bool onTouchBegan(Touch*touch,Event*unused_event); virtual void onTouchMoved(Touch*touch,Event*unused_event); vir


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

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


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

onTouchBegan

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

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


onTouchMoved

简单点来说,如果:

1.Layer 只有一层的情况:

1
virtual bool onTouchBegan(CCTouch*pTouch,CCEvent*pEvent);

a.返回false,则ccTouchMoved(),ccTouchEnded()不会再接收到消息

b.返回true,则ccTouchMoved(),ccTouchEnded()可以接收到消息


2.Layer 有多层的情况:

1
virtual bool onTouchBegan(CCTouch*pTouch,CCEvent*pEvent);

a.返回false,则本层的onTouchMoved(),onTouchEnded()不会再接收到消息,但是本层之下的其它层会接收到消息

b.返回true,则本层的onTouchMoved(),onTouchEnded()可以接收到消息,但是本层之下的其它层不能再接收到消息


单点触摸简单用法

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

1
2
3
4
5
6
7
autodispatcher=Director::getInstance()->getEventDispatcher();
autolistener=EventListenerTouchOneByOne::create();
listener->onTouchBegan=CC_CALLBACK_2(GameLayer::onTouchBegan, this );
listener->onTouchMoved=CC_CALLBACK_2(GameLayer::onTouchMoved, this );
listener->onTouchEnded=CC_CALLBACK_2(GameLayer::onTouchEnded, this );
listener->setSwallowTouches( true ); //不向下传递触摸
dispatcher->addEventListenerWithSceneGraphPriority(listener, this );


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


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


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

1
2
3
4
5
6
autodispatcher=Director::getInstance()->getEventDispatcher();
autolistener1=EventListenerTouchAllAtOnce::create();
listener1->onTouchesBegan=CC_CALLBACK_2(GameLayer::onTouchesBegan, this );
listener1->onTouchesMoved=CC_CALLBACK_2(GameLayer::onTouchesMoved, this );
listener1->onTouchesEnded=CC_CALLBACK_2(GameLayer::onTouchesEnded, this );
dispatcher->addEventListenerWithSceneGraphPriority(listener1, this );


或者setTouchEnabled(true),然后重写layer的onTouchsxxx函数


关于eventDispatcher

1) 获取方法:

1
autodispatcher=Director::getInstance()->getEventDispatcher();


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

触摸事件 (EventListenerTouch)

键盘响应事件 (EventListenerKeyboard)

加速记录事件 (EventListenerAcceleration)

鼠标响应事件 (EventListenerMouse)

自定义事件 (EventListenerCustom)


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


2)优先权

1.优先级越低,越先响应事件

2.如果优先级相同,则上层的(z轴)先接收触摸事件


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

1
2
void EventDispatcher::addEventListenerWithSceneGraphPriority(EventListener*listener,Node*node)
void EventDispatcher::addEventListenerWithFixedPriority(EventListener*listener, int fixedPriority)


代码展开一下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
void EventDispatcher::addEventListenerWithSceneGraphPriority(EventListener*listener,Node*node)
{
CCASSERT(listener&&node, "Invalidparameters." );
CCASSERT(!listener->isRegistered(), "Thelistenerhasbeenregistered." );
if (!listener->checkAvailable())
return ;
listener->setSceneGraphPriority(node);
listener->setFixedPriority(0);
listener->setRegistered( true );
addEventListener(listener);
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
void EventDispatcher::addEventListenerWithFixedPriority(EventListener*listener, int fixedPriority)
{
CCASSERT(listener, "Thelistenerhasbeenregistered." );
CCASSERT(fixedPriority!=0, "0priorityisforbiddenforfixedprioritysinceit'susedforscenegraphbasedpriority." );
if (!listener->checkAvailable())
return ;
listener->setSceneGraphPriority(nullptr);
listener->setFixedPriority(fixedPriority);
listener->setRegistered( true );
listener->setPaused( false );
addEventListener(listener);
}


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


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


移除方法:

1
dispatcher->removeEventListener(listener);


其实还可以这么用:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
//1加入用户触摸事件侦听
autolistener=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 );

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读