#include "physic.hpp" #include #include #include //#include #include "CMotionState.h" #include "OgreRoot.h" #include "btKinematicCharacterController.h" #include "BtOgrePG.h" #include "BtOgreGP.h" #include "BtOgreExtras.h" #define BIT(x) (1<<(x)) namespace OEngine { namespace Physic { enum collisiontypes { COL_NOTHING = 0, //setWorldTransform( transform ); btScalar externalCapsuleHeight = 130; btScalar externalCapsuleWidth = 16; externalCollisionShape = new btCapsuleShapeZ( externalCapsuleWidth, externalCapsuleHeight ); externalCollisionShape->setMargin( 0.05 ); externalGhostObject->setCollisionShape( externalCollisionShape ); externalGhostObject->setCollisionFlags( btCollisionObject::CF_CHARACTER_OBJECT ); // Internal capsule internalGhostObject = new PairCachingGhostObject(name); internalGhostObject->setWorldTransform( transform ); //internalGhostObject->getBroadphaseHandle()->s btScalar internalCapsuleHeight = 120; btScalar internalCapsuleWidth = 15; internalCollisionShape = new btCapsuleShapeZ( internalCapsuleWidth, internalCapsuleHeight ); internalCollisionShape->setMargin( 0.05 ); internalGhostObject->setCollisionShape( internalCollisionShape ); internalGhostObject->setCollisionFlags( btCollisionObject::CF_CHARACTER_OBJECT ); mCharacter = new btKinematicCharacterController( externalGhostObject,internalGhostObject,btScalar( 10 ),1,9.8,20,9.8,0.2 ); mCharacter->setUpAxis(btKinematicCharacterController::Z_AXIS); mCharacter->setUseGhostSweepTest(false); mCharacter->mCollision = false; setGravity(0); } PhysicActor::~PhysicActor() { delete mCharacter; delete internalGhostObject; delete internalCollisionShape; delete externalGhostObject; delete externalCollisionShape; } void PhysicActor::setGravity(float gravity) { mCharacter->setGravity(gravity); } void PhysicActor::enableCollisions(bool collision) { mCharacter->mCollision = collision; } void PhysicActor::setWalkDirection(const btVector3& mvt) { mCharacter->setWalkDirection( mvt ); } void PhysicActor::Rotate(const btQuaternion& quat) { externalGhostObject->getWorldTransform().setRotation( externalGhostObject->getWorldTransform().getRotation() * quat ); internalGhostObject->getWorldTransform().setRotation( internalGhostObject->getWorldTransform().getRotation() * quat ); } void PhysicActor::setRotation(const btQuaternion& quat) { externalGhostObject->getWorldTransform().setRotation( quat ); internalGhostObject->getWorldTransform().setRotation( quat ); } btVector3 PhysicActor::getPosition(void) { return internalGhostObject->getWorldTransform().getOrigin(); } btQuaternion PhysicActor::getRotation(void) { return internalGhostObject->getWorldTransform().getRotation(); } void PhysicActor::setPosition(const btVector3& pos) { internalGhostObject->getWorldTransform().setOrigin(pos); externalGhostObject->getWorldTransform().setOrigin(pos); } //////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////// RigidBody::RigidBody(btRigidBody::btRigidBodyConstructionInfo& CI,std::string name) :btRigidBody(CI),mName(name) { }; /////////////////////////////////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////////////////////////////// PhysicEngine::PhysicEngine() { // Set up the collision configuration and dispatcher collisionConfiguration = new btDefaultCollisionConfiguration(); dispatcher = new btCollisionDispatcher(collisionConfiguration); // The actual physics solver solver = new btSequentialImpulseConstraintSolver; //TODO: memory leak? btOverlappingPairCache* pairCache = new btSortedOverlappingPairCache(); pairCache->setInternalGhostPairCallback( new btGhostPairCallback() ); broadphase = new btDbvtBroadphase(pairCache); // The world. dynamicsWorld = new btDiscreteDynamicsWorld(dispatcher,broadphase,solver,collisionConfiguration); dynamicsWorld->setGravity(btVector3(0,0,-10)); if(BulletShapeManager::getSingletonPtr() == NULL) { new BulletShapeManager(); } //TODO:singleton? ShapeLoader = new ManualBulletShapeLoader(); isDebugCreated = false; } void PhysicEngine::createDebugRendering() { if(!isDebugCreated) { Ogre::SceneManagerEnumerator::SceneManagerIterator iter = Ogre::Root::getSingleton().getSceneManagerIterator(); iter.begin(); Ogre::SceneManager* scn = iter.getNext(); Ogre::SceneNode* node = scn->getRootSceneNode()->createChildSceneNode(); node->pitch(Ogre::Degree(-90)); mDebugDrawer = new BtOgre::DebugDrawer(node, dynamicsWorld); dynamicsWorld->setDebugDrawer(mDebugDrawer); isDebugCreated = true; dynamicsWorld->debugDrawWorld(); } } void PhysicEngine::setDebugRenderingMode(int mode) { if(!isDebugCreated) { createDebugRendering(); } mDebugDrawer->setDebugMode(mode); } PhysicEngine::~PhysicEngine() { delete dynamicsWorld; delete solver; delete collisionConfiguration; delete dispatcher; delete broadphase; delete ShapeLoader; } RigidBody* PhysicEngine::createRigidBody(std::string mesh,std::string name) { //get the shape from the .nif ShapeLoader->load(mesh,"General"); BulletShapeManager::getSingletonPtr()->load(mesh,"General"); BulletShapePtr shape = BulletShapeManager::getSingleton().getByName(mesh,"General"); //create the motionState CMotionState* newMotionState = new CMotionState(this,name); //create the real body btRigidBody::btRigidBodyConstructionInfo CI = btRigidBody::btRigidBodyConstructionInfo(0,newMotionState,shape->Shape); RigidBody* body = new RigidBody(CI,name); body->collide = shape->collide; return body; } void PhysicEngine::addRigidBody(RigidBody* body) { if(body->collide) { dynamicsWorld->addRigidBody(body,COL_WORLD,COL_WORLD|COL_ACTOR_INTERNAL|COL_ACTOR_EXTERNAL); } else { dynamicsWorld->addRigidBody(body,COL_WORLD,COL_NOTHING); } body->setActivationState(DISABLE_DEACTIVATION); RigidBodyMap[body->mName] = body; } void PhysicEngine::removeRigidBody(std::string name) { RigidBody* body = RigidBodyMap[name]; if(body != NULL) { broadphase->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher); std::map::iterator it = PhysicActorMap.begin(); for(;it!=PhysicActorMap.end();it++) { it->second->internalGhostObject->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher); it->second->externalGhostObject->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher); } dynamicsWorld->removeRigidBody(RigidBodyMap[name]); } } void PhysicEngine::deleteRigidBody(std::string name) { RigidBody* body = RigidBodyMap[name]; if(body != NULL) { delete body; RigidBodyMap[name] = NULL; } } RigidBody* PhysicEngine::getRigidBody(std::string name) { RigidBody* body = RigidBodyMap[name]; return body; } void PhysicEngine::stepSimulation(double deltaT) { dynamicsWorld->stepSimulation(deltaT,1,1/30.); if(isDebugCreated) { mDebugDrawer->step(); } } void PhysicEngine::addCharacter(std::string name) { PhysicActor* newActor = new PhysicActor(name); dynamicsWorld->addCollisionObject( newActor->externalGhostObject, COL_ACTOR_EXTERNAL, COL_WORLD |COL_ACTOR_EXTERNAL ); dynamicsWorld->addCollisionObject( newActor->internalGhostObject, COL_ACTOR_INTERNAL, COL_WORLD |COL_ACTOR_INTERNAL ); dynamicsWorld->addAction( newActor->mCharacter ); PhysicActorMap[name] = newActor; } void PhysicEngine::removeCharacter(std::string name) { PhysicActor* act = PhysicActorMap[name]; if(act != NULL) { dynamicsWorld->removeCollisionObject(act->externalGhostObject); dynamicsWorld->removeCollisionObject(act->internalGhostObject); dynamicsWorld->removeAction(act->mCharacter); delete act; PhysicActorMap[name] = NULL; } } PhysicActor* PhysicEngine::getCharacter(std::string name) { PhysicActor* act = PhysicActorMap[name]; return act; } void PhysicEngine::emptyEventLists(void) { } std::pair PhysicEngine::rayTest(btVector3& from,btVector3& to) { std::string name = ""; float d = -1.; btCollisionWorld::ClosestRayResultCallback resultCallback(from, to); dynamicsWorld->rayTest(from, to, resultCallback); if (resultCallback.hasHit()) { if(resultCallback.m_collisionFilterGroup == COL_WORLD) { name = dynamic_cast(*resultCallback.m_collisionObject).mName; } if(resultCallback.m_collisionFilterGroup == COL_ACTOR_EXTERNAL || resultCallback.m_collisionFilterGroup == COL_ACTOR_INTERNAL) { name = dynamic_cast(*resultCallback.m_collisionObject).mName; } d = resultCallback.m_closestHitFraction; } return std::pair(name,d); } }};