上篇我们完成了简单的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来控制坦克的移动。
下面我们编译运行程序看看效果:
完整代码下载地址: