使用cocos2dx结合bullet设计一款简陋的桌球游戏,就是为了回顾前期学过的bullet。
首先要把桌球游戏需要的基本资源准备好,15个球,1个白球,1张台球桌,球杆可有可无。
先看看目前实现的效果
至于这张台球桌的模型,我是随便设计一下
当然这个模型只是为了加载raw文件(静态网格数据),为了使模型的贴图显示出来,
我使用Blender直接创建了一个Plane,然后将台球桌的贴图贴在Plane上,于是就能以假乱真的
实现一个台球桌
在游戏开始前,初始化物理环境,加载一张台球桌,设置相应的物理属性,按规则摆放好台球。
1.设置重力为(0.f, -9.8f, 0.f),以模拟真实的物理环境。
_world = PhysicsWorld3D::createWithDebug(this, btVector3(0.f, -9.8f, 0.f));
2.加载台球桌
首先将台球桌的贴图模型加载进游戏,设置相应的位置,
然后加载台球桌的物理网格,还记得PhysicisMesh3D吗,并调整好位置,
当然比较不好设置的就是台球桌的物理属性,什么摩擦系数啊,弹性系数啊,滚动摩擦系数啊
void HelloWorld::initTable()
{
auto tableSprite = Sprite3D::create("ball/table.c3b");
this->addChild(tableSprite);
tableSprite->setPosition3D(Vec3(0.f, -0.57f, 0.f));
tableSprite->setCameraMask(2);
_tableMesh = PhysicsMesh3D::constuct("table.raw");
_world->addTriangleMesh(_tableMesh, btVector3(0, -0.57f, 0), PhysicsMaterial3D(0.f, 0.5f, 0.2f, 0.2f));
}
3.摆放台球
对于15球来说摆放的顺序是这样的,在网上找的规则
黑8放在第三行的中间位置,白色的为全色球,黑色的为花色球。
可以这样设想,只要定义一个数组存放每个位置的球号就行了,
BALLS_NUMBER[0] = 1;
BALLS_NUMBER[1] = 2;
BALLS_NUMBER[2] = 9;
BALLS_NUMBER[3] = 10;
BALLS_NUMBER[4] = 8;
BALLS_NUMBER[5] = 3;
BALLS_NUMBER[6] = 4;
BALLS_NUMBER[7] = 11;
BALLS_NUMBER[8] = 5;
BALLS_NUMBER[9] = 12;
BALLS_NUMBER[10] = 13;
BALLS_NUMBER[11] = 6;
BALLS_NUMBER[12] = 14;
BALLS_NUMBER[13] = 15;
BALLS_NUMBER[14] = 7;
球号是我自己按照规则随便放的。
下面就是如果将这些球放好,
假设每个球半径为0.57f,球都在Y坐标为0的位置,那么关键就是如何确定每个球的
X,Z.台球摆放好无论多少行都是个等边三角形,以3行为例
先放置第一个球,以后每一行的第一个球都是按照蓝色箭头的方向放置,假设上为Z,右为X
那么第二行第一个球就是(ball[1].posX + ball.radius, ball[1].posZ + √3*ball.radius)
设方向向量为dir(ball.radius, 0, √3*ball.radius);
即ball[2].pos = ball[1].pos + dir
第三行就是ball[3].pos = ball[1].pos + dir * 2;
一次类推ball[n].pos = ball[1].pos + dir * (n-1);
对于同一行的第k个球就是同一行的第一个球.pos.x - 2 * radius;
Sprite3D* ballSprite;
btRigidBody* ballBody;
int curNumber = -1;
float offsetZ = -7.f;
Vec3 ballPos;
const Vec3 dir = Vec3(-0.57f, 0.f, -0.987269f);
for (int i=0; i<5; ++i)
{
ballPos = dir * i;
ballPos.x += -1.14f;
ballPos.z += offsetZ;
for (int j=0; j<=i; ++j)
{
curNumber++; // 第几个球
ballPos.x += 1.14f; // 每行第k个都是上一个球的X+ 2 * radius
ballSprite = Sprite3D::create("ball/ball.c3b", StringUtils::format("ball/ball_%d.png", BALLS_NUMBER[curNumber]));
this->addChild(ballSprite);
ballSprite->setPosition3D(ballPos);
ballSprite->setCameraMask(2);
ballBody = _world->addSphere(0.57f, btVector3(ballPos.x, ballPos.y, ballPos.z), PhysicsMaterial3D(4.2f, 0.2f, 0.9f, 0.15f));
ballBody->setUserPointer(ballSprite);
_balls.push_back(ballBody);
}
}
看上面代码
Sprite3D::create("ball/ball.c3b",
StringUtils::format("ball/ball_%d.png", BALLS_NUMBER[curNumber]));
根据提前的设计加载相应的球号。
ballPos = dir * i;
ballPos.x += -1.14f;
ballPos.z += offsetZ;
设置每行第一个球的位置
最后就是加载白球,白球要特别独立出来
// white ball
ballSprite = Sprite3D::create("ball/ball.c3b", "ball/ball_white.png");
this->addChild(ballSprite);
ballSprite->setPosition3D(Vec3(0.f, 0.f, 5.f));
ballSprite->setCameraMask(2);
_whiteBallBody = _world->addSphere(0.57f, btVector3(0.f, 0.f, 5.f), PhysicsMaterial3D(4.2f, 0.2f, 0.9f, 0.15f));
_whiteBallBody->setUserPointer(ballSprite);
4.更新物理世界
_world->update(delta);
float m[16];
for (auto ballBody : _balls)
{
ballBody->getWorldTransform().getOpenGLMatrix(m);
static_cast(ballBody->getUserPointer())->setNodeToParentTransform(Mat4(m));
}
_whiteBallBody->getWorldTransform().getOpenGLMatrix(m);
static_cast(_whiteBallBody->getUserPointer())->setNodeToParentTransform(Mat4(m));
每一帧都去更新实际上是很浪费资源的,当所有的球都不动时,其实没必要更新,但是只有不到20个球,
性能不会影响,当游戏中出现大量的物体时,就要重载btMotionState,这个以后讨论。
5.测试一下
当点击屏幕是给白球施加一个冲量,记住一定要先唤醒物体,不然不会有效果的
_whiteBallBody->setActivationState(ACTIVE_TAG);
_whiteBallBody->applyCentralImpulse(btVector3(0.f, 0.f, -60.5f));
总结:
不是美工,模型什么的设计很费劲,贴图都是网上找的。
台球桌,台球的物理属性,调整麻烦,目前调整的还不好
对于添加的Sprite3D一定要设置CameraMask不然是不会被看到的。
添加光照,使物体具有立体感
// light
auto light = SpotLight::create(Vec3(0, -1.f, 0.f), Vec3(0.f, 0.f, 0.f), Color3B::WHITE, 0.f, 0.5f, 1000.f);
light->setPosition3D(Vec3(0.f, 100.f, 0.f));
this->addChild(light);
light->setCameraMask(2);
源码下载
Bullet库的设置方法请参考http://blog.csdn.net/ctxdecs/article/details/42045099