2011-02-22 14:02:50 +01:00
|
|
|
#include "physic.hpp"
|
|
|
|
#include <btBulletDynamicsCommon.h>
|
|
|
|
#include <btBulletCollisionCommon.h>
|
2012-03-13 17:09:50 +01:00
|
|
|
#include <BulletCollision/CollisionShapes/btHeightfieldTerrainShape.h>
|
2013-02-24 13:52:23 -08:00
|
|
|
#include <components/nifbullet/bulletnifloader.hpp>
|
2011-02-22 14:02:50 +01:00
|
|
|
#include "CMotionState.h"
|
|
|
|
#include "OgreRoot.h"
|
|
|
|
#include "btKinematicCharacterController.h"
|
|
|
|
#include "BtOgrePG.h"
|
|
|
|
#include "BtOgreGP.h"
|
|
|
|
#include "BtOgreExtras.h"
|
|
|
|
|
2012-03-13 17:09:50 +01:00
|
|
|
#include <boost/lexical_cast.hpp>
|
2012-07-26 19:38:33 +04:00
|
|
|
#include <boost/format.hpp>
|
2012-03-13 17:09:50 +01:00
|
|
|
|
2011-02-22 14:02:50 +01:00
|
|
|
namespace OEngine {
|
|
|
|
namespace Physic
|
|
|
|
{
|
2011-04-03 13:12:12 +02:00
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
PhysicActor::PhysicActor(const std::string &name, const std::string &mesh, PhysicEngine *engine, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation, float scale)
|
|
|
|
: mName(name), mEngine(engine), mMesh(mesh), mBoxScaledTranslation(0,0,0), mBoxRotationInverse(0,0,0,0)
|
2013-04-11 18:02:44 +02:00
|
|
|
, mBody(0), mRaycastingBody(0), onGround(false), collisionMode(true), mBoxRotation(0,0,0,0), verticalForce(0.0f)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2012-09-15 13:23:49 -04:00
|
|
|
mBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, position, rotation, &mBoxScaledTranslation, &mBoxRotation);
|
2013-03-08 23:46:25 +01:00
|
|
|
mRaycastingBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, position, rotation, &mBoxScaledTranslation, &mBoxRotation, true);
|
2012-09-15 13:23:49 -04:00
|
|
|
Ogre::Quaternion inverse = mBoxRotation.Inverse();
|
|
|
|
mBoxRotationInverse = btQuaternion(inverse.x, inverse.y, inverse.z,inverse.w);
|
2013-05-07 19:35:10 +01:00
|
|
|
mEngine->addRigidBody(mBody, false, mRaycastingBody,true); //Add rigid body to dynamics world, but do not add to object map
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
PhysicActor::~PhysicActor()
|
|
|
|
{
|
2013-02-07 16:18:29 -08:00
|
|
|
if(mBody)
|
|
|
|
{
|
2012-09-03 20:32:20 -04:00
|
|
|
mEngine->dynamicsWorld->removeRigidBody(mBody);
|
|
|
|
delete mBody;
|
|
|
|
}
|
2013-03-08 23:46:25 +01:00
|
|
|
if(mRaycastingBody)
|
|
|
|
{
|
|
|
|
mEngine->dynamicsWorld->removeRigidBody(mRaycastingBody);
|
|
|
|
delete mRaycastingBody;
|
|
|
|
}
|
2012-09-16 17:29:16 -04:00
|
|
|
}
|
2011-03-17 11:00:46 +01:00
|
|
|
|
|
|
|
void PhysicActor::enableCollisions(bool collision)
|
|
|
|
{
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2013-07-30 22:00:48 +02:00
|
|
|
if(collision && !collisionMode) enableCollisionBody();
|
|
|
|
if(!collision && collisionMode) disableCollisionBody();
|
2012-09-05 17:44:11 -04:00
|
|
|
collisionMode = collision;
|
2011-03-20 21:51:40 +01:00
|
|
|
}
|
|
|
|
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-19 04:18:15 -08:00
|
|
|
void PhysicActor::setPosition(const Ogre::Vector3 &pos)
|
|
|
|
{
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2013-02-19 04:18:15 -08:00
|
|
|
if(pos != getPosition())
|
2013-03-08 23:46:25 +01:00
|
|
|
{
|
2013-02-19 04:18:15 -08:00
|
|
|
mEngine->adjustRigidBody(mBody, pos, getRotation(), mBoxScaledTranslation, mBoxRotation);
|
2013-03-08 23:46:25 +01:00
|
|
|
mEngine->adjustRigidBody(mRaycastingBody, pos, getRotation(), mBoxScaledTranslation, mBoxRotation);
|
|
|
|
}
|
2013-02-19 04:18:15 -08:00
|
|
|
}
|
|
|
|
|
2013-02-07 15:48:41 -08:00
|
|
|
void PhysicActor::setRotation(const Ogre::Quaternion &quat)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2012-09-15 13:23:49 -04:00
|
|
|
if(!quat.equals(getRotation(), Ogre::Radian(0))){
|
|
|
|
mEngine->adjustRigidBody(mBody, getPosition(), quat, mBoxScaledTranslation, mBoxRotation);
|
2013-03-08 23:46:25 +01:00
|
|
|
mEngine->adjustRigidBody(mRaycastingBody, getPosition(), quat, mBoxScaledTranslation, mBoxRotation);
|
|
|
|
|
2012-09-15 13:23:49 -04:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-19 04:18:15 -08:00
|
|
|
|
2012-09-17 17:23:26 -04:00
|
|
|
Ogre::Vector3 PhysicActor::getPosition()
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2012-09-10 20:29:24 -04:00
|
|
|
btVector3 vec = mBody->getWorldTransform().getOrigin();
|
|
|
|
Ogre::Quaternion rotation = Ogre::Quaternion(mBody->getWorldTransform().getRotation().getW(), mBody->getWorldTransform().getRotation().getX(),
|
|
|
|
mBody->getWorldTransform().getRotation().getY(), mBody->getWorldTransform().getRotation().getZ());
|
|
|
|
Ogre::Vector3 transrot = rotation * mBoxScaledTranslation;
|
2012-09-12 18:30:32 -04:00
|
|
|
Ogre::Vector3 visualPosition = Ogre::Vector3(vec.getX(), vec.getY(), vec.getZ()) - transrot;
|
2012-09-10 20:29:24 -04:00
|
|
|
return visualPosition;
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2012-09-17 17:23:26 -04:00
|
|
|
Ogre::Quaternion PhysicActor::getRotation()
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2012-09-12 18:30:32 -04:00
|
|
|
btQuaternion quat = mBody->getWorldTransform().getRotation() * mBoxRotationInverse;
|
|
|
|
return Ogre::Quaternion(quat.getW(), quat.getX(), quat.getY(), quat.getZ());
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2012-09-12 18:30:32 -04:00
|
|
|
void PhysicActor::setScale(float scale){
|
|
|
|
//We only need to change the scaled box translation, box rotations remain the same.
|
2013-04-11 18:02:44 +02:00
|
|
|
assert(mBody);
|
2012-09-12 18:30:32 -04:00
|
|
|
mBoxScaledTranslation = mBoxScaledTranslation / mBody->getCollisionShape()->getLocalScaling().getX();
|
|
|
|
mBoxScaledTranslation *= scale;
|
2013-04-11 18:02:44 +02:00
|
|
|
Ogre::Vector3 pos = getPosition();
|
|
|
|
Ogre::Quaternion rot = getRotation();
|
2012-09-12 18:30:32 -04:00
|
|
|
if(mBody){
|
|
|
|
mEngine->dynamicsWorld->removeRigidBody(mBody);
|
2013-03-31 13:50:57 +02:00
|
|
|
mEngine->dynamicsWorld->removeRigidBody(mRaycastingBody);
|
2012-09-12 18:30:32 -04:00
|
|
|
delete mBody;
|
2013-03-31 13:50:57 +02:00
|
|
|
delete mRaycastingBody;
|
2012-09-12 18:30:32 -04:00
|
|
|
}
|
|
|
|
//Create the newly scaled rigid body
|
2013-04-11 18:02:44 +02:00
|
|
|
mBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, pos, rot);
|
|
|
|
mRaycastingBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, pos, rot, 0, 0, true);
|
2013-05-07 19:35:10 +01:00
|
|
|
mEngine->addRigidBody(mBody, false, mRaycastingBody,true); //Add rigid body to dynamics world, but do not add to object map
|
2012-09-12 18:30:32 -04:00
|
|
|
}
|
|
|
|
|
2013-02-05 07:41:57 -08:00
|
|
|
Ogre::Vector3 PhysicActor::getHalfExtents() const
|
|
|
|
{
|
|
|
|
if(mBody)
|
|
|
|
{
|
|
|
|
btBoxShape *box = static_cast<btBoxShape*>(mBody->getCollisionShape());
|
|
|
|
if(box != NULL)
|
|
|
|
{
|
|
|
|
btVector3 size = box->getHalfExtentsWithMargin();
|
|
|
|
return Ogre::Vector3(size.getX(), size.getY(), size.getZ());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Ogre::Vector3(0.0f);
|
|
|
|
}
|
|
|
|
|
2013-02-05 09:24:22 -08:00
|
|
|
void PhysicActor::setVerticalForce(float force)
|
|
|
|
{
|
|
|
|
verticalForce = force;
|
|
|
|
}
|
|
|
|
|
|
|
|
float PhysicActor::getVerticalForce() const
|
|
|
|
{
|
|
|
|
return verticalForce;
|
|
|
|
}
|
|
|
|
|
2013-02-06 21:44:58 -08:00
|
|
|
void PhysicActor::setOnGround(bool grounded)
|
|
|
|
{
|
|
|
|
onGround = grounded;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PhysicActor::getOnGround() const
|
|
|
|
{
|
2013-05-01 16:56:16 +02:00
|
|
|
return collisionMode && onGround;
|
2013-02-06 21:44:58 -08:00
|
|
|
}
|
2013-02-05 09:24:22 -08:00
|
|
|
|
2013-06-27 19:42:27 -07:00
|
|
|
void PhysicActor::disableCollisionBody()
|
|
|
|
{
|
|
|
|
mEngine->dynamicsWorld->removeRigidBody(mBody);
|
|
|
|
}
|
2013-08-03 15:26:53 +02:00
|
|
|
|
2013-06-27 19:42:27 -07:00
|
|
|
void PhysicActor::enableCollisionBody()
|
|
|
|
{
|
2013-07-31 18:46:32 +02:00
|
|
|
mEngine->dynamicsWorld->addRigidBody(mBody,CollisionType_Actor,CollisionType_World|CollisionType_HeightMap);
|
2013-06-27 19:42:27 -07:00
|
|
|
}
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
2011-02-22 14:02:50 +01:00
|
|
|
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
RigidBody::RigidBody(btRigidBody::btRigidBodyConstructionInfo& CI,std::string name)
|
2012-03-25 18:12:00 +02:00
|
|
|
: btRigidBody(CI)
|
|
|
|
, mName(name)
|
2013-03-14 03:04:02 +01:00
|
|
|
, mPlaceable(false)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2012-03-25 18:12:00 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2012-03-25 18:12:00 +02:00
|
|
|
RigidBody::~RigidBody()
|
|
|
|
{
|
|
|
|
delete getMotionState();
|
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
2011-02-22 14:02:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-30 16:02:41 +02:00
|
|
|
PhysicEngine::PhysicEngine(BulletShapeLoader* shapeLoader) :
|
|
|
|
mDebugActive(0)
|
2013-07-31 18:46:32 +02:00
|
|
|
, mSceneMgr(NULL)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
|
|
|
// Set up the collision configuration and dispatcher
|
|
|
|
collisionConfiguration = new btDefaultCollisionConfiguration();
|
|
|
|
dispatcher = new btCollisionDispatcher(collisionConfiguration);
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
// The actual physics solver
|
|
|
|
solver = new btSequentialImpulseConstraintSolver;
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2012-03-24 22:03:08 -04:00
|
|
|
//btOverlappingPairCache* pairCache = new btSortedOverlappingPairCache();
|
2012-03-25 18:12:00 +02:00
|
|
|
pairCache = new btSortedOverlappingPairCache();
|
2012-04-08 17:08:36 +02:00
|
|
|
|
2012-02-06 23:15:19 +01:00
|
|
|
//pairCache->setInternalGhostPairCallback( new btGhostPairCallback() );
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2012-03-24 22:03:08 -04:00
|
|
|
broadphase = new btDbvtBroadphase();
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
// The world.
|
|
|
|
dynamicsWorld = new btDiscreteDynamicsWorld(dispatcher,broadphase,solver,collisionConfiguration);
|
|
|
|
dynamicsWorld->setGravity(btVector3(0,0,-10));
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
if(BulletShapeManager::getSingletonPtr() == NULL)
|
|
|
|
{
|
|
|
|
new BulletShapeManager();
|
|
|
|
}
|
|
|
|
//TODO:singleton?
|
|
|
|
mShapeLoader = shapeLoader;
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
isDebugCreated = false;
|
2012-03-25 16:12:23 +02:00
|
|
|
mDebugDrawer = NULL;
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
void PhysicEngine::createDebugRendering()
|
|
|
|
{
|
|
|
|
if(!isDebugCreated)
|
|
|
|
{
|
2013-02-19 04:03:24 +01:00
|
|
|
Ogre::SceneNode* node = mSceneMgr->getRootSceneNode()->createChildSceneNode();
|
2011-04-03 13:12:12 +02:00
|
|
|
mDebugDrawer = new BtOgre::DebugDrawer(node, dynamicsWorld);
|
|
|
|
dynamicsWorld->setDebugDrawer(mDebugDrawer);
|
|
|
|
isDebugCreated = true;
|
|
|
|
dynamicsWorld->debugDrawWorld();
|
|
|
|
}
|
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-03-18 13:24:47 +01:00
|
|
|
void PhysicEngine::setDebugRenderingMode(int mode)
|
|
|
|
{
|
|
|
|
if(!isDebugCreated)
|
|
|
|
{
|
|
|
|
createDebugRendering();
|
|
|
|
}
|
|
|
|
mDebugDrawer->setDebugMode(mode);
|
2012-03-30 16:02:41 +02:00
|
|
|
mDebugActive = mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PhysicEngine::toggleDebugRendering()
|
|
|
|
{
|
|
|
|
setDebugRenderingMode(!mDebugActive);
|
|
|
|
return mDebugActive;
|
2011-03-18 13:24:47 +01:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-19 04:03:24 +01:00
|
|
|
void PhysicEngine::setSceneManager(Ogre::SceneManager* sceneMgr)
|
|
|
|
{
|
|
|
|
mSceneMgr = sceneMgr;
|
|
|
|
}
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
PhysicEngine::~PhysicEngine()
|
|
|
|
{
|
2012-06-14 21:27:55 +02:00
|
|
|
HeightFieldContainer::iterator hf_it = mHeightFieldMap.begin();
|
|
|
|
for (; hf_it != mHeightFieldMap.end(); ++hf_it)
|
|
|
|
{
|
|
|
|
dynamicsWorld->removeRigidBody(hf_it->second.mBody);
|
|
|
|
delete hf_it->second.mShape;
|
|
|
|
delete hf_it->second.mBody;
|
|
|
|
}
|
|
|
|
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBodyContainer::iterator rb_it = mCollisionObjectMap.begin();
|
|
|
|
for (; rb_it != mCollisionObjectMap.end(); ++rb_it)
|
|
|
|
{
|
|
|
|
if (rb_it->second != NULL)
|
|
|
|
{
|
|
|
|
dynamicsWorld->removeRigidBody(rb_it->second);
|
|
|
|
|
|
|
|
delete rb_it->second;
|
|
|
|
rb_it->second = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rb_it = mRaycastingObjectMap.begin();
|
|
|
|
for (; rb_it != mRaycastingObjectMap.end(); ++rb_it)
|
2012-03-25 16:12:23 +02:00
|
|
|
{
|
|
|
|
if (rb_it->second != NULL)
|
|
|
|
{
|
|
|
|
dynamicsWorld->removeRigidBody(rb_it->second);
|
|
|
|
|
|
|
|
delete rb_it->second;
|
|
|
|
rb_it->second = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-01 11:15:43 +02:00
|
|
|
PhysicActorContainer::iterator pa_it = mActorMap.begin();
|
|
|
|
for (; pa_it != mActorMap.end(); ++pa_it)
|
2012-03-25 16:12:23 +02:00
|
|
|
{
|
|
|
|
if (pa_it->second != NULL)
|
|
|
|
{
|
|
|
|
delete pa_it->second;
|
|
|
|
pa_it->second = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete mDebugDrawer;
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
delete dynamicsWorld;
|
|
|
|
delete solver;
|
|
|
|
delete collisionConfiguration;
|
|
|
|
delete dispatcher;
|
|
|
|
delete broadphase;
|
2012-03-25 18:12:00 +02:00
|
|
|
delete pairCache;
|
2011-04-03 13:12:12 +02:00
|
|
|
delete mShapeLoader;
|
2012-11-08 19:25:06 +01:00
|
|
|
|
|
|
|
delete BulletShapeManager::getSingletonPtr();
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
|
|
|
|
2012-03-13 17:09:50 +01:00
|
|
|
void PhysicEngine::addHeightField(float* heights,
|
|
|
|
int x, int y, float yoffset,
|
|
|
|
float triSize, float sqrtVerts)
|
|
|
|
{
|
|
|
|
const std::string name = "HeightField_"
|
|
|
|
+ boost::lexical_cast<std::string>(x) + "_"
|
|
|
|
+ boost::lexical_cast<std::string>(y);
|
|
|
|
|
|
|
|
// find the minimum and maximum heights (needed for bullet)
|
2012-09-23 19:36:37 +02:00
|
|
|
float minh = heights[0];
|
|
|
|
float maxh = heights[0];
|
2012-03-13 17:09:50 +01:00
|
|
|
for (int i=0; i<sqrtVerts*sqrtVerts; ++i)
|
|
|
|
{
|
|
|
|
float h = heights[i];
|
2012-07-03 12:30:50 +02:00
|
|
|
|
2012-03-13 17:09:50 +01:00
|
|
|
if (h>maxh) maxh = h;
|
|
|
|
if (h<minh) minh = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
btHeightfieldTerrainShape* hfShape = new btHeightfieldTerrainShape(
|
|
|
|
sqrtVerts, sqrtVerts, heights, 1,
|
|
|
|
minh, maxh, 2,
|
|
|
|
PHY_FLOAT,true);
|
|
|
|
|
|
|
|
hfShape->setUseDiamondSubdivision(true);
|
|
|
|
|
|
|
|
btVector3 scl(triSize, triSize, 1);
|
|
|
|
hfShape->setLocalScaling(scl);
|
|
|
|
|
|
|
|
CMotionState* newMotionState = new CMotionState(this,name);
|
|
|
|
|
|
|
|
btRigidBody::btRigidBodyConstructionInfo CI = btRigidBody::btRigidBodyConstructionInfo(0,newMotionState,hfShape);
|
|
|
|
RigidBody* body = new RigidBody(CI,name);
|
|
|
|
body->getWorldTransform().setOrigin(btVector3( (x+0.5)*triSize*(sqrtVerts-1), (y+0.5)*triSize*(sqrtVerts-1), (maxh+minh)/2.f));
|
|
|
|
|
2012-06-06 21:08:20 +02:00
|
|
|
HeightField hf;
|
|
|
|
hf.mBody = body;
|
|
|
|
hf.mShape = hfShape;
|
|
|
|
|
|
|
|
mHeightFieldMap [name] = hf;
|
|
|
|
|
2013-05-07 19:35:10 +01:00
|
|
|
dynamicsWorld->addRigidBody(body,CollisionType_HeightMap|CollisionType_Raycasting,
|
|
|
|
CollisionType_World|CollisionType_Actor|CollisionType_Raycasting);
|
2012-03-13 17:09:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void PhysicEngine::removeHeightField(int x, int y)
|
|
|
|
{
|
|
|
|
const std::string name = "HeightField_"
|
|
|
|
+ boost::lexical_cast<std::string>(x) + "_"
|
|
|
|
+ boost::lexical_cast<std::string>(y);
|
|
|
|
|
2012-06-06 21:08:20 +02:00
|
|
|
HeightField hf = mHeightFieldMap [name];
|
|
|
|
|
|
|
|
dynamicsWorld->removeRigidBody(hf.mBody);
|
|
|
|
delete hf.mShape;
|
|
|
|
delete hf.mBody;
|
2012-06-15 09:15:37 +02:00
|
|
|
|
|
|
|
mHeightFieldMap.erase(name);
|
2012-03-13 17:09:50 +01:00
|
|
|
}
|
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::adjustRigidBody(RigidBody* body, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation,
|
|
|
|
const Ogre::Vector3 &scaledBoxTranslation, const Ogre::Quaternion &boxRotation)
|
|
|
|
{
|
2012-08-14 18:04:58 -04:00
|
|
|
btTransform tr;
|
2013-02-07 16:18:29 -08:00
|
|
|
Ogre::Quaternion boxrot = rotation * boxRotation;
|
|
|
|
Ogre::Vector3 transrot = boxrot * scaledBoxTranslation;
|
2012-09-06 20:11:59 -04:00
|
|
|
Ogre::Vector3 newPosition = transrot + position;
|
2013-02-07 16:18:29 -08:00
|
|
|
|
2012-09-06 20:11:59 -04:00
|
|
|
tr.setOrigin(btVector3(newPosition.x, newPosition.y, newPosition.z));
|
2013-02-07 16:18:29 -08:00
|
|
|
tr.setRotation(btQuaternion(boxrot.x,boxrot.y,boxrot.z,boxrot.w));
|
2012-08-14 18:04:58 -04:00
|
|
|
body->setWorldTransform(tr);
|
|
|
|
}
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::boxAdjustExternal(const std::string &mesh, RigidBody* body,
|
|
|
|
float scale, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation)
|
|
|
|
{
|
2012-08-14 20:29:48 -04:00
|
|
|
std::string sid = (boost::format("%07.3f") % scale).str();
|
|
|
|
std::string outputstring = mesh + sid;
|
|
|
|
//std::cout << "The string" << outputstring << "\n";
|
|
|
|
|
|
|
|
//get the shape from the .nif
|
|
|
|
mShapeLoader->load(outputstring,"General");
|
|
|
|
BulletShapeManager::getSingletonPtr()->load(outputstring,"General");
|
|
|
|
BulletShapePtr shape = BulletShapeManager::getSingleton().getByName(outputstring,"General");
|
|
|
|
|
2013-03-08 23:46:25 +01:00
|
|
|
adjustRigidBody(body, position, rotation, shape->mBoxTranslation * scale, shape->mBoxRotation);
|
2012-08-14 20:29:48 -04:00
|
|
|
}
|
2012-08-14 18:04:58 -04:00
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
RigidBody* PhysicEngine::createAndAdjustRigidBody(const std::string &mesh, const std::string &name,
|
|
|
|
float scale, const Ogre::Vector3 &position, const Ogre::Quaternion &rotation,
|
2013-03-14 03:04:02 +01:00
|
|
|
Ogre::Vector3* scaledBoxTranslation, Ogre::Quaternion* boxRotation, bool raycasting, bool placeable)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2012-07-26 19:38:33 +04:00
|
|
|
std::string sid = (boost::format("%07.3f") % scale).str();
|
|
|
|
std::string outputstring = mesh + sid;
|
2012-06-11 19:55:10 -04:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
//get the shape from the .nif
|
2012-06-19 13:28:06 -04:00
|
|
|
mShapeLoader->load(outputstring,"General");
|
|
|
|
BulletShapeManager::getSingletonPtr()->load(outputstring,"General");
|
|
|
|
BulletShapePtr shape = BulletShapeManager::getSingleton().getByName(outputstring,"General");
|
2012-09-06 20:11:59 -04:00
|
|
|
|
2013-03-14 03:04:02 +01:00
|
|
|
if (placeable && !raycasting && shape->mCollisionShape && !shape->mHasCollisionNode)
|
|
|
|
return NULL;
|
|
|
|
|
2013-03-08 23:46:25 +01:00
|
|
|
if (!shape->mCollisionShape && !raycasting)
|
|
|
|
return NULL;
|
|
|
|
if (!shape->mRaycastingShape && raycasting)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!raycasting)
|
|
|
|
shape->mCollisionShape->setLocalScaling( btVector3(scale,scale,scale));
|
|
|
|
else
|
|
|
|
shape->mRaycastingShape->setLocalScaling( btVector3(scale,scale,scale));
|
2011-04-03 13:12:12 +02:00
|
|
|
|
|
|
|
//create the motionState
|
|
|
|
CMotionState* newMotionState = new CMotionState(this,name);
|
|
|
|
|
|
|
|
//create the real body
|
2013-03-08 23:46:25 +01:00
|
|
|
btRigidBody::btRigidBodyConstructionInfo CI = btRigidBody::btRigidBodyConstructionInfo
|
|
|
|
(0,newMotionState, raycasting ? shape->mRaycastingShape : shape->mCollisionShape);
|
2011-04-03 13:12:12 +02:00
|
|
|
RigidBody* body = new RigidBody(CI,name);
|
2013-03-14 03:04:02 +01:00
|
|
|
body->mPlaceable = placeable;
|
2012-08-12 00:36:49 -04:00
|
|
|
|
2012-09-10 20:29:24 -04:00
|
|
|
if(scaledBoxTranslation != 0)
|
2013-03-08 23:46:25 +01:00
|
|
|
*scaledBoxTranslation = shape->mBoxTranslation * scale;
|
2012-09-10 20:29:24 -04:00
|
|
|
if(boxRotation != 0)
|
2013-03-08 23:46:25 +01:00
|
|
|
*boxRotation = shape->mBoxRotation;
|
2012-09-10 20:29:24 -04:00
|
|
|
|
2013-03-08 23:46:25 +01:00
|
|
|
adjustRigidBody(body, position, rotation, shape->mBoxTranslation * scale, shape->mBoxRotation);
|
2012-08-12 00:36:49 -04:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
return body;
|
2012-05-17 16:12:55 +02:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-05-06 00:46:50 +01:00
|
|
|
void PhysicEngine::addRigidBody(RigidBody* body, bool addToMap, RigidBody* raycastingBody,bool actor)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
if(!body && !raycastingBody)
|
|
|
|
return; // nothing to do
|
|
|
|
|
|
|
|
const std::string& name = (body ? body->mName : raycastingBody->mName);
|
|
|
|
|
2013-05-06 00:46:50 +01:00
|
|
|
if (body){
|
2013-05-07 19:35:10 +01:00
|
|
|
if(actor) dynamicsWorld->addRigidBody(body,CollisionType_Actor,CollisionType_World|CollisionType_HeightMap);
|
|
|
|
else dynamicsWorld->addRigidBody(body,CollisionType_World,CollisionType_World|CollisionType_Actor|CollisionType_HeightMap);
|
2013-05-06 00:46:50 +01:00
|
|
|
}
|
2013-03-08 23:46:25 +01:00
|
|
|
|
|
|
|
if (raycastingBody)
|
|
|
|
dynamicsWorld->addRigidBody(raycastingBody,CollisionType_Raycasting,CollisionType_Raycasting|CollisionType_World);
|
|
|
|
|
|
|
|
if(addToMap){
|
|
|
|
removeRigidBody(name);
|
|
|
|
deleteRigidBody(name);
|
|
|
|
|
|
|
|
if (body)
|
|
|
|
mCollisionObjectMap[name] = body;
|
|
|
|
if (raycastingBody)
|
|
|
|
mRaycastingObjectMap[name] = raycastingBody;
|
2012-05-17 16:12:55 +02:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::removeRigidBody(const std::string &name)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBodyContainer::iterator it = mCollisionObjectMap.find(name);
|
|
|
|
if (it != mCollisionObjectMap.end() )
|
|
|
|
{
|
|
|
|
RigidBody* body = it->second;
|
|
|
|
if(body != NULL)
|
|
|
|
{
|
|
|
|
dynamicsWorld->removeRigidBody(body);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
it = mRaycastingObjectMap.find(name);
|
|
|
|
if (it != mRaycastingObjectMap.end() )
|
2011-03-23 19:17:45 +01:00
|
|
|
{
|
|
|
|
RigidBody* body = it->second;
|
|
|
|
if(body != NULL)
|
2011-03-07 14:55:18 +01:00
|
|
|
{
|
2012-03-25 16:12:23 +02:00
|
|
|
dynamicsWorld->removeRigidBody(body);
|
2011-03-07 14:55:18 +01:00
|
|
|
}
|
2011-03-23 19:17:45 +01:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::deleteRigidBody(const std::string &name)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBodyContainer::iterator it = mCollisionObjectMap.find(name);
|
|
|
|
if (it != mCollisionObjectMap.end() )
|
2011-03-23 19:17:45 +01:00
|
|
|
{
|
|
|
|
RigidBody* body = it->second;
|
2013-08-03 15:26:53 +02:00
|
|
|
|
2011-03-23 19:17:45 +01:00
|
|
|
if(body != NULL)
|
|
|
|
{
|
|
|
|
delete body;
|
|
|
|
}
|
2013-03-08 23:46:25 +01:00
|
|
|
mCollisionObjectMap.erase(it);
|
|
|
|
}
|
|
|
|
it = mRaycastingObjectMap.find(name);
|
|
|
|
if (it != mRaycastingObjectMap.end() )
|
|
|
|
{
|
|
|
|
RigidBody* body = it->second;
|
|
|
|
|
|
|
|
if(body != NULL)
|
2012-06-18 13:03:00 -04:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
delete body;
|
|
|
|
}
|
|
|
|
mRaycastingObjectMap.erase(it);
|
2011-03-23 19:17:45 +01:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
|
|
|
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBody* PhysicEngine::getRigidBody(const std::string &name, bool raycasting)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBodyContainer* map = raycasting ? &mRaycastingObjectMap : &mCollisionObjectMap;
|
|
|
|
RigidBodyContainer::iterator it = map->find(name);
|
|
|
|
if (it != map->end() )
|
2012-05-29 16:45:43 +02:00
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
RigidBody* body = (*map)[name];
|
2012-05-29 16:45:43 +02:00
|
|
|
return body;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-03-08 23:46:25 +01:00
|
|
|
return NULL;
|
2012-05-29 16:45:43 +02:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
|
|
|
|
2013-04-28 14:59:15 +02:00
|
|
|
class ContactTestResultCallback : public btCollisionWorld::ContactResultCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
std::vector<std::string> mResult;
|
2013-04-28 16:19:40 +02:00
|
|
|
|
|
|
|
// added in bullet 2.81
|
|
|
|
// this is just a quick hack, as there does not seem to be a BULLET_VERSION macro?
|
|
|
|
#if defined(BT_COLLISION_OBJECT_WRAPPER_H)
|
|
|
|
virtual btScalar addSingleResult(btManifoldPoint& cp,
|
2013-04-28 14:59:15 +02:00
|
|
|
const btCollisionObjectWrapper* colObj0Wrap,int partId0,int index0,
|
|
|
|
const btCollisionObjectWrapper* colObj1Wrap,int partId1,int index1)
|
|
|
|
{
|
|
|
|
const RigidBody* body = dynamic_cast<const RigidBody*>(colObj0Wrap->m_collisionObject);
|
2013-07-31 02:31:00 +02:00
|
|
|
if (body && !(colObj0Wrap->m_collisionObject->getBroadphaseHandle()->m_collisionFilterGroup
|
|
|
|
& CollisionType_Raycasting))
|
2013-04-28 14:59:15 +02:00
|
|
|
mResult.push_back(body->mName);
|
2013-07-31 02:31:00 +02:00
|
|
|
|
2013-04-28 14:59:15 +02:00
|
|
|
return 0.f;
|
|
|
|
}
|
2013-04-28 16:19:40 +02:00
|
|
|
#else
|
|
|
|
virtual btScalar addSingleResult(btManifoldPoint& cp, const btCollisionObject* col0, int partId0, int index0,
|
2013-04-28 15:56:09 +02:00
|
|
|
const btCollisionObject* col1, int partId1, int index1)
|
|
|
|
{
|
|
|
|
const RigidBody* body = dynamic_cast<const RigidBody*>(col0);
|
2013-07-31 02:31:00 +02:00
|
|
|
if (body && !(col0->getBroadphaseHandle()->m_collisionFilterGroup
|
|
|
|
& CollisionType_Raycasting))
|
2013-04-28 15:56:09 +02:00
|
|
|
mResult.push_back(body->mName);
|
2013-08-03 11:26:36 +02:00
|
|
|
|
2013-04-28 15:56:09 +02:00
|
|
|
return 0.f;
|
|
|
|
}
|
2013-04-28 16:19:40 +02:00
|
|
|
#endif
|
2013-04-28 14:59:15 +02:00
|
|
|
};
|
|
|
|
|
2013-08-03 11:26:36 +02:00
|
|
|
struct AabbResultCallback : public btBroadphaseAabbCallback {
|
|
|
|
std::vector<RigidBody*> hits;
|
|
|
|
//AabbResultCallback(){}
|
|
|
|
virtual bool process(const btBroadphaseProxy* proxy) {
|
|
|
|
RigidBody* collisionObject = static_cast<RigidBody*>(proxy->m_clientObject);
|
|
|
|
if(proxy->m_collisionFilterGroup == CollisionType_Actor && (collisionObject->mName != "player"))
|
|
|
|
this->hits.push_back(collisionObject);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-07-31 10:04:09 +02:00
|
|
|
|
2013-08-03 11:26:36 +02:00
|
|
|
std::pair<std::string,btVector3> PhysicEngine::sphereTest(float radius,btVector3& pos)
|
2013-07-31 10:04:09 +02:00
|
|
|
{
|
2013-08-03 11:26:36 +02:00
|
|
|
AabbResultCallback callback;
|
|
|
|
/*btDefaultMotionState* newMotionState =
|
|
|
|
new btDefaultMotionState(btTransform(btQuaternion(0,0,0,1),pos));
|
|
|
|
btCollisionShape * shape = new btSphereShape(radius);
|
|
|
|
btRigidBody::btRigidBodyConstructionInfo CI = btRigidBody::btRigidBodyConstructionInfo
|
|
|
|
(0,newMotionState, shape);
|
|
|
|
RigidBody* body = new RigidBody(CI,"hitDetectionShpere__");
|
|
|
|
btTransform tr = body->getWorldTransform();
|
|
|
|
tr.setOrigin(pos);
|
|
|
|
body->setWorldTransform(tr);
|
|
|
|
dynamicsWorld->addRigidBody(body,CollisionType_Actor,CollisionType_World|CollisionType_World);
|
|
|
|
body->setWorldTransform(tr);*/
|
2013-07-31 10:04:09 +02:00
|
|
|
|
2013-08-03 11:26:36 +02:00
|
|
|
btVector3 aabbMin = pos - radius*btVector3(1.0f, 1.0f, 1.0f);
|
|
|
|
btVector3 aabbMax = pos + radius*btVector3(1.0f, 1.0f, 1.0f);
|
|
|
|
|
|
|
|
broadphase->aabbTest(aabbMin,aabbMax,callback);
|
2013-08-03 15:26:53 +02:00
|
|
|
for(int i=0;i<static_cast<int> (callback.hits.size()); ++i)
|
2013-08-03 11:26:36 +02:00
|
|
|
{
|
2013-08-03 11:33:34 +02:00
|
|
|
float d = (callback.hits[i]->getWorldTransform().getOrigin()-pos).length();
|
|
|
|
if(d<radius)
|
2013-08-03 11:26:36 +02:00
|
|
|
{
|
2013-08-03 11:33:34 +02:00
|
|
|
std::pair<std::string,float> rayResult = this->rayTest(pos,callback.hits[i]->getWorldTransform().getOrigin());
|
|
|
|
if(rayResult.second>d || rayResult.first == callback.hits[i]->mName)
|
|
|
|
return std::make_pair(callback.hits[i]->mName,callback.hits[i]->getWorldTransform().getOrigin());
|
2013-08-03 11:26:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
//ContactTestResultCallback callback;
|
|
|
|
//dynamicsWorld->contactTest(body, callback);
|
|
|
|
//dynamicsWorld->removeRigidBody(body);
|
|
|
|
//delete body;
|
|
|
|
//delete shape;
|
|
|
|
//if(callback.mResultName.empty()) return std::make_pair(std::string(""),btVector3(0,0,0));
|
|
|
|
/*for(int i=0;i<callback.mResultName.size();i++)
|
|
|
|
{
|
|
|
|
//TODO: raycasting
|
|
|
|
if(callback.mResultName[i] != "hitDetectionShpere__")
|
|
|
|
return std::pair<std::string,btVector3>(callback.mResultName[i],callback.mResultContact[i]);
|
|
|
|
*/
|
|
|
|
return std::make_pair(std::string(""),btVector3(0,0,0));
|
2013-07-31 10:04:09 +02:00
|
|
|
}
|
|
|
|
|
2013-04-28 14:59:15 +02:00
|
|
|
std::vector<std::string> PhysicEngine::getCollisions(const std::string& name)
|
|
|
|
{
|
|
|
|
RigidBody* body = getRigidBody(name);
|
|
|
|
ContactTestResultCallback callback;
|
|
|
|
dynamicsWorld->contactTest(body, callback);
|
|
|
|
return callback.mResult;
|
|
|
|
}
|
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
void PhysicEngine::stepSimulation(double deltaT)
|
|
|
|
{
|
2013-04-18 16:46:38 +02:00
|
|
|
// This seems to be needed for character controller objects
|
|
|
|
dynamicsWorld->stepSimulation(deltaT,10, 1/60.0);
|
2011-04-03 13:12:12 +02:00
|
|
|
if(isDebugCreated)
|
|
|
|
{
|
|
|
|
mDebugDrawer->step();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::addCharacter(const std::string &name, const std::string &mesh,
|
|
|
|
const Ogre::Vector3 &position, float scale, const Ogre::Quaternion &rotation)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2012-03-25 16:12:23 +02:00
|
|
|
// Remove character with given name, so we don't make memory
|
|
|
|
// leak when character would be added twice
|
|
|
|
removeCharacter(name);
|
|
|
|
|
2012-09-03 20:32:20 -04:00
|
|
|
PhysicActor* newActor = new PhysicActor(name, mesh, this, position, rotation, scale);
|
2013-08-03 15:26:53 +02:00
|
|
|
|
2012-09-03 20:32:20 -04:00
|
|
|
|
|
|
|
//dynamicsWorld->addAction( newActor->mCharacter );
|
2013-05-01 11:15:43 +02:00
|
|
|
mActorMap[name] = newActor;
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
void PhysicEngine::removeCharacter(const std::string &name)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-05-01 11:15:43 +02:00
|
|
|
PhysicActorContainer::iterator it = mActorMap.find(name);
|
|
|
|
if (it != mActorMap.end() )
|
2011-03-23 19:17:45 +01:00
|
|
|
{
|
|
|
|
PhysicActor* act = it->second;
|
|
|
|
if(act != NULL)
|
|
|
|
{
|
2013-08-03 15:26:53 +02:00
|
|
|
|
2011-03-23 19:17:45 +01:00
|
|
|
delete act;
|
|
|
|
}
|
2013-05-01 11:15:43 +02:00
|
|
|
mActorMap.erase(it);
|
2011-03-23 19:17:45 +01:00
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2013-02-07 16:18:29 -08:00
|
|
|
PhysicActor* PhysicEngine::getCharacter(const std::string &name)
|
2011-04-03 13:12:12 +02:00
|
|
|
{
|
2013-05-01 11:15:43 +02:00
|
|
|
PhysicActorContainer::iterator it = mActorMap.find(name);
|
|
|
|
if (it != mActorMap.end() )
|
2012-05-29 16:45:43 +02:00
|
|
|
{
|
2013-05-01 11:15:43 +02:00
|
|
|
PhysicActor* act = mActorMap[name];
|
2012-05-29 16:45:43 +02:00
|
|
|
return act;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2011-04-03 13:12:12 +02:00
|
|
|
}
|
2011-02-22 14:02:50 +01:00
|
|
|
|
2011-04-03 13:12:12 +02:00
|
|
|
void PhysicEngine::emptyEventLists(void)
|
|
|
|
{
|
|
|
|
}
|
2011-02-22 20:53:02 +01:00
|
|
|
|
2013-05-07 19:42:21 +01:00
|
|
|
std::pair<std::string,float> PhysicEngine::rayTest(btVector3& from,btVector3& to,bool raycastingObjectOnly,bool ignoreHeightMap)
|
2011-02-22 20:53:02 +01:00
|
|
|
{
|
|
|
|
std::string name = "";
|
2011-03-22 21:28:18 +01:00
|
|
|
float d = -1;
|
2011-02-22 20:53:02 +01:00
|
|
|
|
2011-03-22 21:28:18 +01:00
|
|
|
btCollisionWorld::ClosestRayResultCallback resultCallback1(from, to);
|
2013-05-11 18:38:27 +02:00
|
|
|
if(raycastingObjectOnly)
|
|
|
|
resultCallback1.m_collisionFilterMask = CollisionType_Raycasting;
|
|
|
|
else
|
|
|
|
resultCallback1.m_collisionFilterMask = CollisionType_World;
|
2013-05-07 19:42:21 +01:00
|
|
|
|
2013-05-11 18:38:27 +02:00
|
|
|
if(!ignoreHeightMap)
|
|
|
|
resultCallback1.m_collisionFilterMask = resultCallback1.m_collisionFilterMask | CollisionType_HeightMap;
|
2011-03-22 21:28:18 +01:00
|
|
|
dynamicsWorld->rayTest(from, to, resultCallback1);
|
|
|
|
if (resultCallback1.hasHit())
|
2011-02-22 20:53:02 +01:00
|
|
|
{
|
2012-06-20 18:22:18 +02:00
|
|
|
name = static_cast<const RigidBody&>(*resultCallback1.m_collisionObject).mName;
|
2013-07-31 18:46:32 +02:00
|
|
|
d = resultCallback1.m_closestHitFraction;;
|
2011-03-22 21:28:18 +01:00
|
|
|
}
|
|
|
|
|
2011-02-22 20:53:02 +01:00
|
|
|
return std::pair<std::string,float>(name,d);
|
|
|
|
}
|
2012-03-25 12:26:44 +02:00
|
|
|
|
2013-04-30 20:26:59 +02:00
|
|
|
// callback that ignores player in results
|
|
|
|
struct OurClosestConvexResultCallback : public btCollisionWorld::ClosestConvexResultCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OurClosestConvexResultCallback(const btVector3& convexFromWorld,const btVector3& convexToWorld)
|
|
|
|
: btCollisionWorld::ClosestConvexResultCallback(convexFromWorld, convexToWorld) {}
|
|
|
|
|
|
|
|
virtual btScalar addSingleResult(btCollisionWorld::LocalConvexResult& convexResult,bool normalInWorldSpace)
|
|
|
|
{
|
|
|
|
if (const RigidBody* body = dynamic_cast<const RigidBody*>(convexResult.m_hitCollisionObject))
|
|
|
|
if (body->mName == "player")
|
|
|
|
return 0;
|
|
|
|
return btCollisionWorld::ClosestConvexResultCallback::addSingleResult(convexResult, normalInWorldSpace);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::pair<bool, float> PhysicEngine::sphereCast (float radius, btVector3& from, btVector3& to)
|
|
|
|
{
|
|
|
|
OurClosestConvexResultCallback callback(from, to);
|
2013-05-07 19:35:10 +01:00
|
|
|
callback.m_collisionFilterMask = OEngine::Physic::CollisionType_World|OEngine::Physic::CollisionType_HeightMap;
|
2013-04-30 20:26:59 +02:00
|
|
|
|
|
|
|
btSphereShape shape(radius);
|
|
|
|
const btQuaternion btrot(0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
|
|
btTransform from_ (btrot, from);
|
|
|
|
btTransform to_ (btrot, to);
|
|
|
|
|
|
|
|
dynamicsWorld->convexSweepTest(&shape, from_, to_, callback);
|
|
|
|
|
|
|
|
if (callback.hasHit())
|
|
|
|
return std::make_pair(true, callback.m_closestHitFraction);
|
|
|
|
else
|
|
|
|
return std::make_pair(false, 1);
|
|
|
|
}
|
|
|
|
|
2012-03-25 12:26:44 +02:00
|
|
|
std::vector< std::pair<float, std::string> > PhysicEngine::rayTest2(btVector3& from, btVector3& to)
|
|
|
|
{
|
|
|
|
MyRayResultCallback resultCallback1;
|
2013-03-08 23:46:25 +01:00
|
|
|
resultCallback1.m_collisionFilterMask = CollisionType_Raycasting;
|
2012-03-25 12:26:44 +02:00
|
|
|
dynamicsWorld->rayTest(from, to, resultCallback1);
|
2012-06-20 18:22:18 +02:00
|
|
|
std::vector< std::pair<float, const btCollisionObject*> > results = resultCallback1.results;
|
2012-03-25 12:26:44 +02:00
|
|
|
|
|
|
|
std::vector< std::pair<float, std::string> > results2;
|
|
|
|
|
2012-06-20 18:22:18 +02:00
|
|
|
for (std::vector< std::pair<float, const btCollisionObject*> >::iterator it=results.begin();
|
2012-03-25 12:26:44 +02:00
|
|
|
it != results.end(); ++it)
|
|
|
|
{
|
2012-06-20 18:22:18 +02:00
|
|
|
results2.push_back( std::make_pair( (*it).first, static_cast<const RigidBody&>(*(*it).second).mName ) );
|
2012-03-25 12:26:44 +02:00
|
|
|
}
|
|
|
|
|
2012-03-25 22:53:00 +02:00
|
|
|
std::sort(results2.begin(), results2.end(), MyRayResultCallback::cmp);
|
|
|
|
|
2012-03-25 12:26:44 +02:00
|
|
|
return results2;
|
|
|
|
}
|
2012-07-24 18:52:08 +04:00
|
|
|
|
2012-07-26 16:14:11 +04:00
|
|
|
void PhysicEngine::getObjectAABB(const std::string &mesh, float scale, btVector3 &min, btVector3 &max)
|
2012-07-25 18:58:55 +04:00
|
|
|
{
|
2012-07-26 19:38:33 +04:00
|
|
|
std::string sid = (boost::format("%07.3f") % scale).str();
|
|
|
|
std::string outputstring = mesh + sid;
|
2012-07-24 18:52:08 +04:00
|
|
|
|
|
|
|
mShapeLoader->load(outputstring, "General");
|
|
|
|
BulletShapeManager::getSingletonPtr()->load(outputstring, "General");
|
|
|
|
BulletShapePtr shape =
|
|
|
|
BulletShapeManager::getSingleton().getByName(outputstring, "General");
|
|
|
|
|
|
|
|
btTransform trans;
|
|
|
|
trans.setIdentity();
|
|
|
|
|
2013-03-21 21:08:23 +01:00
|
|
|
if (shape->mRaycastingShape)
|
|
|
|
shape->mRaycastingShape->getAabb(trans, min, max);
|
|
|
|
else if (shape->mCollisionShape)
|
|
|
|
shape->mCollisionShape->getAabb(trans, min, max);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
min = btVector3(0,0,0);
|
|
|
|
max = btVector3(0,0,0);
|
|
|
|
}
|
2012-07-24 18:52:08 +04:00
|
|
|
}
|
2013-05-01 11:15:43 +02:00
|
|
|
|
|
|
|
bool PhysicEngine::isAnyActorStandingOn (const std::string& objectName)
|
|
|
|
{
|
|
|
|
for (PhysicActorContainer::iterator it = mActorMap.begin(); it != mActorMap.end(); ++it)
|
|
|
|
{
|
|
|
|
if (!it->second->getOnGround())
|
|
|
|
continue;
|
|
|
|
Ogre::Vector3 pos = it->second->getPosition();
|
|
|
|
btVector3 from (pos.x, pos.y, pos.z);
|
|
|
|
btVector3 to = from - btVector3(0,0,5);
|
|
|
|
std::pair<std::string, float> result = rayTest(from, to);
|
|
|
|
if (result.first == objectName)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|