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. eclipse 查看源代码

    文地址:http://blog.csdn.net/sushengmiyan/article/details/18798473 本文作者:sushengmiyan 我们在使用Eclipse的时候,经常是 ...

  2. MySQL没有远程连接权限设置

    GRANT ALL PRIVILEGES ON *.* TO 'root'@'%' IDENTIFIED BY '123456' WITH GRANT OPTION; flush privileges ...

  3. TP表单验证

    [表单验证] javascript jquery 在服务器端通过tp框架实现表单验证 用户名.密码.重复密码.邮箱.qq.手机号码.爱好.学历 具体步骤: 制作表单 表单form数据通过create( ...

  4. mysql导出csv文件

    SELECT * FROM (select 'id','url','大分类','分类','贴吧名称','关注用户数','帖子数量','简介','帖子名称','楼主ID','发表时间','采集时间',' ...

  5. 一个有趣的IP不同的问题?

    1.我们已经知道了内网和外网的显示是不同的. 2.今天发现了我的飞Q传输上显示的ip和电脑上网络中显示的ip不同,但是传输文件是可以的,至于这个问题目前没有找到合理的解释,解释这个问题,但是这样就奇怪 ...

  6. Introduction to Mathematical Thinking - Week 7

    Q: Why did nineteenth century mathematicians devote time to the proof of self-evident results? Selec ...

  7. exe4j中"this executable was created with an evaluation version of exe4j"

    在使用exe4j时,如果您的exe4j没有注册,在运行有exe4j转换的*.jar为*.exe的可执行文件是会提示:"this executable was created with an ...

  8. QBuffer简单操作(被看做一个标准的可随机访问的文件,支持信号)

    Qt中通过QBuffer类我们可以使用io的方式访问.操作QByteArray中的内容.此时,QByteArray被看做一个标准的可随机访问的文件.例如: QBuffer buffer; char c ...

  9. python多线程(二)

    开启线程的两种方式 #方式一from threading import Threadimport timedef sayhi(name):   time.sleep(2)   print('%s sa ...

  10. 1、Python request(爬虫-百度翻译)

    #encoding=utf-8 import requests def fanyi(): while True: context = input("请输入翻译的内容(退出q):") ...