openmw-tes3coop/libs/openengine/bullet/physic.cpp

600 lines
20 KiB
C++
Raw Normal View History

2011-02-22 13:02:50 +00:00
#include "physic.hpp"
#include <btBulletDynamicsCommon.h>
#include <btBulletCollisionCommon.h>
#include <BulletCollision/CollisionShapes/btHeightfieldTerrainShape.h>
2012-09-15 22:17:42 +00:00
#include "pmove.h"
2011-02-26 15:34:43 +00:00
#include <components/nifbullet/bullet_nif_loader.hpp>
2011-02-22 13:02:50 +00:00
#include "CMotionState.h"
#include "OgreRoot.h"
#include "btKinematicCharacterController.h"
#include "BtOgrePG.h"
#include "BtOgreGP.h"
#include "BtOgreExtras.h"
#include <boost/lexical_cast.hpp>
2012-07-26 15:38:33 +00:00
#include <boost/format.hpp>
2011-02-22 13:02:50 +00:00
#define BIT(x) (1<<(x))
namespace OEngine {
namespace Physic
{
enum collisiontypes {
COL_NOTHING = 0, //<Collide with nothing
COL_WORLD = BIT(0), //<Collide with world objects
COL_ACTOR_INTERNAL = BIT(1), //<Collide internal capsule
COL_ACTOR_EXTERNAL = BIT(2), //<collide with external capsule
COL_RAYCASTING = BIT(3)
};
2012-09-04 00:32:20 +00:00
PhysicActor::PhysicActor(std::string name, std::string mesh, PhysicEngine* engine, Ogre::Vector3 position, Ogre::Quaternion rotation, float scale):
mName(name), mEngine(engine), mMesh(mesh), mBoxScaledTranslation(0,0,0), mBoxRotationInverse(0,0,0,0), mBody(0), collisionMode(false), mBoxRotation(0,0,0,0)
{
2012-09-15 17:23:49 +00:00
mBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, position, rotation, &mBoxScaledTranslation, &mBoxRotation);
Ogre::Quaternion inverse = mBoxRotation.Inverse();
mBoxRotationInverse = btQuaternion(inverse.x, inverse.y, inverse.z,inverse.w);
2012-09-05 21:44:11 +00:00
mEngine->addRigidBody(mBody, false); //Add rigid body to dynamics world, but do not add to object map
2012-09-15 22:17:42 +00:00
pmove = new playerMove;
}
2011-02-22 13:02:50 +00:00
PhysicActor::~PhysicActor()
{
2012-09-04 00:32:20 +00:00
if(mBody){
mEngine->dynamicsWorld->removeRigidBody(mBody);
delete mBody;
}
2012-09-15 22:17:42 +00:00
delete pmove;
}
2011-02-22 13:02:50 +00:00
void PhysicActor::setCurrentWater(bool hasWater, int waterHeight){
pmove->hasWater = hasWater;
if(hasWater){
pmove->waterHeight = waterHeight;
}
}
void PhysicActor::setGravity(float gravity)
{
pmove->ps.gravity = gravity;
}
void PhysicActor::enableCollisions(bool collision)
{
2012-09-05 21:44:11 +00:00
collisionMode = collision;
if(collisionMode)
pmove->ps.move_type=PM_NORMAL;
else
pmove->ps.move_type=PM_NOCLIP;
}
void PhysicActor::setJumpVelocity(float velocity)
{
pmove->ps.jump_velocity = velocity;
}
bool PhysicActor::getCollisionMode()
{
2012-09-05 21:44:11 +00:00
return collisionMode;
}
void PhysicActor::setWalkDirection(const btVector3& mvt)
{
2012-09-04 00:32:20 +00:00
}
2011-02-22 13:02:50 +00:00
2012-09-04 00:32:20 +00:00
2011-02-22 13:02:50 +00:00
2012-09-15 17:23:49 +00:00
void PhysicActor::setRotation(const Ogre::Quaternion quat)
{
2012-09-15 17:23:49 +00:00
if(!quat.equals(getRotation(), Ogre::Radian(0))){
mEngine->adjustRigidBody(mBody, getPosition(), quat, mBoxScaledTranslation, mBoxRotation);
}
}
2011-02-22 13:02:50 +00:00
2012-09-12 22:30:32 +00:00
Ogre::Vector3 PhysicActor::getPosition(void)
{
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 22:30:32 +00:00
Ogre::Vector3 visualPosition = Ogre::Vector3(vec.getX(), vec.getY(), vec.getZ()) - transrot;
return visualPosition;
}
2011-02-22 13:02:50 +00:00
2012-09-12 22:30:32 +00:00
Ogre::Quaternion PhysicActor::getRotation(void)
{
2012-09-12 22:30:32 +00:00
btQuaternion quat = mBody->getWorldTransform().getRotation() * mBoxRotationInverse;
return Ogre::Quaternion(quat.getW(), quat.getX(), quat.getY(), quat.getZ());
}
2011-02-22 13:02:50 +00:00
2012-09-15 20:45:50 +00:00
void PhysicActor::setPosition(const Ogre::Vector3 pos)
{
2012-09-15 20:45:50 +00:00
mEngine->adjustRigidBody(mBody, pos, getRotation(), mBoxScaledTranslation, mBoxRotation);
}
2011-02-22 13:02:50 +00:00
2012-09-12 22:30:32 +00:00
void PhysicActor::setScale(float scale){
Ogre::Vector3 position = getPosition();
Ogre::Quaternion rotation = getRotation();
//We only need to change the scaled box translation, box rotations remain the same.
mBoxScaledTranslation = mBoxScaledTranslation / mBody->getCollisionShape()->getLocalScaling().getX();
mBoxScaledTranslation *= scale;
if(mBody){
mEngine->dynamicsWorld->removeRigidBody(mBody);
delete mBody;
}
//Create the newly scaled rigid body
mBody = mEngine->createAndAdjustRigidBody(mMesh, mName, scale, position, rotation);
mEngine->addRigidBody(mBody, false); //Add rigid body to dynamics world, but do not add to object map
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////
2011-02-22 13:02:50 +00:00
RigidBody::RigidBody(btRigidBody::btRigidBodyConstructionInfo& CI,std::string name)
: btRigidBody(CI)
, mName(name)
{
}
2011-02-22 13:02:50 +00:00
RigidBody::~RigidBody()
{
delete getMotionState();
}
2011-02-22 13:02:50 +00:00
///////////////////////////////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////////////////////////////
2011-02-22 13:02:50 +00:00
PhysicEngine::PhysicEngine(BulletShapeLoader* shapeLoader) :
mDebugActive(0)
{
// Set up the collision configuration and dispatcher
collisionConfiguration = new btDefaultCollisionConfiguration();
dispatcher = new btCollisionDispatcher(collisionConfiguration);
2011-02-22 13:02:50 +00:00
// The actual physics solver
solver = new btSequentialImpulseConstraintSolver;
2011-02-22 13:02:50 +00:00
2012-03-25 02:03:08 +00:00
//btOverlappingPairCache* pairCache = new btSortedOverlappingPairCache();
pairCache = new btSortedOverlappingPairCache();
//pairCache->setInternalGhostPairCallback( new btGhostPairCallback() );
2011-02-22 13:02:50 +00:00
2012-03-25 02:03:08 +00:00
broadphase = new btDbvtBroadphase();
2011-02-22 13:02:50 +00:00
// The world.
dynamicsWorld = new btDiscreteDynamicsWorld(dispatcher,broadphase,solver,collisionConfiguration);
dynamicsWorld->setGravity(btVector3(0,0,-10));
2011-02-22 13:02:50 +00:00
if(BulletShapeManager::getSingletonPtr() == NULL)
{
new BulletShapeManager();
}
//TODO:singleton?
mShapeLoader = shapeLoader;
2011-02-22 13:02:50 +00:00
isDebugCreated = false;
mDebugDrawer = NULL;
}
2011-02-22 13:02:50 +00:00
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();
}
}
2011-02-22 13:02:50 +00:00
void PhysicEngine::setDebugRenderingMode(int mode)
{
if(!isDebugCreated)
{
createDebugRendering();
}
mDebugDrawer->setDebugMode(mode);
mDebugActive = mode;
}
bool PhysicEngine::toggleDebugRendering()
{
setDebugRenderingMode(!mDebugActive);
return mDebugActive;
}
2011-02-22 13:02:50 +00:00
PhysicEngine::~PhysicEngine()
{
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;
}
2012-09-15 22:17:42 +00:00
RigidBodyContainer::iterator rb_it = ObjectMap.begin();
for (; rb_it != ObjectMap.end(); ++rb_it)
{
if (rb_it->second != NULL)
{
dynamicsWorld->removeRigidBody(rb_it->second);
delete rb_it->second;
rb_it->second = NULL;
}
}
PhysicActorContainer::iterator pa_it = PhysicActorMap.begin();
for (; pa_it != PhysicActorMap.end(); ++pa_it)
{
if (pa_it->second != NULL)
{
2012-09-04 00:32:20 +00:00
delete pa_it->second;
pa_it->second = NULL;
}
}
delete mDebugDrawer;
delete dynamicsWorld;
delete solver;
delete collisionConfiguration;
delete dispatcher;
delete broadphase;
delete pairCache;
delete mShapeLoader;
}
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)
float minh;
float maxh;
for (int i=0; i<sqrtVerts*sqrtVerts; ++i)
{
float h = heights[i];
if (i==0)
{
minh = h;
maxh = h;
}
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->collide = true;
body->getWorldTransform().setOrigin(btVector3( (x+0.5)*triSize*(sqrtVerts-1), (y+0.5)*triSize*(sqrtVerts-1), (maxh+minh)/2.f));
2012-06-06 19:08:20 +00:00
HeightField hf;
hf.mBody = body;
hf.mShape = hfShape;
mHeightFieldMap [name] = hf;
dynamicsWorld->addRigidBody(body,COL_WORLD,COL_WORLD|COL_ACTOR_INTERNAL|COL_ACTOR_EXTERNAL);
}
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 19:08:20 +00:00
HeightField hf = mHeightFieldMap [name];
dynamicsWorld->removeRigidBody(hf.mBody);
delete hf.mShape;
delete hf.mBody;
2012-06-15 07:15:37 +00:00
mHeightFieldMap.erase(name);
}
void PhysicEngine::adjustRigidBody(RigidBody* body, Ogre::Vector3 position, Ogre::Quaternion rotation,
Ogre::Vector3 scaledBoxTranslation, Ogre::Quaternion boxRotation){
2012-08-14 22:04:58 +00:00
btTransform tr;
rotation = rotation * boxRotation;
Ogre::Vector3 transrot = rotation * scaledBoxTranslation;
Ogre::Vector3 newPosition = transrot + position;
tr.setOrigin(btVector3(newPosition.x, newPosition.y, newPosition.z));
2012-08-14 22:04:58 +00:00
tr.setRotation(btQuaternion(rotation.x,rotation.y,rotation.z,rotation.w));
body->setWorldTransform(tr);
}
void PhysicEngine::boxAdjustExternal(std::string mesh, RigidBody* body, float scale, Ogre::Vector3 position, Ogre::Quaternion rotation){
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");
adjustRigidBody(body, position, rotation, shape->boxTranslation * scale, shape->boxRotation);
}
2012-08-14 22:04:58 +00:00
2012-09-05 21:44:11 +00:00
RigidBody* PhysicEngine::createAndAdjustRigidBody(std::string mesh,std::string name,float scale, Ogre::Vector3 position, Ogre::Quaternion rotation,
2012-09-15 17:23:49 +00:00
Ogre::Vector3* scaledBoxTranslation, Ogre::Quaternion* boxRotation)
{
2012-07-26 15:38:33 +00:00
std::string sid = (boost::format("%07.3f") % scale).str();
std::string outputstring = mesh + sid;
2012-06-11 23:55:10 +00:00
//get the shape from the .nif
2012-06-19 17:28:06 +00:00
mShapeLoader->load(outputstring,"General");
BulletShapeManager::getSingletonPtr()->load(outputstring,"General");
BulletShapePtr shape = BulletShapeManager::getSingleton().getByName(outputstring,"General");
shape->Shape->setLocalScaling( btVector3(scale,scale,scale));
//create the motionState
CMotionState* newMotionState = new CMotionState(this,name);
//create the real body
2012-06-19 17:28:06 +00:00
btRigidBody::btRigidBodyConstructionInfo CI = btRigidBody::btRigidBodyConstructionInfo(0,newMotionState,shape->Shape);
RigidBody* body = new RigidBody(CI,name);
2011-02-22 19:53:02 +00:00
body->collide = shape->collide;
if(scaledBoxTranslation != 0)
*scaledBoxTranslation = shape->boxTranslation * scale;
if(boxRotation != 0)
2012-09-15 17:23:49 +00:00
*boxRotation = shape->boxRotation;
adjustRigidBody(body, position, rotation, shape->boxTranslation * scale, shape->boxRotation);
return body;
}
2011-02-22 13:02:50 +00:00
2012-09-05 21:44:11 +00:00
void PhysicEngine::addRigidBody(RigidBody* body, bool addToMap)
{
if(body)
2011-02-22 19:53:02 +00:00
{
if(body->collide)
{
dynamicsWorld->addRigidBody(body,COL_WORLD,COL_WORLD|COL_ACTOR_INTERNAL|COL_ACTOR_EXTERNAL);
}
else
{
dynamicsWorld->addRigidBody(body,COL_RAYCASTING,COL_RAYCASTING|COL_WORLD);
}
body->setActivationState(DISABLE_DEACTIVATION);
2012-09-05 21:44:11 +00:00
if(addToMap){
2012-09-15 22:17:42 +00:00
RigidBody* oldBody = ObjectMap[body->mName];
2012-09-05 21:44:11 +00:00
if (oldBody != NULL)
{
dynamicsWorld->removeRigidBody(oldBody);
delete oldBody;
}
2012-09-15 22:17:42 +00:00
ObjectMap[body->mName] = body;
}
}
}
2011-02-22 13:02:50 +00:00
void PhysicEngine::removeRigidBody(std::string name)
{
2012-09-15 22:17:42 +00:00
RigidBodyContainer::iterator it = ObjectMap.find(name);
if (it != ObjectMap.end() )
2011-03-23 18:17:45 +00:00
{
RigidBody* body = it->second;
if(body != NULL)
{
// broadphase->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher);
/*PhysicActorContainer::iterator it2 = PhysicActorMap.begin();
for(;it2!=PhysicActorMap.end();it++)
{
it2->second->internalGhostObject->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher);
it2->second->externalGhostObject->getOverlappingPairCache()->removeOverlappingPairsContainingProxy(body->getBroadphaseProxy(),dispatcher);
}*/
dynamicsWorld->removeRigidBody(body);
}
2011-03-23 18:17:45 +00:00
}
}
2011-02-22 13:02:50 +00:00
void PhysicEngine::deleteRigidBody(std::string name)
{
2012-09-15 22:17:42 +00:00
RigidBodyContainer::iterator it = ObjectMap.find(name);
if (it != ObjectMap.end() )
2011-03-23 18:17:45 +00:00
{
RigidBody* body = it->second;
2012-06-19 17:28:06 +00:00
//btScaledBvhTriangleMeshShape* scaled = dynamic_cast<btScaledBvhTriangleMeshShape*> (body->getCollisionShape());
2012-06-18 17:03:00 +00:00
2011-03-23 18:17:45 +00:00
if(body != NULL)
{
delete body;
}
2012-06-19 17:28:06 +00:00
/*if(scaled != NULL)
2012-06-18 17:03:00 +00:00
{
delete scaled;
2012-06-19 17:28:06 +00:00
}*/
2012-09-15 22:17:42 +00:00
ObjectMap.erase(it);
2011-03-23 18:17:45 +00:00
}
}
RigidBody* PhysicEngine::getRigidBody(std::string name)
{
2012-09-15 22:17:42 +00:00
RigidBodyContainer::iterator it = ObjectMap.find(name);
if (it != ObjectMap.end() )
2012-05-29 14:45:43 +00:00
{
2012-09-15 22:17:42 +00:00
RigidBody* body = ObjectMap[name];
2012-05-29 14:45:43 +00:00
return body;
}
else
{
return 0;
}
}
void PhysicEngine::stepSimulation(double deltaT)
{
dynamicsWorld->stepSimulation(deltaT,10, 1/60.0);
if(isDebugCreated)
{
mDebugDrawer->step();
}
}
2012-09-04 00:32:20 +00:00
void PhysicEngine::addCharacter(std::string name, std::string mesh,
Ogre::Vector3 position, float scale, Ogre::Quaternion rotation)
{
// Remove character with given name, so we don't make memory
// leak when character would be added twice
removeCharacter(name);
2012-09-04 00:32:20 +00:00
PhysicActor* newActor = new PhysicActor(name, mesh, this, position, rotation, scale);
//dynamicsWorld->addAction( newActor->mCharacter );
PhysicActorMap[name] = newActor;
}
void PhysicEngine::removeCharacter(std::string name)
{
2011-03-23 18:17:45 +00:00
//std::cout << "remove";
PhysicActorContainer::iterator it = PhysicActorMap.find(name);
2011-03-23 18:17:45 +00:00
if (it != PhysicActorMap.end() )
{
PhysicActor* act = it->second;
if(act != NULL)
{
2012-09-04 00:32:20 +00:00
2011-03-23 18:17:45 +00:00
delete act;
}
2011-03-23 21:49:23 +00:00
PhysicActorMap.erase(it);
2011-03-23 18:17:45 +00:00
}
}
2011-02-22 13:02:50 +00:00
PhysicActor* PhysicEngine::getCharacter(std::string name)
{
2012-05-29 14:45:43 +00:00
PhysicActorContainer::iterator it = PhysicActorMap.find(name);
if (it != PhysicActorMap.end() )
{
PhysicActor* act = PhysicActorMap[name];
return act;
}
else
{
return 0;
}
}
2011-02-22 13:02:50 +00:00
void PhysicEngine::emptyEventLists(void)
{
}
2011-02-22 19:53:02 +00:00
std::pair<std::string,float> PhysicEngine::rayTest(btVector3& from,btVector3& to)
{
std::string name = "";
2011-03-22 20:28:18 +00:00
float d = -1;
2011-02-22 19:53:02 +00:00
2011-03-22 20:28:18 +00:00
float d1 = 10000.;
btCollisionWorld::ClosestRayResultCallback resultCallback1(from, to);
resultCallback1.m_collisionFilterMask = COL_WORLD|COL_RAYCASTING;
2011-03-22 20:28:18 +00:00
dynamicsWorld->rayTest(from, to, resultCallback1);
if (resultCallback1.hasHit())
2011-02-22 19:53:02 +00:00
{
name = static_cast<const RigidBody&>(*resultCallback1.m_collisionObject).mName;
2011-03-22 20:28:18 +00:00
d1 = resultCallback1.m_closestHitFraction;
d = d1;
}
btCollisionWorld::ClosestRayResultCallback resultCallback2(from, to);
resultCallback2.m_collisionFilterMask = COL_ACTOR_INTERNAL|COL_ACTOR_EXTERNAL;
dynamicsWorld->rayTest(from, to, resultCallback2);
float d2 = 10000.;
if (resultCallback2.hasHit())
{
d2 = resultCallback1.m_closestHitFraction;
if(d2<=d1)
2011-02-22 19:53:02 +00:00
{
name = static_cast<const PairCachingGhostObject&>(*resultCallback2.m_collisionObject).mName;
2011-03-22 20:28:18 +00:00
d = d2;
2011-02-22 19:53:02 +00:00
}
}
2011-03-22 20:28:18 +00:00
2011-02-22 19:53:02 +00:00
return std::pair<std::string,float>(name,d);
}
std::vector< std::pair<float, std::string> > PhysicEngine::rayTest2(btVector3& from, btVector3& to)
{
MyRayResultCallback resultCallback1;
resultCallback1.m_collisionFilterMask = COL_WORLD|COL_RAYCASTING;
dynamicsWorld->rayTest(from, to, resultCallback1);
std::vector< std::pair<float, const btCollisionObject*> > results = resultCallback1.results;
MyRayResultCallback resultCallback2;
resultCallback2.m_collisionFilterMask = COL_ACTOR_INTERNAL|COL_ACTOR_EXTERNAL;
dynamicsWorld->rayTest(from, to, resultCallback2);
std::vector< std::pair<float, const btCollisionObject*> > actorResults = resultCallback2.results;
std::vector< std::pair<float, std::string> > results2;
for (std::vector< std::pair<float, const btCollisionObject*> >::iterator it=results.begin();
it != results.end(); ++it)
{
results2.push_back( std::make_pair( (*it).first, static_cast<const RigidBody&>(*(*it).second).mName ) );
}
for (std::vector< std::pair<float, const btCollisionObject*> >::iterator it=actorResults.begin();
it != actorResults.end(); ++it)
{
results2.push_back( std::make_pair( (*it).first, static_cast<const PairCachingGhostObject&>(*(*it).second).mName ) );
}
2012-03-25 20:53:00 +00:00
std::sort(results2.begin(), results2.end(), MyRayResultCallback::cmp);
return results2;
}
void PhysicEngine::getObjectAABB(const std::string &mesh, float scale, btVector3 &min, btVector3 &max)
2012-07-25 14:58:55 +00:00
{
2012-07-26 15:38:33 +00:00
std::string sid = (boost::format("%07.3f") % scale).str();
std::string outputstring = mesh + sid;
mShapeLoader->load(outputstring, "General");
BulletShapeManager::getSingletonPtr()->load(outputstring, "General");
BulletShapePtr shape =
BulletShapeManager::getSingleton().getByName(outputstring, "General");
btTransform trans;
trans.setIdentity();
shape->Shape->getAabb(trans, min, max);
}
2011-02-26 15:34:43 +00:00
}};