public:t-gede-14-1:lab9
This is an old revision of the document!
Table of Contents
LAB9: Particle Systems (Optional)
In this lab we will scratch the surface of using the Bullet physics blibrary with Ogre. This lab is intirely optional but can replace another previously unfinished lab, and counts toward participation.
Unfortunately I was unable to give me the proper amount of time to make this lab interesting enough, but the classic brick wall
Discussion
Discussion thread for this lab is on Piazza
Preperation
You can use your own project or start fresh with the base application code
Lab
- Setup Project.
- Downlaod Bullet
- Build Bullet
- Run <bullet root>\build\vs2010.bat to generate the VS 2010 project files.
- Open the <bullet root>\build\vs2010\0BulletSolution.sln With you VS of choice.
- Make sure you compile using the v100 version of the visual studio compiler
Note: that you can select properties for one project, then multi select the projects you want to change the properties for to change the value in multiple projects at the same time. - Change the C/C++→Code Generation→Runtime Library to Multi Threaded DLL (/MDd (Debug) and /MD (Release))
- Build the solution for debug and release.
- Add
<Bullet Root>\lib;
toLinker→General→Additional Library Directories
(Debug and Release) - Add
<Bullet Root>\src;
toC++→General→Additional Include Directories
(Debug and Release) - Add
BulletCollision_vs2010_debug.lib;BulletDynamics_vs2010_debug.lib;LinearMath_vs2010_debug.lib
to theLinker→Input→Additional Dependencies
(Debug) - Add
BulletCollision_vs2010.lib;BulletDynamics_vs2010.lib;LinearMath_vs2010.lib
to theLinker→Input→Additional Dependencies
(Release) - Create class Physics.
// Header! #ifndef PHYSICS_H #define PHYSICS_H #include "btBulletCollisionCommon.h" #include "btBulletDynamicsCommon.h" #include <vector> #include <map> class Physics{ btDefaultCollisionConfiguration* collisionConfiguration; btCollisionDispatcher* dispatcher; btBroadphaseInterface* overlappingPairCache; btSequentialImpulseConstraintSolver* solver; btDiscreteDynamicsWorld* dynamicsWorld; std::vector<btCollisionShape *> collisionShapes; std::map<std::string, btRigidBody *> physicsAccessors; public: Physics(); virtual ~Physics(); btDiscreteDynamicsWorld* getDynamicsWorld(); }; #endif //PHYSICS_H // Implementation! #include "Physics.h" Physics::~Physics() { delete dynamicsWorld; delete solver; delete overlappingPairCache; delete dispatcher; delete collisionConfiguration; } void Physics::initObjects() { collisionConfiguration = new btDefaultCollisionConfiguration(); dispatcher = new btCollisionDispatcher(collisionConfiguration); overlappingPairCache = new btDbvtBroadphase(); solver = new btSequentialImpulseConstraintSolver(); dynamicsWorld = new btDiscreteDynamicsWorld(dispatcher, overlappingPairCache, solver, collisionConfiguration); } btDiscreteDynamicsWorld* Physics::getDynamicsWorld() { return dynamicsWorld; }
- Create a pointer to physics in your Application, and initialize it in the startup function.
- Before we start there are some basic changes that have to made to the environment, in order to make the physics look more realist, we must make sure that our environmet is in a logical scale. If you have the handout code, Sinbad will be around 9.5 units high, Bullet library uses 1 unit as equal to 1 meter.
- You can check out Sinbad's actualt height by doing
_SinbadNode→getBoundingBox().getSize()
, this function will return the bounding box the the sceneNode, which is an axis aligned box that encompasses the entities of the SceneNode and is defined by the vertex extremas + an additional 2% scale. - Now scale the sinbad node down by 0.2
_SinbadNode→scale(Ogre::Vector3(0.2f,0.2f,0.2f));
- Change the nearClippingPlaneDistance of the camera to 0.5
- In the frameListener class, change the movement speed of the camera to something around 10-20.
- Now to create Rigid Bodies we must first let bullet know what form of rigid body we want, and we must also tell bullet the size, position and orientation of our RigidBody.
- Add this function to your Physics class.
btRigidBody* AddDynamicCubeRigidBoidy(Ogre::SceneNode* node, Ogre::Entity* ent, btScalar mass) btRigidBody* Physics::AddDynamicCubeRigidBoidy(Ogre::SceneNode* node, Ogre::Entity* ent, btScalar mass) { Ogre::Vector3 aabbSize = ent->getBoundingBox().getSize() / 1.02; btScalar x = 0.5 * node->getScale().x * aabbSize.x; btScalar y = 0.5 * node->getScale().y * aabbSize.y; btScalar z = 0.5 * node->getScale().z * aabbSize.z; btCollisionShape* colShape = new btBoxShape(btVector3(x, y, z)); /// Create Dynamic Objects btTransform startTransform; startTransform.setIdentity(); //rigidbody is dynamic if and only if mass is non zero, otherwise static bool isDynamic = (mass != 0.f); btVector3 localInertia(0, 0, 0); if (isDynamic) colShape->calculateLocalInertia(mass, localInertia); startTransform.setOrigin(btVector3(node->getPosition().x, node->getPosition().y, node->getPosition().z)); btQuaternion initRotation(node->getOrientation().x, node->getOrientation().y, node->getOrientation().z, node->getOrientation().w); startTransform.setRotation(initRotation); OgreMotionState* motionState = new OgreMotionState(startTransform, node); btRigidBody::btRigidBodyConstructionInfo rbInfo(mass, motionState, colShape, localInertia); btRigidBody* body = new btRigidBody(rbInfo); dynamicsWorld->addRigidBody(body); return body; }
- As you might have noticed, that function uses a class called OgreMotionState, this is a class that we must implement in order for Bullet to automatically transoform our SceneNode according to the movement of the physics object.
- Add the file OgreMotionState.h to your project.
#ifndef OGREMOTIONSTATE_H #define OGREMOTIONSTATE_H #include "LinearMath\btMotionState.h" #include "OGRE\Ogre.h" class OgreMotionState : public btMotionState { public: OgreMotionState(const btTransform &initialpos, Ogre::SceneNode *node) { mVisibleobj = node; mPos1 = initialpos; } virtual ~OgreMotionState() { } void setNode(Ogre::SceneNode *node) { mVisibleobj = node; } virtual void getWorldTransform(btTransform &worldTrans) const { worldTrans = mPos1; } virtual void setWorldTransform(const btTransform &worldTrans) { if (NULL == mVisibleobj) return; btQuaternion rot = worldTrans.getRotation(); mVisibleobj->setOrientation(rot.w(), rot.x(), rot.y(), rot.z()); btVector3 pos = worldTrans.getOrigin(); mVisibleobj->setPosition(pos.x(), pos.y(), pos.z()); } protected: Ogre::SceneNode *mVisibleobj; btTransform mPos1; }; #endif //OGREMOTIONSTATE_H
When You Are Finished
Upload your commented source files into Lab9 in MySchool (zip them up if more than one).
/var/www/cadia.ru.is/wiki/data/attic/public/t-gede-14-1/lab9.1395697839.txt.gz · Last modified: 2024/04/29 13:32 (external edit)