GameScene类虽然是占用游戏最多时间的类,但是里面的东西不是很多,最重要的就是碰撞检测了,碰撞检测代码如下:

 void GameScene::detectionCrash()
{ CCArray* bulletsToDelete = CCArray::create();//创建一个CCArray,用以存放待删除的子弹,也就是此帧中被检测到碰撞的子弹 
bulletsToDelete->retain();//必须调用retain,CCArray内部调用了autoRelease
CCObject* bt,*et; CCArray* enemyToDelete = CCArray::create();//创建一个CCArray,用以存放待删除的敌机,也就是此子弹击中的敌机
enemyToDelete->retain();//调用retain   CCRect rectHero = this->heroLayer->getHero()->boundingBox();
float x = rectHero.origin.x + rectHero.size.width * 0.3;
float y = rectHero.origin.y + rectHero.size.height * 0.4;
float width = rectHero.size.width * 0.3;
float height = rectHero.size.height * 0.6;
CCRect rect_HeroForCrash = CCRectMake(x, y, width, height); //检测敌机和hero是否相撞
CCARRAY_FOREACH(this->enemyLayer->m_enemys,et)//遍历所有敌机
{
//break; Enemy* enemy = (Enemy*)et;
if (enemy->getLife() == )
{
break;
} // CCPoint rect1 = this->heroLayer->getHero()->getPosition();
//boundingBox 获取的是相对于父节点的左下角为原点的一个rect,所以要比较两个精灵是否相交,他们的父节点的坐标原点和大小应该一样
if(enemy->boundingBox().intersectsRect(rect_HeroForCrash))
{ this->heroLayer->setIsHeroLive(false);
this->heroLayer->setHeroLifes(this->heroLayer->getHeroLifes() - ); enemyLayer->stopTakeEnemy();
enemyLayer->bomb(enemy);
enemyLayer->removeAllEnmeys(); SimpleAudioEngine::sharedEngine()->pauseBackgroundMusic(); int tempHightScore = GameScene::getHightestScore();
if (this->m_totalScore > tempHightScore)
{
GameScene::saveHightestScore(this->m_totalScore);
} char life[];
sprintf(life, "%d", this->heroLayer->getHeroLifes());
CCLabelTTF* lbLife = (CCLabelTTF*)this->getChildByTag(tagOfLife);
lbLife->setString(life); this->heroLayer->heroBomb(0.1f);
if (this->heroLayer->getHeroLifes() == )
{ this->scheduleOnce(schedule_selector(GameScene::gameOverCallback), 2.0f);
}
else
{
this->scheduleOnce(schedule_selector(GameScene::newLife), 2.0f);
} return;
}
} //检测敌机和子弹是否相撞
CCARRAY_FOREACH(this->heroLayer->getBullets()->m_bullets,bt)//遍历所有子弹
{
CCSprite* bullet = (CCSprite*)bt; CCARRAY_FOREACH(this->enemyLayer->m_enemys,et)//遍历所有敌机
{
Enemy* enemy3 = (Enemy*)et; if(enemy3->boundingBox().intersectsRect(bullet->boundingBox()))
{ if (enemy3->getLife() > )
{
enemy3->loseLife();
bulletsToDelete->addObject(bullet);//把待删除子弹放入CCArray  
}
else if (enemy3->getLife() == )
{
enemy3->loseLife();
bulletsToDelete->addObject(bullet);//把待删除子弹放入CCArray  
enemyToDelete->addObject(enemy3);//把待删除敌机放入CCArray  this->m_totalScore += enemy3->getScore(); char str1[];
sprintf(str1, "%d",(int)this->m_totalScore); CCLabelTTF* label1 = (CCLabelTTF*)this->getChildByTag(tagOfScore); label1->setString(str1);
} }
} } CCARRAY_FOREACH(enemyToDelete,et)//遍历所有此帧中碰撞死亡的敌机,必须是死亡
{
CCSprite* enemy3 = (CCSprite*)et;
this->enemyLayer->bomb(enemy3);//执行爆炸
}
enemyToDelete->release();//release CCARRAY_FOREACH(bulletsToDelete,bt)//遍历所有此帧中碰撞的子弹
{
CCSprite* bullet = (CCSprite*)bt;
this->heroLayer->getBullets()->removeBullet(bullet);//执行移除
}
bulletsToDelete->release();//release
}

还记得之前说过 boundingbox 方法是获取节点以父节点左下角为原点的一个矩形吧,该矩形的大小就是节点各种变形后的大小,矩形的左下角坐标就是节点变形后的左下角坐标

bullet是加到 ccspriteBatchNode 上的,ccspriteBatchNode是加到 BulletLayer上面的,ccspriteBatchNode和bulletLayer都是铺满屏幕的。所以子弹调用 boundingbox 获得的矩形是以屏幕左下角为原点的。敌机的原理也是如此。hero是直接加到 herolayer 上的,父节点同样是铺满屏幕的,所以他们的父节点的左下角的坐标都一样的,这就是通过 boundingbox 检测他们是否碰撞的前提。

有些纹理周围有比较大的空白地方,这就会造成两个节点看起来还没有接触就会发生碰撞,这就需要对碰撞进行更精确的判断,可以通过什么像素判断法之类的。我这里采取最简单的处理方法,就是通过节点的boundingbox获取到一个rect,然后对这个rect加工一下:

     CCRect rectHero = this->heroLayer->getHero()->boundingBox();
float x = rectHero.origin.x + rectHero.size.width * 0.3;
float y = rectHero.origin.y + rectHero.size.height * 0.4;
float width = rectHero.size.width * 0.3;
float height = rectHero.size.height * 0.6;
CCRect rect_HeroForCrash = CCRectMake(x, y, width, height);

这样子可以获取节点 boundingbox 里面的某一部分来作为碰撞的检测部分

cocos2dx打飞机项目笔记六:GameScene类和碰撞检测 boundingbox的更多相关文章

  1. cocos2dx打飞机项目笔记三:HeroLayer类和坐标系

    HeroLayer类主要是处理hero的一些相关东西,以及调用bulletLayer的一些方法,因为子弹是附属于hero的~~ HeroLayer 类的成员如下: class HeroLayer : ...

  2. cocos2dx打飞机项目笔记四:Enemy类和EnemyLayer类

    Enemy类没什么内容,就create和init方法,根据参数来创建不同的敌机,头文件代码如下: //飞机的类型 enum planeType {smallPlane, midPlane, bigPl ...

  3. cocos2dx打飞机项目笔记二:BulletLayer类

    BulletLayer.h 内容如下 class BulletLayer : public cocos2d::CCLayer { public: CC_SYNTHESIZE(bool, m_IsHer ...

  4. cocos2dx打飞机项目笔记一:项目结构介绍

    最近在学习cocos2dx引擎,版本是2.1.3,开发环境是win7 + vs2010,模仿微信打飞机游戏,开发中参考了 csdn 偶尔e网事 的系列文章:http://blog.csdn.net/c ...

  5. cocos2dx打飞机项目笔记七:各种回调:定时器schedule、普通回调callFunc、菜单回调menu_selector、事件回调event_selector

    各种回调函数的定义: typedef void (CCObject::*SEL_SCHEDULE)(float); typedef void (CCObject::*SEL_CallFunc)(); ...

  6. cocos2dx打飞机项目笔记五:CCSpriteBatchNode 的使用

    在上一节里,在头文件看到 定义了一个 CCSpriteBatchNode* batchNode;,在addEnemy方法里看到 batchNode->addChild(enemy); 新建的敌机 ...

  7. python学习笔记六--用户自定义类

    一.类: 1. 面向对象. 2. 定义了新的对象类型. 定义了两个属性:name,pay 定义了两个方法:lastName,giveRaise

  8. php笔记(六)PHP类与对象之对象接口

    接口的实现 <?php //interface 关键字用于定义一个接口 interface ICanEat{ //接口里面的方法不需要实现 public function eat($food); ...

  9. python学习笔记(六)、类

    Python与java.c++等都被视为一种面向对象的语言.通过创建自定义类,用于处理各种业务逻辑.面向对象有封装.继承.多态三个特征,这也是面子对象语言的通用特征. 1 封装 封装,是值向外部隐藏内 ...

随机推荐

  1. tinyint(4),tinyint(80)有什么区别

    tinyint格式: TINYINT[(M)] [UNSIGNED] [ZEROFILL] M默认为4 Tinyint占用1字节的存储空间,即8位(bit). 带符号的范围是-128到127.无符号的 ...

  2. 大量数据导入导致mysql自动重启

    昨天晚上第十款做数据库迁移,数据库版本:Version: '5.1.61',数据量27G左右 message报错信息如下: Mar :: VM_163_210_tlinux kernel: [, oo ...

  3. ThinkPHP带表情无限级评论回复

    今天就tp中(含表情)无限级评论回复做一个个人总结. 1.准备TP基本框架 2.数据库,数据表的建立 A.先说说数据库(表)的建立. a-1,数据库:blog a-2,数据表:bolg_comment ...

  4. Eigen求矩阵行列式 及 行列式本质

    转置.伴随.行列式.逆矩阵 小矩阵(4 * 4及以下)eigen会自动优化,默认采用LU分解,效率不高 #include <iostream> #include <Eigen/Den ...

  5. hdu 4705(树形DP)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4705 思路:反面考虑,用总的方案数减去A,B,C三点在同一路径上的方案数.于是我们可以确定中间点B,在 ...

  6. OpenCV学习笔记五:opencv_legacy模块

    opencv_legacy,顾名思义,该模块是用于兼容以前的opencv代码而设立的. 如果你希望用最新的opencv代码和特性,请勿使用该模块.

  7. 同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)四种调用方式

    1. 概念理解        在进行网络编程时,我们常常见到同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)四种调用方式:   同步/异步主要针对C端: 同步:    ...

  8. 41个Web开发者必须收藏的JavaScript实用技巧

    1. 将彻底屏蔽鼠标右键 oncontextmenu=”window.event.returnValue=false” < table border oncontextmenu=return(f ...

  9. Android Studio使用百度地图问题总结

    一.常见问题APP Scode码校验失败 一般出现这个问题都是ak不正确导致 可能出错的地方 1.SHA1值没有正确获取 正确获取SHA1值:在左下角打开Terminal终端,进入debug.keys ...

  10. Android无线测试之—UiAutomator UiSelector API介绍之六

    对象搜索—类名与包名 一.类名属性定位对象 返回值 API 描述 UiSelector calssName(String className) 完整类名匹配 UiSelector calssNameM ...