上篇我们完成了简单的AI编写,但是各个坦克移动时之间是可以重合的,

这节课我们来完成坦克之间的碰撞检测,还是在上篇的EnemyAI中完成。

1.我先现在坦克类Tank中添加两个成员变量:

	CC_SYNTHESIZE(CCRect, mMovedRect, MovedRect);
CC_SYNTHESIZE(bool, IsBlock, Block);

mMovedRect是移动后的位置,目的是保存移动后的位置来检测碰撞,

如果发现与其他坦克碰撞了,则不执行移动动作;

IsBlock是用来标记坦克是否被碰撞的。

2.然后我们还得在Tank类中添加一个移动函数,根据是否阻塞来控制坦克移动:

	void move();

具体实现很简单,如下:

void Tank::move()
{
if (!IsBlock)
{
setPosition(ccp(mMovedRect.getMidX(), mMovedRect.getMidY()));
}
}

3.既然是碰撞检测,当然得有简单的碰撞函数先是检测两个矩形区域是否相交:

bool EnemyAI::IsRectIntersect(CCRect rectA, CCRect rectB)
{
float left = max(rectA.getMinX(), rectB.getMinX());
float right = min(rectA.getMaxX(), rectB.getMaxX());
if (left > right)
return false; float top = min(rectA.getMaxY(), rectB.getMaxY());
float bottom = max(rectA.getMinY(), rectB.getMinY());
if (top < bottom)
return false; return true;
}

如上,检测原理很简单,

假设A矩形在B矩形左边,那么A矩形的右边的X坐标小于B矩形左边的X坐标,

那么两个矩形不相交,反之A矩形在B矩形右边亦然。

在假设A矩形在B矩形上方或下方,检测原理和上面相同。

如果检测到矩形相交,则返回true。

4.我们再到EnemyAI中添加一个函数,来检测所有坦克之间的碰撞:

void EnemyAI::collisionTest()
{
//坦克和敌方坦克之间的碰撞检测
CCObject* pObj;
CCARRAY_FOREACH(mEnemyTanks, pObj)
{
Tank* enemyTank = (Tank*)pObj;
if (IsRectIntersect(mTank->getMovedRect(), enemyTank->getMovedRect()))
{
enemyTank->setBlock(true);
mTank->setBlock(true);
}
} //敌方坦克之间的碰撞
CCArray* ccTmpArray = CCArray::create();
ccTmpArray->addObjectsFromArray(mEnemyTanks);
while (ccTmpArray->count())
{
CCObject* pObj;
Tank* tmpTank = (Tank*)ccTmpArray->lastObject();
ccTmpArray->removeLastObject();
CCARRAY_FOREACH(ccTmpArray, pObj)
{
if (IsRectIntersect(tmpTank->getMovedRect(), ((Tank*)pObj)->getMovedRect()))
{
tmpTank->setBlock(true);
((Tank*)pObj)->setBlock(true);
ccTmpArray->removeObject(pObj);
}
}
}
}

如上,我们先单个检测玩家坦克和地方坦克的碰撞,然后检测敌方坦克之间的碰撞:

我们先从敌方坦克数组中取出一辆坦克,然后将他从数组中移除,

再跟其他所有坦克进行碰撞检测,如果发现有碰撞的坦克,设置他的IsBlock阻塞标记为true,

然后将它从数组中移除,如此循环,直到数组中所有坦克检测完成。

5.最后我们需要在EnemyAI中的void EnemyAI::tankAction(float delta)

中检测碰撞以及控制坦克行为:

void EnemyAI::tankAction(float delta)
{
CCObject* pObj;
CCARRAY_FOREACH(mEnemyTanks, pObj)
{
Tank* tank = (Tank*)pObj; //坦克按照上次的方向一直往前走
int Rotation = tank->getRotation();
tank->command((enumOrder)(Rotation / 90 + 1)); //坦克每隔一秒开一次火
tank->setBulletDelta(tank->getBulletDelta() + delta);
if (tank->getBulletDelta() > 1)
{
//开火后,如果子弹在飞行中,归零计时
if (tank->command(cmdFire))
{
tank->setBulletDelta(0.0);
}
} //检测坦克之间的碰撞
collisionTest(); //如果坦克阻塞,换个方向
if (tank->getBlock())
tank->setRotation((int)(CCRANDOM_0_1() * 3.2) * 90);
//如果上面的判断完成后,坦克根据自己的阻塞状态移动
tank->move();
}
mTank->move();
}

可以看到在检测了碰撞后,用move来控制坦克的移动。

下面我们编译运行程序看看效果:

完整代码下载地址:

http://download.csdn.net/detail/yincheng01/6773421

cocos2d-x游戏开发系列教程-坦克大战游戏之所有坦克之间的碰撞检测的更多相关文章

  1. HTML5游戏开发系列教程6(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-6/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...

  2. HTML5游戏开发系列教程7(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-7/ 今天我们将完成我们第一个完整的游戏--打砖块.这次教程中,将 ...

  3. HTML5游戏开发系列教程5(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-5/ 最终我决定准备下一篇游戏开发系列的文章,我们将继续使用can ...

  4. HTML5游戏开发系列教程4(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-4/ 这篇文章是我们继续使用canvas来进行HTML5游戏开发系 ...

  5. cocos2d-x游戏开发系列教程-坦克大战游戏启动界面的编写

    用前面介绍的方法,创建一个cocos2d-x项目,可以看到新项目内容如下图:

  6. HTML5游戏开发系列教程8(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-8/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...

  7. HTML5游戏开发系列教程10(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-10/ 最后我们将继续使用canvas来进行HTML5游戏开发系列 ...

  8. HTML5游戏开发系列教程9(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-9/ 今天我们将继续使用canvas来进行HTML5游戏开发系列的 ...

  9. cocos2d-x游戏开发系列教程-坦克大战游戏之子弹的碰撞检测处理

    在上篇我们加上了简单的坦克之间的碰撞检测,这篇我们继续加上子弹之间, 子弹与坦克之间的碰撞检测,对于上一篇碰撞处理不太完美的地方我们继续改进. 1.子弹之间的碰撞 //玩家子弹和敌方子弹之间的碰撞 C ...

随机推荐

  1. LeetCode 二叉树后序遍历(binary-tree-postorder-traversal)

    Given a binary tree, return the postorder traversal of its nodes' values. For example:Given binary t ...

  2. POJ 1458 Common Subsequence(LCS最长公共子序列)

    POJ 1458 Common Subsequence(LCS最长公共子序列)解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?c ...

  3. debian6 更新python版本到python3.3

    1.下载python3.3安装包 #wget wget --no-cookie --no-check-certificate --header "Cookie:gpw_e24=http%3A ...

  4. CentOS 6.5 PYPI本地源制作

    转载:blog.csdn.net/tantexian   一.安装pip2pi工具: pip install pip2pi 或 git clone https://github.com/wolever ...

  5. python成长之路10

    断点续传   python2.7 多继承  py35多继承   socketserver源码     支持并发处理socket   i/o多路复用   上节回顾     socket          ...

  6. 2013 南京邀请赛 K题 yet another end of the world

    /** 大意:给定一组x[],y[],z[] 确定有没有两个不同的x[i], x[j] 看是否存在一个ID使得 y[i]<=ID%x[i]<=z[i] y[j]<=ID%x[j]&l ...

  7. Qt Creator error: LNK1123: 转换到 COFF 期间失败: 文件无效或损坏

    Qt Creator error: LNK1123: 转换到 COFF 期间失败: 文件无效或损坏   治标又治本的解决方法:   找到在 { C:\Windows\Microsoft.NET\Fra ...

  8. 转:STL使用入门( Using STL)

    1 介绍 我最开始结束C++编程是从DOS下的Borland C++开始的.那时他们在最新版本3.1中就包含了一套模板库用来做collection.那真是个好东东.当我开始使用Visual C++ 2 ...

  9. CMake学习小结

    假定有vegagis工程,工程的目录结构如下: #--vegagis#  |--src 源文件目录#     |--gui 界面工程,输出类型:dll,依赖于QT的QtCore.QtGui.QtXml ...

  10. cmake学习笔记(五)

    在cmake 学习笔记(三) 中简单学习了 find_package 的 model 模式,在cmake 学习笔记(四)中了解一个CMakeCache相关的东西.但靠这些知识还是不能看懂PySide使 ...