1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 15:35:23 +00:00

345 lines
10 KiB
C++
Raw Normal View History

2011-02-22 14:02:50 +01:00
#ifndef OENGINE_BULLET_PHYSIC_H
#define OENGINE_BULLET_PHYSIC_H
2011-02-26 16:34:43 +01:00
#include <BulletDynamics/Dynamics/btRigidBody.h>
2011-02-22 14:02:50 +01:00
#include "BulletCollision/CollisionDispatch/btGhostObject.h"
#include <string>
#include <list>
#include <map>
#include "BulletShapeLoader.h"
2012-06-17 20:56:10 -04:00
#include "BulletCollision/CollisionShapes/btScaledBvhTriangleMeshShape.h"
2011-02-22 14:02:50 +01:00
2012-09-15 18:17:42 -04:00
2011-02-22 14:02:50 +01:00
class btRigidBody;
class btBroadphaseInterface;
class btDefaultCollisionConfiguration;
class btSequentialImpulseConstraintSolver;
class btCollisionDispatcher;
class btDiscreteDynamicsWorld;
2012-06-06 21:08:20 +02:00
class btHeightfieldTerrainShape;
2011-02-22 14:02:50 +01:00
namespace BtOgre
{
class DebugDrawer;
2011-02-22 14:02:50 +01:00
}
2013-02-19 04:03:24 +01:00
namespace Ogre
{
class SceneManager;
}
2011-02-22 14:02:50 +01:00
namespace MWWorld
{
class World;
2011-02-22 14:02:50 +01:00
}
2012-09-15 18:17:42 -04:00
2011-02-22 14:02:50 +01:00
namespace OEngine {
namespace Physic
{
class CMotionState;
struct PhysicEvent;
2012-09-03 20:32:20 -04:00
class PhysicEngine;
class RigidBody;
2011-02-22 14:02:50 +01:00
2011-02-22 20:53:02 +01:00
/**
2011-02-26 16:34:43 +01:00
*This is just used to be able to name objects.
2011-02-22 20:53:02 +01:00
*/
class PairCachingGhostObject : public btPairCachingGhostObject
{
public:
PairCachingGhostObject(std::string name)
:btPairCachingGhostObject(),mName(name)
{
}
virtual ~PairCachingGhostObject(){}
2011-02-22 20:53:02 +01:00
std::string mName;
};
/**
2012-09-15 16:45:50 -04:00
* A physic actor uses a rigid body based on box shapes.
* Pmove is used to move the physic actor around the dynamic world.
*/
class PhysicActor
{
public:
PhysicActor(const std::string &name, const std::string &mesh, PhysicEngine *engine, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation, float scale);
2011-02-22 14:02:50 +01:00
~PhysicActor();
2011-02-22 14:02:50 +01:00
2012-09-15 13:23:49 -04:00
/**
* This adjusts the rotation of a PhysicActor
* If we have any problems with this (getting stuck in pmove) we should change it
* from setting the visual orientation to setting the orientation of the rigid body directly.
*/
void setRotation(const Ogre::Quaternion &quat);
void enableCollisions(bool collision);
bool getCollisionMode();
2012-09-15 16:45:50 -04:00
/**
* This returns the visual position of the PhysicActor (used to position a scenenode).
* Note - this is different from the position of the contained mBody.
*/
2012-09-17 17:23:26 -04:00
Ogre::Vector3 getPosition();
2011-02-22 14:02:50 +01:00
2012-09-15 16:45:50 -04:00
/**
* Returns the visual orientation of the PhysicActor
*/
2012-09-17 17:23:26 -04:00
Ogre::Quaternion getRotation();
2011-02-22 14:02:50 +01:00
2012-09-15 16:45:50 -04:00
/**
* Sets the scale of the PhysicActor
*/
2012-09-12 18:30:32 -04:00
void setScale(float scale);
2011-02-22 14:02:50 +01:00
/**
* Returns the half extents for this PhysiActor
*/
Ogre::Vector3 getHalfExtents() const;
/**
* Sets the current amount of vertical force (gravity) affecting this physic actor
*/
void setVerticalForce(float force);
/**
* Gets the current amount of vertical force (gravity) affecting this physic actor
*/
float getVerticalForce() const;
void setOnGround(bool grounded);
bool getOnGround() const;
//HACK: in Visual Studio 2010 and presumably above, this structures alignment
// must be 16, but the built in operator new & delete don't properly
// perform this alignment.
#if _MSC_VER >= 1600
void * operator new (size_t Size) { return _aligned_malloc (Size, 16); }
void operator delete (void * Data) { _aligned_free (Data); }
#endif
2012-09-03 20:32:20 -04:00
private:
2012-09-15 18:17:42 -04:00
2012-09-03 20:32:20 -04:00
OEngine::Physic::RigidBody* mBody;
Ogre::Vector3 mBoxScaledTranslation;
btQuaternion mBoxRotationInverse;
2012-09-15 13:23:49 -04:00
Ogre::Quaternion mBoxRotation;
float verticalForce;
bool onGround;
2012-09-05 17:44:11 -04:00
bool collisionMode;
2012-09-03 20:32:20 -04:00
std::string mMesh;
PhysicEngine* mEngine;
2012-09-15 16:45:50 -04:00
std::string mName;
};
2011-02-22 14:02:50 +01:00
/**
*This class is just an extension of normal btRigidBody in order to add extra info.
*When bullet give back a btRigidBody, you can just do a static_cast to RigidBody,
*so one never should use btRigidBody directly!
*/
class RigidBody: public btRigidBody
{
public:
RigidBody(btRigidBody::btRigidBodyConstructionInfo& CI,std::string name);
virtual ~RigidBody();
std::string mName;
2011-02-22 20:53:02 +01:00
//is this body used for raycasting only?
bool mCollide;
bool mIgnore;
};
2012-06-06 21:08:20 +02:00
struct HeightField
{
btHeightfieldTerrainShape* mShape;
RigidBody* mBody;
};
/**
* The PhysicEngine class contain everything which is needed for Physic.
* It's needed that Ogre Resources are set up before the PhysicEngine is created.
* Note:deleting it WILL NOT delete the RigidBody!
* TODO:unload unused resources?
*/
class PhysicEngine
{
public:
/**
* Note that the shapeLoader IS destroyed by the phyic Engine!!
*/
PhysicEngine(BulletShapeLoader* shapeLoader);
/**
* It DOES destroy the shape loader!
*/
~PhysicEngine();
/**
2012-09-05 17:44:11 -04:00
* Creates a RigidBody. It does not add it to the simulation.
* After created, the body is set to the correct rotation, position, and scale
*/
RigidBody* createAndAdjustRigidBody(const std::string &mesh, const std::string &name,
float scale, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation,
2012-09-15 13:23:49 -04:00
Ogre::Vector3* scaledBoxTranslation = 0, Ogre::Quaternion* boxRotation = 0);
/**
* Adjusts a rigid body to the right position and rotation
*/
void adjustRigidBody(RigidBody* body, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation,
const Ogre::Vector3 &scaledBoxTranslation = Ogre::Vector3::ZERO,
const Ogre::Quaternion &boxRotation = Ogre::Quaternion::IDENTITY);
/**
Mainly used to (but not limited to) adjust rigid bodies based on box shapes to the right position and rotation.
*/
void boxAdjustExternal(const std::string &mesh, RigidBody* body, float scale, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation);
/**
* Add a HeightField to the simulation
*/
void addHeightField(float* heights,
int x, int y, float yoffset,
float triSize, float sqrtVerts);
/**
* Remove a HeightField from the simulation
*/
void removeHeightField(int x, int y);
/**
* Add a RigidBody to the simulation
*/
2012-09-05 17:44:11 -04:00
void addRigidBody(RigidBody* body, bool addToMap = true);
/**
* Remove a RigidBody from the simulation. It does not delete it, and does not remove it from the RigidBodyMap.
*/
void removeRigidBody(const std::string &name);
/**
* Delete a RigidBody, and remove it from RigidBodyMap.
*/
void deleteRigidBody(const std::string &name);
/**
* Return a pointer to a given rigid body.
* TODO:check if exist
*/
RigidBody* getRigidBody(const std::string &name);
/**
* Create and add a character to the scene, and add it to the ActorMap.
*/
void addCharacter(const std::string &name, const std::string &mesh,
const Ogre::Vector3 &position, float scale, const Ogre::Quaternion &rotation);
/**
* Remove a character from the scene. TODO:delete it! for now, a small memory leak^^ done?
*/
void removeCharacter(const std::string &name);
/**
* Return a pointer to a character
* TODO:check if the actor exist...
*/
PhysicActor* getCharacter(const std::string &name);
/**
* This step the simulation of a given time.
*/
void stepSimulation(double deltaT);
/**
* Empty events lists
*/
void emptyEventLists(void);
/**
* Create a debug rendering. It is called by setDebgRenderingMode if it's not created yet.
* Important Note: this will crash if the Render is not yet initialise!
*/
void createDebugRendering();
/**
* Set the debug rendering mode. 0 to turn it off.
* Important Note: this will crash if the Render is not yet initialise!
*/
void setDebugRenderingMode(int mode);
bool toggleDebugRendering();
void getObjectAABB(const std::string &mesh, float scale, btVector3 &min, btVector3 &max);
2013-02-19 04:03:24 +01:00
void setSceneManager(Ogre::SceneManager* sceneMgr);
/**
* Return the closest object hit by a ray. If there are no objects, it will return ("",-1).
*/
2011-02-22 20:53:02 +01:00
std::pair<std::string,float> rayTest(btVector3& from,btVector3& to);
/**
* Return all objects hit by a ray.
*/
std::vector< std::pair<float, std::string> > rayTest2(btVector3& from, btVector3& to);
//event list of non player object
std::list<PhysicEvent> NPEventList;
2011-02-22 14:02:50 +01:00
//event list affecting the player
std::list<PhysicEvent> PEventList;
2011-02-22 14:02:50 +01:00
//Bullet Stuff
btOverlappingPairCache* pairCache;
btBroadphaseInterface* broadphase;
btDefaultCollisionConfiguration* collisionConfiguration;
btSequentialImpulseConstraintSolver* solver;
btCollisionDispatcher* dispatcher;
btDiscreteDynamicsWorld* dynamicsWorld;
2011-02-22 14:02:50 +01:00
//the NIF file loader.
BulletShapeLoader* mShapeLoader;
2011-02-22 14:02:50 +01:00
2012-06-06 21:08:20 +02:00
typedef std::map<std::string, HeightField> HeightFieldContainer;
HeightFieldContainer mHeightFieldMap;
typedef std::map<std::string,RigidBody*> RigidBodyContainer;
2012-09-15 18:17:42 -04:00
RigidBodyContainer ObjectMap;
typedef std::map<std::string, PhysicActor*> PhysicActorContainer;
PhysicActorContainer PhysicActorMap;
2011-02-22 14:02:50 +01:00
2013-02-19 04:03:24 +01:00
Ogre::SceneManager* mSceneMgr;
//debug rendering
BtOgre::DebugDrawer* mDebugDrawer;
bool isDebugCreated;
bool mDebugActive;
};
2011-02-22 14:02:50 +01:00
struct MyRayResultCallback : public btCollisionWorld::RayResultCallback
{
virtual btScalar addSingleResult( btCollisionWorld::LocalRayResult& rayResult, bool bNormalInWorldSpace)
{
results.push_back( std::make_pair(rayResult.m_hitFraction, rayResult.m_collisionObject) );
return rayResult.m_hitFraction;
}
2012-03-25 22:53:00 +02:00
static bool cmp( const std::pair<float, std::string>& i, const std::pair<float, std::string>& j )
{
2012-03-25 20:52:56 +02:00
if( i.first > j.first ) return false;
if( j.first > i.first ) return true;
return false;
}
std::vector < std::pair<float, const btCollisionObject*> > results;
};
2011-02-22 14:02:50 +01:00
}}
2011-02-26 16:34:43 +01:00
#endif