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

Cocos2d-x 3.0 触摸机制

发布时间:2020-12-14 16:35:28 所属栏目:百科 来源:网络整理
导读:在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); onT

在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);
onTouchEnded(Touch*touch,Event*unused_event);
onTouchCancelled(Touch*touch,Event*unused_event);
//多点触摸
onTouchesBegan( const std::vector<Touch*>&touches,Event*unused_event);
onTouchesMoved( onTouchesEnded( onTouchesCancelled(

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

onTouchBegan

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

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

onTouchMoved

简单点来说,如果:

1.Layer 只有一层的情况:

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

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

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

2.Layer 有多层的情况:

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

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

单点触摸简单用法

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

7
autodispatcher=Director::getInstance()->getEventDispatcher();
autolistener=EventListenerTouchOneByOne::create();
listener->onTouchBegan=CC_CALLBACK_2(GameLayer::onTouchBegan, this );
listener->onTouchMoved=CC_CALLBACK_2(GameLayer::onTouchMoved,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">);
listener->onTouchEnded=CC_CALLBACK_2(GameLayer::onTouchEnded,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">);
listener->setSwallowTouches( true ); //不向下传递触摸
dispatcher->addEventListenerWithSceneGraphPriority(listener,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">);

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

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

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

6
autolistener1=EventListenerTouchAllAtOnce::create();
listener1->onTouchesBegan=CC_CALLBACK_2(GameLayer::onTouchesBegan,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">listener1->onTouchesMoved=CC_CALLBACK_2(GameLayer::onTouchesMoved,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">listener1->onTouchesEnded=CC_CALLBACK_2(GameLayer::onTouchesEnded,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">dispatcher->addEventListenerWithSceneGraphPriority(listener1,serif; font-size:14px"> 或者setTouchEnabled(true),然后重写layer的onTouchsxxx函数

关于eventDispatcher

1) 获取方法:

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

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

触摸事件 (EventListenerTouch)

键盘响应事件 (EventListenerKeyboard)

加速记录事件 (EventListenerAcceleration)

鼠标响应事件 (EventListenerMouse)

自定义事件 (EventListenerCustom)

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

2)优先权:

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

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

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

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

代码展开一下:

11
12
13
14
{
CCASSERT(listener&&node, "Invalidparameters." );
CCASSERT(!listener->isRegistered(),monospace!important; font-size:1em!important; min-height:inherit!important; color:blue!important">"Thelistenerhasbeenregistered." );
if (!listener->checkAvailable())
return ;
listener->setSceneGraphPriority(node);
listener->setFixedPriority(0);
listener->setRegistered( );
addEventListener(listener);
}
14
15
16
fixedPriority)
CCASSERT(listener,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">CCASSERT(fixedPriority!=0,monospace!important; font-size:1em!important; min-height:inherit!important; color:blue!important">"0priorityisforbiddenforfixedprioritysinceit'susedforscenegraphbasedpriority." );
(!listener->checkAvailable())
;
listener->setSceneGraphPriority(nullptr);
listener->setFixedPriority(fixedPriority);
);
listener->setPaused( false );
addEventListener(listener);
}

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

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

移除方法:

dispatcher->removeEventListener(listener);

其实还可以这么用:

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->onTouchBegan=[&](Touch*t,Event*e){
//改变贪食蛇移动的方向
col=t->getLocation().x/32;
row=t->getLocation().y/32;
spHeadCol=spHead->getPositionX()/32;
spHeadRow=spHead->getPositionY()/32;
( abs (spHeadCol-col)> (spHeadRow-row))
{
(spHeadCol<col)
{
spHead->m_dir=ENUM_DIR::DIR_RIGHT;
} else
{
spHead->m_dir=ENUM_DIR::DIR_LEFT;
}
}
else
{ (spHeadRow<row)
{
spHead->m_dir=ENUM_DIR::DIR_UP;
else
{
spHead->m_dir=ENUM_DIR::DIR_DOWN;
}
}
return ;
}; //这条语句像不像Java中的匿名接口对象,像不像Objective-C中Block,他有个很华丽的名字lambda,读音为:腊母达,相信未来你会喜欢上这妹子的
//3注册这个侦听到消息分发器中
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener,monospace!important; font-size:1em!important; min-height:inherit!important; color:black!important">);

(编辑:李大同)

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