事实上我也非常吃惊…居然写到第七篇了,我估计也就是四篇的内容,感觉非常奇妙,我也不会非常唠叨什么吖(小若:32个喷!
),怎么都到第七篇了。
笨木头花心贡献,啥?花心?不呢,是用心~
转载请注明,原文地址: http://www.benmutou.com/blog/archives/920
文章来源:笨木头与游戏开发
碰撞监听
首先,确保我们创建物理对象的时候,给对象设置了碰撞条件(假设你是一步步按着教程来写的代码,那就是设置好了):
- body->setCategoryBitmask(1); // 0001
- body->setCollisionBitmask(1); // 0001
- body->setContactTestBitmask(1); // 0001
这样我们才干监听到它们的碰撞事件,至于原理。就不说了。以我的唠叨程度,不是一两篇内容能说完的。
然后。我们给TollgateScene加入一个函数声明:
- /* 碰撞检測 */
- bool onContactBegin(PhysicsContact& contact);
这是碰撞事件開始时的回调函数,监听碰撞事件非常easy。我们修改一下TollgateScene的init函数:
- bool TollgateScene::init()
- {
- if (!Layer::init())
- {
- return false;
- }
- /* 创建主角 */
- Size visibleSize = Director::getInstance()->getVisibleSize();
- m_player = Player::create();
- m_player->setPosition(Point(visibleSize.width * 0.5f, visibleSize.height * 0.85f));
- this->addChild(m_player, 5);
- /* 创建操作UI */
- createOprUI();
- /* 碰撞监听 */
- auto contactListener = EventListenerPhysicsContact::create();
- contactListener->onContactBegin = CC_CALLBACK_1(TollgateScene::onContactBegin, this);
- _eventDispatcher->addEventListenerWithSceneGraphPriority(contactListener, this);
- this->schedule(schedule_selector(TollgateScene::logic));
- return true;
- }
流程就是这样:
1. 创建EventListenerPhysicsContact对象,能够看做是碰撞监听回调接口
2. 绑定onContactBegin事件的回调函数
3. 将监听接口加入到统一的事件派发器里(addEventListenerWithSceneGraphPriority)
Cocos2d-x3.0的监听事件修改非常大(但非常好用)。这里就不多解释了,网上非常多文章有介绍。
假设大家有去看看EventListenerPhysicsContact的源代码的话,会发现,碰撞事件不仅仅仅仅有onContactBegin一个,其它的事件我就不说了。这里仅仅是要onContactBegin。作用是在两个物理对象開始发生碰撞的时候调用。
好,最后看看onContactBegin函数实现:
- bool TollgateScene::onContactBegin(PhysicsContact& contact)
- {
- auto nodeA = (Sprite*)contact.getShapeA()->getBody()->getNode();
- auto nodeB = (Sprite*)contact.getShapeB()->getBody()->getNode();
- return true;
- }
当中获取到的nodeA和nodeB就是发生碰撞的两个节点对象。
如今。看看碰撞检測是否正常吧。用调试模式执行游戏。然后在onContactBegin函数里打个断点,看看主角碰到墙的时候有没有进入这个断点吧~
假设有,那就代表成功了。
然后,另一点别忘了,在TollgateScene的onExit函数里,把监听事件给取消了:
- void TollgateScene::onExit()
- {
- Layer::onExit();
- _eventDispatcher->removeEventListenersForTarget(this);
- }
怎么知道发生碰撞的两个节点各自是谁?
如今我们仅仅知道有两个节点碰撞了。也能取到两个节点对象,可是,它们都是谁啊?根本不认识啊。那怎么进行下一步操作呢?我想给Player对象加血,那怎么办呢?
没关系,节点有一个非常万能的函数:setTag。
给节点设置Tag可不是仅仅用来从Layer里获取子节点对象。我们还能够用来区分这些节点是谁,应该是。区分这些节点是属于哪一类东西。
创建一个生物分类表
好。是时候新建一个头文件了,为了迎合这份高大上的教程,我们就称之为生物分类表吧~
创建一个头文件,命名为ObjectTag.h,内容例如以下:
- #ifndef ObjectTag_H
- #define ObjectTag_H
- #define ObjectTag_Player 1
- #define ObjectTag_Border 2
- #define ObjectTag_Monster 3
- #endif
这大实用处,别着急~
给各种物体设置Tag吧
好了,如今我们要给主角和墙(或者称之为锯齿)设定生物类别了。在Player的init函数的最后加上一句代码:
- bool Player::init()
- {
- /* 这里省略了非常多代码 */
- this->setTag(ObjectTag_Player);
- return true;
- }
当然,ObjectTag.h头文件也别忘了加上。
然后。给BackgroundLayer的createBorder函数最后也加上一句代码:
- Sprite* BackgroundLayer::createBorder(Point pos)
- {
- /* 这里省略了非常多代码 */
- border->setTag(ObjectTag_Border);
- return border;
- }
開始区分谁是谁
OK了,主角和墙都有了各自的生物类型了~如今我们能够区分碰撞的两个对象各自是谁了。
我们要在TollgateScene的onContactBegin函数里做处理:
- bool TollgateScene::onContactBegin(PhysicsContact& contact)
- {
- auto nodeA = (Sprite*)contact.getShapeA()->getBody()->getNode();
- auto nodeB = (Sprite*)contact.getShapeB()->getBody()->getNode();
- if (nodeA == NULL || nodeB == NULL)
- {
- return true;
- }
- Node* playerNode = NULL; /* 玩家对象 */
- Node* other = NULL; /* 怪物或墙等其它对象 */
- if (nodeA->getTag() == ObjectTag_Player)
- {
- playerNode = nodeA;
- other = nodeB;
- }
- else if (nodeB->getTag() == ObjectTag_Player)
- {
- playerNode = nodeB;
- other = nodeA;
- }
- else
- {
- /* 假设两个碰撞的物体中,不存在玩家对象,就忽略,不做处理 */
- return true;
- }
- Player* player = (Player*)playerNode;
- /* 碰撞到边缘锯齿(墙),+1血 */
- if (other->getTag() == ObjectTag_Border)
- {
- /* 扣-1血。就相当于加1血 */
- player->beAtked(-1);
- log("player cur HP:%d", player->getiHP());
- }
- return true;
- }
(小若:这么长的代码。打死我我也不看~!
)
这段代码要做的事情事实上非常easy,萝莉一下,萝莉、罗莉。罗列,嗯(这输入法坏了,一定是):
1.推断nodeA的Tag是不是ObjectTag_Player,假设是。那么nodeA就是Player对象了,同一时候。nodeB仅仅能是墙或者是怪物对象了(由于游戏里仅仅有一个Player对象)
2.假设nodeA不是Player,那就继续推断nodeB
3.假设nodeA和nodeB都不是Player对象,那我们就不做处理。由于怪物和怪物之间的碰撞不须要做处理
4.假设找到Player对象,那就推断other对象是不是墙,是的话。那就让Player加1滴血
5.由于这个实例缺少非常多功能,比方UI、数据绑定、碰撞时产生的动画效果之类的。所以没法直观地看到Player加血的动作,仅仅好用打印日志的方式来查看了。
有朋友提醒我漏了解说Player的beAtked函数,这里补充一下。如代码:
- void Player::beAtked(int iValue)
- {
- if (iValue < 0)
- {
- cure(-iValue);
- }
- else
- {
- hurt(std::abs(iValue));
- }
- }
- void Player::hurt(int iValue)
- {
- setiHP(getiHP() - iValue);
- }
- void Player::cure(int iValue)
- {
- setiHP(getiHP() + iValue);
- }
好了,如今用调试模式执行游戏(键盘F5),使劲让主角撞墙吧。然后看看日志输出:
player cur HP:101
player cur HP:102
player cur HP:103
player cur HP:104
player cur HP:105
player cur HP:106
player cur HP:107
player cur HP:108
player cur HP:109
假设有相似以上的日志输出。那就证明我们成功了~
Cocos2d-x3.0游戏实例之《别救我》第七篇——物理世界的碰撞检測的更多相关文章
-
cocos2d-x 3.0游戏实例学习笔记 《跑酷》第七步--物理碰撞检測(1)
说明:这里是借鉴:晓风残月前辈的博客,他是将泰然网的跑酷教程,用cocos2d-x 2.X 版本号重写的,眼下我正在学习cocos2d-X3.0 于是就用cocos2d-X 3.0重写,并做相关笔记 ...
-
Cocos2d-x3.0游戏实例《不要救我》第十篇(结束)——使用Json配置数据类型的怪物
如今我们有2种类型的怪物,并且创建的时候是写死在代码里的,这是要作死的节奏~ 所以.必须可配置.不然会累死人的. ; i < size; ++i) { int id = root[i][&quo ...
-
Cocos2d-x3.0游戏实例《不要救我》第一章——前言
我们可以学习? 这是一个非常easy游戏.但更多的东西用(对于初学者).至少,对于它的一个例子,有点多. 笨木头花心贡献.啥?花心?不呢.是用心~ 转载请注明,原文地址:http://www.benm ...
-
Cocos2d-x3.0游戏实例《不要救我》三——背景滚动周期
好.让我们来解释一下这个无限循环滚动的背景.这方面的知识一直讲到烂.我以前的文章还介绍了.所以不是那么特别清楚. 笨木头花心贡献,啥?花心?不呢,是用心~ 转载请注明,原文地址:http://www. ...
-
实例介绍Cocos2d-x中Box2D物理引擎:碰撞检測
在Box2D中碰撞事件通过实现b2ContactListener类函数实现,b2ContactListener是Box2D提供的抽象类,它的抽象函数:virtual void BeginContact ...
-
Cocos2d-x3.0游戏实例之《别救我》第六篇——从代码中获取UI控件
这篇的内容非常easy,获取UI控件,然后使用它. 还记得我们在UI编辑器中给三个button分别命名了吧? 如今要用上了. 笨木头花心贡献,啥?花心?不呢,是用心~ 转载请注明,原文地址: http ...
-
Cocos2d-x3.0游戏实例之《别救我》第八篇——TiledMap实现关卡编辑器
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/musicvs/article/details/25368273 好吧.我真心全然搞不懂.我如今仅仅只 ...
-
Cocos2d-x3.0游戏实例之《别救我》第二篇——创建物理世界
这篇我要给大家介绍两个知识点: 1. 创建游戏物理世界 2. 没了(小若:我噗) 害怕了?不用操心.这太简单了~! 笨木头花心贡献.啥?花心?不呢.是用心~ 转载请注明,原文地址:http://www ...
-
cocos2d-x ios游戏开发初认识(八) 触摸事件与碰撞检測
玩过植物大战僵尸都知道,要在草坪里放一朵向日葵或者其他的植物仅仅需触摸那个植物将其拖入到想要摆放的位置,这事实上就是这节要写的触摸事件.还能够发现当我们的僵尸出来的时候,我们的小豌豆会发子弹攻击僵尸, ...
随机推荐
-
sql 列转行 实例
select * from dbo.orders group by custid with cube select custid ,row_number() over(order by e ...
-
redis 认证密码
[root@cache01 ~]# grep "requirepass" /app/server/redis/conf/6379.conf # If the master is p ...
-
highcharts报表插件之expoting参数的使用
exporting 参数配置 本文转载自:http://blog.csdn.net/myjlvzlp/article/details/8531275 说明:导出及打印选项 打印导出功能的配置项. 1. ...
-
js点击弹出div层
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
-
mysql远程授权
切换到mysql的mysql数据库,找到user表: cmd:mysql -u root -p cmd:use mysql; cmd:select host,user,password from us ...
-
sqlplus命令历史解决方案
在Linux上使用sqlplus比较痛苦,因为不能使用上下方向键来调出命令历史,也不能使用左右键移动光标对输入的命令进行修改,甚至连Backspace键都不能用(不过我发现大部分Backspace不能 ...
-
用python在excel中读取与生成随机数写入excel中
今天是我第一次发博客,就关于python在excel中的应用作为我的第一篇吧. 具体要求是:在一份已知的excel表格中读取学生的学号与姓名,再将这些数据放到新的excel表中的第一列与第二列,最后再 ...
-
[转帖]Ansible管理windows集群
Ansible管理windows集群 http://www.cnblogs.com/Dev0ps/p/10026908.html 写的挺好的 我关注点还是不够好呢 最近公司新项目需要安装400+win ...
-
linux 下mysql服务的管理
一.mysql服务的管理 1.1 mysql启动与关闭 linux下启动mysql: /etc/init.d/mysqld start 关闭进程: ps -ef | grep mysql 找到进程号 ...
-
linux网络编程:splice函数和tee( )函数高效的零拷贝
splice( )函数 在两个文件描述符之间移动数据,同sendfile( )函数一样,也是零拷贝. 函数原型: #include <fcntl.h> ssize_t splice(int ...