2011-02-22 14:02:50 +01:00
/*
Bullet Continuous Collision Detection and Physics Library
Copyright ( c ) 2003 - 2008 Erwin Coumans http : //bulletphysics.com
This software is provided ' as - is ' , without any express or implied warranty .
In no event will the authors be held liable for any damages arising from the use of this software .
2011-02-26 16:34:43 +01:00
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it freely ,
2011-02-22 14:02:50 +01:00
subject to the following restrictions :
1. The origin of this software must not be misrepresented ; you must not claim that you wrote the original software . If you use this software in a product , an acknowledgment in the product documentation would be appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not be misrepresented as being the original software .
3. This notice may not be removed or altered from any source distribution .
*/
# ifndef KINEMATIC_CHARACTER_CONTROLLER_H
# define KINEMATIC_CHARACTER_CONTROLLER_H
# include "LinearMath/btVector3.h"
2011-02-26 16:34:43 +01:00
# include "LinearMath/btQuickprof.h"
2011-02-22 14:02:50 +01:00
2011-02-26 16:34:43 +01:00
# include "BulletDynamics/Character/btCharacterControllerInterface.h"
2011-02-22 14:02:50 +01:00
# include "BulletCollision/BroadphaseCollision/btCollisionAlgorithm.h"
class btCollisionShape ;
class btRigidBody ;
class btCollisionWorld ;
class btCollisionDispatcher ;
class btPairCachingGhostObject ;
///btKinematicCharacterController is an object that supports a sliding motion in a world.
///It uses a ghost object and convex sweep test to test for upcoming collisions. This is combined with discrete collision detection to recover from penetrations.
///Interaction between btKinematicCharacterController and dynamic rigid bodies needs to be explicity implemented by the user.
class btKinematicCharacterController : public btCharacterControllerInterface
{
public :
enum UpAxis
{
X_AXIS = 0 ,
Y_AXIS = 1 ,
Z_AXIS = 2
} ;
private :
2011-04-03 13:12:12 +02:00
btPairCachingGhostObject * externalGhostObject ; // use this for querying collisions for sliding and move
btPairCachingGhostObject * internalGhostObject ; // and this for recoreving from penetrations
2011-02-26 16:34:43 +01:00
2011-04-03 13:12:12 +02:00
btScalar m_verticalVelocity ;
btScalar m_verticalOffset ;
btScalar m_fallSpeed ;
btScalar m_jumpSpeed ;
btScalar m_maxJumpHeight ;
btScalar m_maxSlopeRadians ; // Slope angle that is set (used for returning the exact value)
btScalar m_maxSlopeCosine ; // Cosine equivalent of m_maxSlopeRadians (calculated once when set, for optimization)
btScalar m_gravity ;
btScalar m_recoveringFactor ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
btScalar m_stepHeight ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
///this is the desired walk direction, set by the user
btVector3 m_walkDirection ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
///keep track of the contact manifolds
btManifoldArray m_manifoldArray ;
2011-02-22 14:02:50 +01:00
///Gravity attributes
2011-04-03 13:12:12 +02:00
bool m_wasJumping ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
bool m_useGhostObjectSweepTest ;
bool m_useWalkDirection ;
btScalar m_velocityTimeInterval ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
UpAxis m_upAxis ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
static btVector3 * getUpAxisDirections ( ) ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
bool recoverFromPenetration ( btCollisionWorld * collisionWorld ) ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
btVector3 stepUp ( btCollisionWorld * collisionWorld , const btVector3 & currentPosition , btScalar & currentStepOffset ) ;
2011-02-22 14:02:50 +01:00
btVector3 stepForwardAndStrafe ( btCollisionWorld * collisionWorld , const btVector3 & currentPosition , const btVector3 & walkMove ) ;
btScalar addFallOffset ( bool wasJumping , btScalar currentStepOffset , btScalar dt ) ;
btVector3 stepDown ( btCollisionWorld * collisionWorld , const btVector3 & currentPosition , btScalar currentStepOffset ) ;
public :
/// externalGhostObject is used for querying the collisions for sliding along the wall,
/// and internalGhostObject is used for querying the collisions for recovering from large penetrations.
/// These parameters can point on the same object.
/// Using a smaller internalGhostObject can help for removing some flickering but create some
/// stopping artefacts when sliding along stairs or small walls.
/// Don't forget to scale gravity and fallSpeed if you scale the world.
2011-04-03 13:12:12 +02:00
btKinematicCharacterController ( btPairCachingGhostObject * externalGhostObject ,
2011-02-22 14:02:50 +01:00
btPairCachingGhostObject * internalGhostObject ,
btScalar stepHeight ,
btScalar constantScale = btScalar ( 1.0 ) ,
btScalar gravity = btScalar ( 9.8 ) ,
btScalar fallVelocity = btScalar ( 55.0 ) ,
btScalar jumpVelocity = btScalar ( 9.8 ) ,
btScalar recoveringFactor = btScalar ( 0.2 ) ) ;
2011-04-03 13:12:12 +02:00
~ btKinematicCharacterController ( ) ;
2011-02-26 16:34:43 +01:00
2011-03-20 21:51:40 +01:00
void setVerticalVelocity ( float z ) ;
2011-02-22 14:02:50 +01:00
2011-04-03 13:12:12 +02:00
///btActionInterface interface
virtual void updateAction ( btCollisionWorld * collisionWorld , btScalar deltaTime )
{
2011-02-22 14:02:50 +01:00
preStep ( collisionWorld ) ;
2011-04-03 13:12:12 +02:00
playerStep ( collisionWorld , deltaTime ) ;
}
2011-02-26 16:34:43 +01:00
2011-04-03 13:12:12 +02:00
///btActionInterface interface
void debugDraw ( btIDebugDraw * debugDrawer ) ;
2011-02-22 14:02:50 +01:00
void setUpAxis ( UpAxis axis )
2011-04-03 13:12:12 +02:00
{
m_upAxis = axis ;
}
/// This should probably be called setPositionIncrementPerSimulatorStep.
/// This is neither a direction nor a velocity, but the amount to
/// increment the position each simulation iteration, regardless
/// of dt.
/// This call will reset any velocity set by setVelocityForTimeInterval().
virtual void setWalkDirection ( const btVector3 & walkDirection ) ;
/// Caller provides a velocity with which the character should move for
/// the given time period. After the time period, velocity is reset
/// to zero.
/// This call will reset any walk direction set by setWalkDirection().
/// Negative time intervals will result in no motion.
virtual void setVelocityForTimeInterval ( const btVector3 & velocity ,
btScalar timeInterval ) ;
void reset ( ) ;
void warp ( const btVector3 & origin ) ;
void preStep ( btCollisionWorld * collisionWorld ) ;
void playerStep ( btCollisionWorld * collisionWorld , btScalar dt ) ;
void setFallSpeed ( btScalar fallSpeed ) ;
void setJumpSpeed ( btScalar jumpSpeed ) ;
void setMaxJumpHeight ( btScalar maxJumpHeight ) ;
bool canJump ( ) const ;
void jump ( ) ;
void setGravity ( btScalar gravity ) ;
btScalar getGravity ( ) const ;
/// The max slope determines the maximum angle that the controller can walk up.
/// The slope angle is measured in radians.
void setMaxSlope ( btScalar slopeRadians ) ;
btScalar getMaxSlope ( ) const ;
void setUseGhostSweepTest ( bool useGhostObjectSweepTest )
{
m_useGhostObjectSweepTest = useGhostObjectSweepTest ;
}
bool onGround ( ) const ;
2011-03-17 11:00:46 +01:00
//if set to false, there will be no collision.
bool mCollision ;
2011-02-22 14:02:50 +01:00
} ;
# endif // KINEMATIC_CHARACTER_CONTROLLER_H