2014-11-14 13:17:04 +00:00
# include "physics.hh"
2014-11-28 12:52:03 +00:00
2014-11-17 11:57:16 +00:00
Physics : : Physics ( ) {
}
Physics : : ~ Physics ( ) {
}
2014-11-14 13:17:04 +00:00
2015-02-06 17:00:14 +00:00
void Physics : : init ( std : : string geometryPath ) //prepares bullet by creating all initial classes
2014-11-14 13:17:04 +00:00
{
2015-02-13 12:46:41 +00:00
colConfig = new btDefaultCollisionConfiguration ( ) ;
dispatcher = new btCollisionDispatcher ( colConfig ) ;
broadphase = new btDbvtBroadphase ( ) ;
solver = new btSequentialImpulseConstraintSolver ( ) ;
world = new btDiscreteDynamicsWorld ( dispatcher , broadphase , solver , colConfig ) ;
2015-01-13 16:50:15 +00:00
world - > setGravity ( btVector3 ( 0 , - 10 , - 0 ) ) ;
2015-02-06 17:00:14 +00:00
if ( world = = NULL ) {
printf ( " No World after init \n " ) ;
}
this - > geometryPath = geometryPath ;
2014-11-14 13:17:04 +00:00
}
2014-11-17 11:57:16 +00:00
void Physics : : takeUpdateStep ( float timeDiff )
2015-02-13 12:46:41 +00:00
{
2015-03-06 14:01:22 +00:00
if ( simulationActive )
2015-02-13 12:46:41 +00:00
{
2015-03-06 14:01:22 +00:00
for ( unsigned i = 0 ; i < allPositionConstraints . size ( ) ; i + + ) //this handles the spring constraints
{
if ( allPositionConstraints [ i ] . position ! = allPositionConstraints [ i ] . body - > getCenterOfMassPosition ( ) ) //if constraint != position of the body because otherwise dir = 0
{
btVector3 dir = allPositionConstraints [ i ] . position - allPositionConstraints [ i ] . body - > getCenterOfMassPosition ( ) ;
dir = dir * allPositionConstraints [ i ] . strength - allPositionConstraints [ i ] . body - > getLinearVelocity ( )
* allPositionConstraints [ i ] . body - > getLinearVelocity ( ) . length ( ) ;
allPositionConstraints [ i ] . body - > applyCentralForce ( dir * allPositionConstraints [ i ] . strength ) ; //apply a foce upon the object pushing it towards the constraint position
}
}
2015-03-06 15:19:57 +00:00
if ( endgame )
{
currentDirection = playerBall - > getCenterOfMassPosition ( ) ;
currentDirection . setY ( 0 ) ;
currentDirection . normalize ( ) ;
2015-03-09 17:57:40 +00:00
cameraDistance = 15 ;
2015-03-06 15:19:57 +00:00
}
btVector3 position = currentDirection ;
2015-03-06 14:01:22 +00:00
position . normalize ( ) ;
position * = cameraDistance ;
position + = playerBall - > getCenterOfMassPosition ( ) ; //is the position cameraDistance away from the player in the direction of the camera
btVector3 dir = cameraBody - > getCenterOfMassPosition ( ) - position ;
float str = 50 * dir . length ( ) / cameraBody - > getInvMass ( ) ; //getInvMass() returns the inverted mass
cameraBody - > setLinearVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
2015-03-06 16:45:51 +00:00
cameraBody - > applyCentralForce ( - dir * str * 3 ) ; //scale the force by camera mass
2015-03-06 14:01:22 +00:00
counter = 0 ;
2015-03-06 16:45:51 +00:00
float force = cameraBody - > getTotalForce ( ) . length ( ) ;
if ( force > 1.0f )
2015-02-13 12:46:41 +00:00
{
2015-03-06 16:45:51 +00:00
btVector3 currentForce = cameraBody - > getTotalForce ( ) ;
cameraBody - > clearForces ( ) ;
cameraBody - > applyCentralForce ( currentForce * 1.0 / force ) ;
2015-02-13 12:46:41 +00:00
}
2015-03-06 15:19:57 +00:00
2015-03-06 14:01:22 +00:00
world - > stepSimulation ( timeDiff ) ;
2015-02-13 12:46:41 +00:00
}
2015-03-06 14:01:22 +00:00
else
2015-02-13 12:46:41 +00:00
{
2015-03-09 11:30:50 +00:00
for ( unsigned i = 0 ; i < allPositionConstraints . size ( ) ; i + + ) //this handles the spring constraints
{
if ( allPositionConstraints [ i ] . position ! = allPositionConstraints [ i ] . body - > getCenterOfMassPosition ( ) ) //if constraint != position of the body because otherwise dir = 0
{
btVector3 dir = allPositionConstraints [ i ] . position - allPositionConstraints [ i ] . body - > getCenterOfMassPosition ( ) ;
dir = dir * allPositionConstraints [ i ] . strength - allPositionConstraints [ i ] . body - > getLinearVelocity ( )
* allPositionConstraints [ i ] . body - > getLinearVelocity ( ) . length ( ) ;
allPositionConstraints [ i ] . body - > applyCentralForce ( dir * allPositionConstraints [ i ] . strength ) ; //apply a foce upon the object pushing it towards the constraint position
}
}
btVector3 camPos = cameraBody - > getCenterOfMassPosition ( ) ;
2015-03-06 14:01:22 +00:00
if ( sinking )
{
btVector3 currentPos = playerBall - > getCenterOfMassPosition ( ) ;
2015-03-09 15:00:07 +00:00
currentPos - = btVector3 ( 0 , 0.8f * timeDiff , 0 ) ;
2015-03-09 11:30:50 +00:00
float damp = playerBall - > getAngularDamping ( ) ;
playerBall - > setDamping ( playerBall - > getLinearDamping ( ) , 0.9 ) ;
world - > stepSimulation ( timeDiff ) ;
cameraBody - > setCenterOfMassTransform ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , camPos ) ) ;
playerBall - > setDamping ( playerBall - > getLinearDamping ( ) , damp ) ;
2015-03-06 14:01:22 +00:00
playerBall - > setCenterOfMassTransform ( btTransform ( playerBall - > getOrientation ( ) , currentPos ) ) ;
2015-03-06 15:56:20 +00:00
if ( playerBall - > getCenterOfMassPosition ( ) . y ( ) < resetHight - 1.5f )
2015-03-06 14:01:22 +00:00
{
playerBall - > setCenterOfMassTransform ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , btVector3 ( startPosition . x ( ) , startPosition . y ( ) - 3 , startPosition . z ( ) ) ) ) ;
playerBall - > setLinearVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
playerBall - > setAngularVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
2015-03-06 15:56:20 +00:00
currentDirection = btVector3 ( 1 , 1 , 1 ) ;
currentDirection . normalize ( ) ;
forceMoveCamera ( startPosition + currentDirection * cameraDistance ) ;
2015-03-06 14:01:22 +00:00
sinking = false ;
}
}
else
{
btVector3 currentPos = playerBall - > getCenterOfMassPosition ( ) ;
2015-03-09 15:00:07 +00:00
currentPos + = btVector3 ( 0 , 3.0f * timeDiff , 0 ) ;
2015-03-09 11:30:50 +00:00
world - > stepSimulation ( timeDiff ) ;
2015-03-06 14:01:22 +00:00
playerBall - > setCenterOfMassTransform ( btTransform ( playerBall - > getOrientation ( ) , currentPos ) ) ;
2015-03-09 11:30:50 +00:00
cameraBody - > setCenterOfMassTransform ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , camPos ) ) ;
2015-03-06 14:01:22 +00:00
if ( playerBall - > getCenterOfMassPosition ( ) . y ( ) > = startPosition . y ( ) + 1 )
{
sinking = true ;
simulationActive = true ;
}
}
2015-02-13 12:46:41 +00:00
}
2014-11-14 13:17:04 +00:00
}
2015-02-13 12:46:41 +00:00
2015-01-29 12:15:45 +00:00
void Physics : : removePositionConstraint ( int bodyIndice ) //remover function for deleting all pos constraints on one body
2015-01-16 15:17:26 +00:00
{
2015-02-13 12:46:41 +00:00
for ( unsigned i = 0 ; i < allPositionConstraints . size ( ) ; i + + )
{
if ( allPositionConstraints [ i ] . body = = bodies [ bodyIndice ] )
{
allPositionConstraints . erase ( allPositionConstraints . begin ( ) + i ) ;
}
}
2015-01-16 15:17:26 +00:00
}
2015-01-29 12:15:45 +00:00
void Physics : : addPositionConstraint ( int bodyIndice , float strength , glm : : vec3 position ) //function for adding position constraints
2015-01-16 15:17:26 +00:00
{
positionConstraint cons ;
cons . body = bodies [ bodyIndice ] ;
2015-02-14 08:39:16 +00:00
cons . body - > setSleepingThresholds ( 0 , 0 ) ;
2015-01-16 15:17:26 +00:00
cons . strength = strength ;
cons . position = btVector3 ( position . x , position . y , position . z ) ;
allPositionConstraints . push_back ( cons ) ;
2015-02-13 12:46:41 +00:00
}
2015-01-16 15:17:26 +00:00
2015-01-29 12:15:45 +00:00
//players and objects
void Physics : : addPlayer ( float friction , float rad , Entity entity , float mass , float dampningL , float dampningA , unsigned indice )
{
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) = = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ; //these error are to ensure that level can always communicate with physics without having to worry about synching errors
btSphereShape * sphere = new btSphereShape ( rad ) ; //the first thing we need for a rigid body is the shape
btVector3 inertia ( 0 , 0 , 0 ) ;
if ( mass ! = 0.0 )
{
sphere - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ; //from this shape we can then calculate the innertia, as long as the mass != 0 (otherwise inertia = 0)
}
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ; //next we define the motionstate, wich describes the innital position and rotation
2015-03-06 14:01:22 +00:00
startPosition = btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ;
2015-02-13 12:46:41 +00:00
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , sphere , inertia ) ; //next we process all data for the rigid body into info
2015-03-09 16:01:14 +00:00
info . m_friction = friction ; //here we modify the friction and restitution (bounciness) of the object
2015-03-04 12:37:27 +00:00
info . m_restitution = 0.0f ;
2015-02-13 12:46:41 +00:00
playerBall = new btRigidBody ( info ) ; //finally we create the rigid body using the info
2014-12-15 15:02:04 +00:00
2015-01-29 12:15:45 +00:00
playerBall - > setDamping ( dampningL , dampningA ) ; //here we can set the dampning (how much of the motion is lost)
2014-12-15 15:02:04 +00:00
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( playerBall , COL_OBJECTS , COL_OBJECTS | COL_OBJECTS_NO_TERRAIN | COL_TERRAIN ) ; //then we add the rigid body to the wiorld, allowing it to be simulated
2015-01-30 16:29:48 +00:00
2015-02-13 12:46:41 +00:00
bodies . push_back ( playerBall ) ; //next we add the rigid body to our own list (for cleanup and for synchronitaation with level)
2015-01-30 12:32:16 +00:00
//note, while we can always access playerBall through its global name, we add it to this array for synchronization purposes
2014-12-15 15:02:04 +00:00
2015-01-29 12:15:45 +00:00
playerBall - > setSleepingThresholds ( 0 , 0 ) ; //in a final step we make sure that the body never is removed from the active rigid bodies
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ; //one last check to make sure level and physics are in synch
2015-01-29 12:15:45 +00:00
addCamera ( ) ; //now that the player exists add a camera for the player
2014-12-15 15:02:04 +00:00
}
2015-02-06 17:00:14 +00:00
void Physics : : addTerrain ( int width , int length , float * * heightData ) //The terrain adding function
2014-11-14 13:23:10 +00:00
{
2015-02-13 12:46:41 +00:00
float * heightfield = new float [ width * length ] ; //bullet only accepts data in a one dimensional array, so parse data into appropriate format
2015-02-06 17:00:14 +00:00
int highest = - 999999 , j = 0 , i = 0 ;
for ( i = 0 ; i < width ; i + + )
{
for ( j = 0 ; j < length ; j + + ) {
heightfield [ i * length + j ] = heightData [ j ] [ i ] ; //reverse order because they are loaded backwards
2015-02-13 12:46:41 +00:00
2015-02-06 17:00:14 +00:00
if ( heightData [ j ] [ i ] > highest )
2015-02-13 12:46:41 +00:00
highest = heightData [ j ] [ i ] ; //bullet needs to know the highest point of the heightmap
2015-02-06 17:00:14 +00:00
}
}
highest + + ;
2015-02-13 12:46:41 +00:00
btHeightfieldTerrainShape * terrainShape = new btHeightfieldTerrainShape ( length , width , heightfield , highest , 1 , true , false ) ;
btRigidBody : : btRigidBodyConstructionInfo info ( 0 , new btDefaultMotionState ( ) , terrainShape , btVector3 ( 0 , 0 , 0 ) ) ; //next we process all data for the rigid body into info
2015-03-06 12:58:25 +00:00
info . m_friction = 0.8f ;
2015-01-30 16:29:48 +00:00
info . m_restitution = 0 ;
2015-02-13 12:46:41 +00:00
btRigidBody * tBody = new btRigidBody ( info ) ;
tBody - > getWorldTransform ( ) . setOrigin ( btVector3 ( 0 , ( ( float ) highest ) / 2 , 0 ) ) ; //we have to move the origin of our rigid body down, because bullet sets the origin (0,0,0) at (width/2, height/2, length/2) in the map the x and z are correct in our level, but y needs to be addapted
terrainBody = tBody ;
2015-02-06 17:00:14 +00:00
if ( world = = NULL ) {
printf ( " No World while adding terrain. \n " ) ;
}
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( terrainBody , COL_TERRAIN , COL_TERRAIN | COL_OBJECTS ) ; //COL_XXXX are collision masks, allowing us to ignore collisions between certain object groups (required for buttons)
2014-11-14 13:23:10 +00:00
}
2015-02-27 14:58:21 +00:00
void Physics : : addConvexBody ( Entity entity , std : : string path , float mass , float dampningL , float dampningA , unsigned indice , float scaling , bool rotate )
{
if ( bodies . size ( ) = = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
std : : vector < unsigned int > vertexIndices ; //temp lists for data sets
std : : vector < btVector3 > temp_vertices ;
path = " ../ " + geometryPath + path ;
FILE * file = fopen ( path . c_str ( ) , " r " ) ;
if ( file = = NULL ) {
throw std : : invalid_argument ( " Impossible to open the file " ) ; //create correct filepath and report error if cannot open
}
while ( 1 ) {
char lineHeader [ 128 ] ;
//read the first word of the line
int res = fscanf ( file , " %s " , lineHeader ) ;
if ( res = = EOF )
break ; //while not at end do loop
if ( strcmp ( lineHeader , " v " ) = = 0 ) { //if a vertex
glm : : vec3 vertex ;
fscanf ( file , " %f %f %f \n " , & vertex . x , & vertex . y , & vertex . z ) ;
temp_vertices . push_back ( btVector3 ( vertex . x , vertex . y , vertex . z ) ) ; //save vertex in array
}
else if ( strcmp ( lineHeader , " f " ) = = 0 ) { //if face (index for 3 vertexes for a triangle)
std : : string vertex1 , vertex2 , vertex3 ;
unsigned int vertexIndex [ 3 ] , uvIndex [ 3 ] , normalIndex [ 3 ] ;
int matches = fscanf ( file , " %d/%d/%d %d/%d/%d %d/%d/%d \n " , & vertexIndex [ 0 ] , & uvIndex [ 0 ] , & normalIndex [ 0 ] , & vertexIndex [ 1 ] , & uvIndex [ 1 ] , & normalIndex [ 1 ] , & vertexIndex [ 2 ] , & uvIndex [ 2 ] , & normalIndex [ 2 ] ) ;
vertexIndices . push_back ( vertexIndex [ 0 ] ) ;
vertexIndices . push_back ( vertexIndex [ 1 ] ) ;
vertexIndices . push_back ( vertexIndex [ 2 ] ) ; //save 3 indexes in array
}
}
//finally start making body
btTriangleMesh * triMesh = new btTriangleMesh ( ) ;
for ( unsigned i = 2 ; i < vertexIndices . size ( ) ; i + = 3 )
{
triMesh - > addTriangle ( temp_vertices [ vertexIndices [ i ] ] , temp_vertices [ vertexIndices [ i - 1 ] ] , temp_vertices [ vertexIndices [ i - 2 ] ] ) ; //for every face (3 elements in vertexIndices) create triangle use the indices to find correct vertexes to make the triangle
}
btConvexTriangleMeshShape * shape = new btConvexTriangleMeshShape ( triMesh , true ) ;
shape - > setLocalScaling ( btVector3 ( scaling , scaling , scaling ) ) ; //we need to add a scaling here because the objects seem to have diffrent sizes when loaded (no clue why, see composition.xml for exact scaling factors)
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ;
btVector3 inertia ( 0 , 0 , 0 ) ;
if ( mass ! = 0.0 )
{
shape - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ;
}
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , shape , inertia ) ;
btRigidBody * body = new btRigidBody ( info ) ;
body - > setDamping ( dampningL , dampningA ) ;
bodies . push_back ( body ) ;
world - > addRigidBody ( body , COL_OBJECTS , objectsPhysicsCollision ) ;
if ( ! rotate ) //rotate lets certain objects get inertia (0,0,0) (not rotateable)
{
body - > setAngularFactor ( btVector3 ( 0 , 0 , 0 ) ) ;
}
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
}
2015-01-29 12:15:45 +00:00
void Physics : : addTriangleMeshBody ( Entity entity , std : : string path , float mass , float dampningL , float dampningA , unsigned indice , float scaling , bool rotate )
2014-11-14 13:17:04 +00:00
{
2014-12-15 15:02:04 +00:00
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) = = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
2015-01-29 12:15:45 +00:00
std : : vector < unsigned int > vertexIndices ; //temp lists for data sets
2014-12-15 15:02:04 +00:00
std : : vector < btVector3 > temp_vertices ;
2015-02-06 17:00:14 +00:00
path = " ../ " + geometryPath + path ;
2014-12-15 15:02:04 +00:00
FILE * file = fopen ( path . c_str ( ) , " r " ) ;
if ( file = = NULL ) {
2015-01-29 12:15:45 +00:00
throw std : : invalid_argument ( " Impossible to open the file " ) ; //create correct filepath and report error if cannot open
2014-12-15 15:02:04 +00:00
}
while ( 1 ) {
char lineHeader [ 128 ] ;
2015-02-13 12:46:41 +00:00
//read the first word of the line
2014-12-15 15:02:04 +00:00
int res = fscanf ( file , " %s " , lineHeader ) ;
if ( res = = EOF )
2015-02-13 12:46:41 +00:00
break ; //while not at end do loop
2015-01-29 12:15:45 +00:00
if ( strcmp ( lineHeader , " v " ) = = 0 ) { //if a vertex
2014-12-15 15:02:04 +00:00
glm : : vec3 vertex ;
fscanf ( file , " %f %f %f \n " , & vertex . x , & vertex . y , & vertex . z ) ;
2015-01-29 12:15:45 +00:00
temp_vertices . push_back ( btVector3 ( vertex . x , vertex . y , vertex . z ) ) ; //save vertex in array
2014-12-15 15:02:04 +00:00
}
2015-01-29 12:15:45 +00:00
else if ( strcmp ( lineHeader , " f " ) = = 0 ) { //if face (index for 3 vertexes for a triangle)
2014-12-15 15:02:04 +00:00
std : : string vertex1 , vertex2 , vertex3 ;
unsigned int vertexIndex [ 3 ] , uvIndex [ 3 ] , normalIndex [ 3 ] ;
2015-02-13 12:46:41 +00:00
int matches = fscanf ( file , " %d/%d/%d %d/%d/%d %d/%d/%d \n " , & vertexIndex [ 0 ] , & uvIndex [ 0 ] , & normalIndex [ 0 ] , & vertexIndex [ 1 ] , & uvIndex [ 1 ] , & normalIndex [ 1 ] , & vertexIndex [ 2 ] , & uvIndex [ 2 ] , & normalIndex [ 2 ] ) ;
2014-12-15 15:02:04 +00:00
vertexIndices . push_back ( vertexIndex [ 0 ] ) ;
vertexIndices . push_back ( vertexIndex [ 1 ] ) ;
2015-01-29 12:15:45 +00:00
vertexIndices . push_back ( vertexIndex [ 2 ] ) ; //save 3 indexes in array
2014-12-15 15:02:04 +00:00
}
}
2015-02-13 12:46:41 +00:00
//finally start making body
2014-12-15 15:02:04 +00:00
btTriangleMesh * triMesh = new btTriangleMesh ( ) ;
2014-12-19 14:20:05 +00:00
for ( unsigned i = 2 ; i < vertexIndices . size ( ) ; i + = 3 )
2014-12-15 15:02:04 +00:00
{
2015-02-13 12:46:41 +00:00
triMesh - > addTriangle ( temp_vertices [ vertexIndices [ i ] ] , temp_vertices [ vertexIndices [ i - 1 ] ] , temp_vertices [ vertexIndices [ i - 2 ] ] ) ; //for every face (3 elements in vertexIndices) create triangle use the indices to find correct vertexes to make the triangle
2014-12-15 15:02:04 +00:00
}
btBvhTriangleMeshShape * shape = new btBvhTriangleMeshShape ( triMesh , true ) ;
2015-01-29 12:15:45 +00:00
shape - > setLocalScaling ( btVector3 ( scaling , scaling , scaling ) ) ; //we need to add a scaling here because the objects seem to have diffrent sizes when loaded (no clue why, see composition.xml for exact scaling factors)
2015-01-30 16:29:48 +00:00
2015-02-13 12:46:41 +00:00
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
2015-01-30 16:29:48 +00:00
2015-02-13 12:46:41 +00:00
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ;
2014-12-15 15:02:04 +00:00
2015-02-13 12:46:41 +00:00
btVector3 inertia ( 0 , 0 , 0 ) ;
2015-02-13 15:09:18 +00:00
if ( mass ! = 0.0 )
2015-02-13 12:46:41 +00:00
{
shape - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ;
}
2014-12-15 15:02:04 +00:00
2015-02-13 12:46:41 +00:00
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , shape , inertia ) ;
btRigidBody * body = new btRigidBody ( info ) ;
body - > setDamping ( dampningL , dampningA ) ;
bodies . push_back ( body ) ;
world - > addRigidBody ( body , COL_OBJECTS , objectsPhysicsCollision ) ;
2014-12-15 15:02:04 +00:00
2015-02-13 15:09:18 +00:00
if ( ! rotate ) //rotate lets certain objects get inertia (0,0,0) (not rotateable)
{
body - > setAngularFactor ( btVector3 ( 0 , 0 , 0 ) ) ;
}
2014-12-15 15:02:04 +00:00
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
2014-12-15 15:02:04 +00:00
}
2015-01-23 12:31:19 +00:00
void Physics : : addButton ( float width , float height , float length , Entity entity , float mass , float dampningL , float dampningA , unsigned indice , bool rotate )
2015-01-19 13:01:05 +00:00
{
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) = = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
btBoxShape * box = new btBoxShape ( btVector3 ( width / 2 , height / 2 , length / 2 ) ) ;
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ;
btVector3 inertia ( 0 , 0 , 0 ) ;
2015-02-13 15:09:18 +00:00
if ( mass ! = 0.0 ) //&& rotate lets certain objects get inertia (0,0,0) (not rotateable)
2015-02-13 12:46:41 +00:00
{
box - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ;
}
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , box , inertia ) ;
2015-03-09 16:01:14 +00:00
info . m_friction = 0.2 ; //here we modify the friction and restitution (bounciness) of the object
info . m_restitution = 0.0f ;
2015-02-13 12:46:41 +00:00
btRigidBody * body = new btRigidBody ( info ) ;
2015-01-19 13:01:05 +00:00
body - > setDamping ( dampningL , dampningA ) ;
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( body , COL_OBJECTS_NO_TERRAIN , specialPhysicsCollision ) ; //the specialPhysicsCollision allows these objects to not collide with the terrain
bodies . push_back ( body ) ;
2015-02-13 15:09:18 +00:00
if ( ! rotate )
{
body - > setAngularFactor ( btVector3 ( 0 , 0 , 0 ) ) ;
}
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
2015-01-19 13:01:05 +00:00
}
2015-01-23 12:31:19 +00:00
void Physics : : addBox ( float width , float height , float length , Entity entity , float mass , float dampningL , float dampningA , unsigned indice , bool rotate )
2014-11-21 11:31:29 +00:00
{
2015-02-13 12:46:41 +00:00
//similar to other constructors
if ( bodies . size ( ) = = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
btBoxShape * box = new btBoxShape ( btVector3 ( width / 2 , height / 2 , length / 2 ) ) ;
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ;
btVector3 inertia ( 0 , 0 , 0 ) ;
2015-02-13 15:09:18 +00:00
if ( mass ! = 0.0 ) //&& rotate lets certain objects get inertia (0,0,0) (not rotateable)
2015-02-13 12:46:41 +00:00
{
box - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ;
}
2015-02-13 15:09:18 +00:00
2015-02-13 12:46:41 +00:00
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , box , inertia ) ;
btRigidBody * body = new btRigidBody ( info ) ;
2015-01-06 12:31:53 +00:00
body - > setDamping ( dampningL , dampningA ) ;
2014-11-21 11:50:11 +00:00
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( body , COL_OBJECTS , objectsPhysicsCollision ) ;
bodies . push_back ( body ) ;
2015-02-13 15:09:18 +00:00
if ( ! rotate )
{
body - > setAngularFactor ( btVector3 ( 0 , 0 , 0 ) ) ;
}
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
2014-11-21 11:31:29 +00:00
}
2014-11-14 15:23:14 +00:00
2015-01-23 12:31:19 +00:00
void Physics : : addSphere ( float rad , Entity entity , float mass , float dampningL , float dampningA , unsigned indice , bool rotate )
2014-11-14 15:23:14 +00:00
{
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) = = indice ) //(user's initial) height, not the actual height. More...
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
btSphereShape * sphere = new btSphereShape ( rad ) ;
btVector3 inertia ( 0 , 0 , 0 ) ;
if ( mass ! = 0.0 )
{
sphere - > calculateLocalInertia ( ( btScalar ) mass , inertia ) ;
}
glm : : quat glmQuat = glm : : quat_cast ( entity . getRotation ( ) ) ;
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( glmQuat . x , glmQuat . y , glmQuat . z , glmQuat . w ) , btVector3 ( entity . getPosition ( ) . x , entity . getPosition ( ) . y , entity . getPosition ( ) . z ) ) ) ;
btRigidBody : : btRigidBodyConstructionInfo info ( mass , motion , sphere , inertia ) ;
btRigidBody * body = new btRigidBody ( info ) ;
2015-01-06 12:31:53 +00:00
body - > setDamping ( dampningL , dampningA ) ;
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( body , COL_OBJECTS , objectsPhysicsCollision ) ;
bodies . push_back ( body ) ;
2015-02-13 15:09:18 +00:00
if ( ! rotate ) //rotate lets certain objects get inertia (0,0,0) (not rotateable)
{
body - > setAngularFactor ( btVector3 ( 0 , 0 , 0 ) ) ;
}
2014-11-17 15:07:40 +00:00
body - > setSleepingThresholds ( 0 , 0 ) ;
2015-02-13 12:46:41 +00:00
if ( bodies . size ( ) ! = indice )
throw std : : invalid_argument ( " Bodies out of Sync " ) ;
2014-11-14 13:17:04 +00:00
}
2014-11-28 15:46:35 +00:00
2015-03-04 15:01:55 +00:00
void Physics : : prepareCollisionDetection ( )
{
playerTerrainCol = playerObjectColision = false ;
int numManifods = world - > getDispatcher ( ) - > getNumManifolds ( ) ;
}
bool Physics : : playerWithGround ( )
{
return playerTerrainCol ;
}
bool Physics : : playerWithObject ( )
{
return playerObjectColision ;
}
2015-01-30 12:32:16 +00:00
void Physics : : addCamera ( ) //Camera Creator automatically called when player is created
2014-11-28 12:52:03 +00:00
{
2015-03-06 16:45:51 +00:00
btSphereShape * sphere = new btSphereShape ( 1.0f ) ; //we use this to make a more interesting camera, that does not interpenetrate with the terrain/objects
2015-02-13 12:46:41 +00:00
btVector3 inertia ( 0 , 0 , 0 ) ; //rotation handled elsewhere (as it always has to look at the player)
btVector3 direction ( 1 , 1 , 1 ) ;
direction . normalize ( ) ;
2015-02-13 14:20:32 +00:00
direction * = cameraDistance ; //create a offset of length 5 so we have a stable camera at the beginning
2015-02-13 12:46:41 +00:00
btDefaultMotionState * motion = new btDefaultMotionState ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , playerBall - > getCenterOfMassPosition ( ) + direction ) ) ;
btRigidBody : : btRigidBodyConstructionInfo info ( 0.001 , motion , sphere , inertia ) ;
cameraBody = new btRigidBody ( info ) ;
2015-01-30 12:32:16 +00:00
cameraBody - > setDamping ( 0.9 , 0.5 ) ; //this damping factor leaves a relativly smoothe system
2014-11-28 12:52:03 +00:00
2015-02-13 15:34:39 +00:00
info . m_friction = 0 ;
info . m_restitution = 0 ;
2015-02-13 12:46:41 +00:00
world - > addRigidBody ( cameraBody , COL_OBJECTS , objectsPhysicsCollision ) ;
cameraBody - > setGravity ( btVector3 ( 0 , 0 , 0 ) ) ;
cameraBody - > setSleepingThresholds ( 0 , 0 ) ; //very important, otherwise camera may go to sleep, aka not move until next collision
2014-11-28 12:52:03 +00:00
}
2014-12-15 15:02:04 +00:00
//update functions
2014-11-28 12:52:03 +00:00
glm : : vec3 Physics : : getCameraPosition ( )
{
2015-02-13 12:46:41 +00:00
btVector3 origin = cameraBody - > getCenterOfMassPosition ( ) ;
glm : : vec3 save ( origin . getX ( ) , origin . getY ( ) , origin . getZ ( ) ) ;
return save ;
2014-11-28 12:52:03 +00:00
}
2015-02-13 12:46:41 +00:00
glm : : vec3 Physics : : getCameraToPlayer ( ) //returns a glm::vec3 the goes from the camera to the player
2015-01-19 15:51:35 +00:00
{
2015-02-13 12:46:41 +00:00
btVector3 origin = playerBall - > getCenterOfMassPosition ( ) - cameraBody - > getCenterOfMassPosition ( ) ;
glm : : vec3 save ( origin . getX ( ) , origin . getY ( ) , origin . getZ ( ) ) ;
return save ;
2015-01-19 15:51:35 +00:00
}
2015-01-30 12:32:16 +00:00
glm : : vec3 Physics : : getPos ( int i ) //this and the next function are used to synchronize the graphics data and the physics data
2014-11-14 13:17:04 +00:00
{
2015-02-13 12:46:41 +00:00
btVector3 origin = bodies [ i ] - > getCenterOfMassPosition ( ) ;
glm : : vec3 save ( origin . getX ( ) , origin . getY ( ) , origin . getZ ( ) ) ;
return save ;
2014-11-14 13:17:04 +00:00
}
2014-11-17 11:57:16 +00:00
glm : : mat4 Physics : : getRotation ( int i )
2014-11-14 13:17:04 +00:00
{
2015-02-13 12:46:41 +00:00
btQuaternion quat = bodies [ i ] - > getOrientation ( ) ;
glm : : mat4 matrix = glm : : rotate (
quat . getAngle ( ) ,
glm : : vec3 ( quat . getAxis ( ) . getX ( ) , quat . getAxis ( ) . getY ( ) , quat . getAxis ( ) . getZ ( ) )
) ; //somewhat clunky, but basicly creates a rotation matrix out of the angle of the body, and its axis (from the quaterion in bullet)
return matrix ;
2014-11-14 13:17:04 +00:00
}
2015-01-30 12:32:16 +00:00
//these are used to apply a force to the camera body according to the movement of the mouse
2015-02-13 14:20:32 +00:00
void Physics : : updateCameraPos ( glm : : vec2 mouseMovement , float strength , float distance )
2015-01-19 15:30:11 +00:00
{
2015-02-13 14:20:32 +00:00
this - > cameraDistance = distance ;
2015-01-30 12:32:16 +00:00
//note: in mouseMovement x and y are flipped in contrast to bullet
2015-03-04 12:37:27 +00:00
btVector3 dodo = btVector3 ( 0 , 1 , 0 ) . cross ( btVector3 ( currentDirection . x ( ) , 0 , currentDirection . z ( ) ) ) ;
2015-03-06 16:45:51 +00:00
currentDirection = currentDirection . rotate ( dodo , mouseMovement . x / 500 ) ; //mathhelper 3.14159265359
2015-03-04 12:37:27 +00:00
btVector3 compare = currentDirection ;
compare . setY ( 0 ) ;
compare . normalize ( ) ;
if ( currentDirection . angle ( compare ) > = 3.14159265359 / 4 )
{
dodo = btVector3 ( 0 , 1 , 0 ) . cross ( btVector3 ( currentDirection . x ( ) , 0 , currentDirection . z ( ) ) ) ;
compare = compare . rotate ( - dodo , 3.14159265359 / 4 ) ;
if ( currentDirection . y ( ) < 0 )
compare . setY ( - compare . y ( ) ) ;
currentDirection = compare ;
}
2015-03-06 16:45:51 +00:00
currentDirection = currentDirection . rotate ( btVector3 ( 0 , 1 , 0 ) , - mouseMovement . y / 1000 ) ;
2015-03-04 12:37:27 +00:00
currentDirection . normalize ( ) ;
2015-01-19 15:30:11 +00:00
}
2015-01-30 12:32:16 +00:00
//use the crossproduct to correctly apply a torque to the palyer if function called
2014-11-17 15:07:40 +00:00
void Physics : : rollForward ( glm : : vec3 camPos , float strength )
2014-11-14 13:17:04 +00:00
{
2015-03-06 15:56:20 +00:00
if ( ! simulationActive )
return ;
2015-01-29 13:10:28 +00:00
btVector3 pos = cameraBody - > getCenterOfMassPosition ( ) - playerBall - > getCenterOfMassPosition ( ) ;
pos . setY ( 0 ) ;
2014-11-21 11:31:29 +00:00
pos . normalize ( ) ;
2014-11-17 15:07:40 +00:00
pos = btCross ( pos , btVector3 ( 0 , 1 , 0 ) ) ;
2014-11-21 11:31:29 +00:00
pos * = strength ;
2014-11-17 12:12:51 +00:00
playerBall - > applyTorque ( pos ) ;
2014-11-17 15:07:40 +00:00
}
2014-11-17 12:12:51 +00:00
2014-11-17 15:07:40 +00:00
void Physics : : rollBack ( glm : : vec3 camPos , float strength )
{
2015-03-06 15:56:20 +00:00
if ( ! simulationActive )
return ;
2015-01-29 13:10:28 +00:00
btVector3 pos = cameraBody - > getCenterOfMassPosition ( ) - playerBall - > getCenterOfMassPosition ( ) ;
pos . setY ( 0 ) ;
2014-11-21 11:31:29 +00:00
pos . normalize ( ) ;
2014-11-17 15:07:40 +00:00
pos = btCross ( btVector3 ( 0 , 1 , 0 ) , pos ) ;
2014-11-21 11:31:29 +00:00
pos * = strength ;
2014-11-17 15:07:40 +00:00
playerBall - > applyTorque ( pos ) ;
}
void Physics : : rollLeft ( glm : : vec3 camPos , float strength )
{
2015-03-06 15:56:20 +00:00
if ( ! simulationActive )
return ;
2015-01-29 13:10:28 +00:00
btVector3 pos = cameraBody - > getCenterOfMassPosition ( ) - playerBall - > getCenterOfMassPosition ( ) ;
pos . setY ( 0 ) ;
2014-11-21 11:31:29 +00:00
pos . normalize ( ) ;
pos * = strength ;
2014-11-17 15:07:40 +00:00
playerBall - > applyTorque ( pos ) ;
}
void Physics : : rollRight ( glm : : vec3 camPos , float strength )
{
2015-03-06 15:56:20 +00:00
if ( ! simulationActive )
return ;
2015-01-29 13:10:28 +00:00
btVector3 pos = cameraBody - > getCenterOfMassPosition ( ) - playerBall - > getCenterOfMassPosition ( ) ;
pos . setY ( 0 ) ;
2014-11-21 11:31:29 +00:00
pos . normalize ( ) ;
pos * = strength ;
2014-11-17 15:07:40 +00:00
playerBall - > applyTorque ( - pos ) ;
2014-11-14 13:17:04 +00:00
}
2015-01-19 15:51:35 +00:00
2015-01-16 15:17:26 +00:00
//not used right now
void Physics : : addStaticGroundPlane ( )
{
2015-02-13 12:46:41 +00:00
btCollisionShape * groundShape = new btStaticPlaneShape ( btVector3 ( 0 , 1 , 0 ) , 0 ) ;
btDefaultMotionState * groundMotionState = new btDefaultMotionState ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , btVector3 ( 0 , 0 , 0 ) ) ) ;
btRigidBody : : btRigidBodyConstructionInfo groundRigidBodyCI ( 0 , groundMotionState , groundShape , btVector3 ( 0 , 0 , 0 ) ) ;
staticGroundBody = new btRigidBody ( groundRigidBodyCI ) ;
world - > addRigidBody ( staticGroundBody ) ;
} //not needed anymoer, but still good for debugging
2015-01-30 12:32:16 +00:00
2015-01-16 15:17:26 +00:00
2015-02-13 16:22:01 +00:00
2015-02-13 16:47:21 +00:00
void Physics : : forceMove ( glm : : vec3 newPosition , unsigned indice ) //ugly, but needed for reset
2015-02-13 16:22:01 +00:00
{
bodies [ indice ] - > setCenterOfMassTransform ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , btVector3 ( newPosition . x , newPosition . y , newPosition . z ) ) ) ;
2015-03-04 15:01:55 +00:00
bodies [ indice ] - > setLinearVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
2015-03-06 14:01:22 +00:00
bodies [ indice ] - > setAngularVelocity ( btVector3 ( 0 , 0 , 0 ) ) ;
2015-03-04 15:01:55 +00:00
}
void Physics : : forcePlayer ( glm : : vec3 newPosition ) //ugly, but needed for reset
{
2015-03-06 14:01:22 +00:00
if ( ! simulationActive )
return ;
simulationActive = false ;
resetHight = playerBall - > getCenterOfMassPosition ( ) . y ( ) ;
startPosition = btVector3 ( newPosition . x , newPosition . y , newPosition . z ) ;
2015-02-13 16:22:01 +00:00
}
2015-03-06 14:01:22 +00:00
void Physics : : forceMoveCamera ( btVector3 newPosition )
2015-02-13 16:47:21 +00:00
{
2015-03-06 14:01:22 +00:00
cameraBody - > setCenterOfMassTransform ( btTransform ( btQuaternion ( 0 , 0 , 0 , 1 ) , newPosition ) ) ;
2015-02-13 16:47:21 +00:00
}
2015-03-06 15:19:57 +00:00
void Physics : : activateEndgame ( )
{
if ( endgame )
return ;
endgame = true ;
positionConstraint cons ;
cons . body = playerBall ;
cons . strength = 1 ;
cons . position = playerBall - > getCenterOfMassPosition ( ) + btVector3 ( 0 , 15 , 0 ) ;
playerBall - > setGravity ( btVector3 ( 0 , 0 , 0 ) ) ;
allPositionConstraints . push_back ( cons ) ;
}
2015-02-13 12:46:41 +00:00
void Physics : : kill ( ) //delete dynamically allocated memory
2014-11-14 13:17:04 +00:00
{
2015-02-06 17:00:14 +00:00
if ( world = = NULL ) {
return ;
}
2015-02-13 12:46:41 +00:00
//btDynamimcWorld*
for ( unsigned i = 0 ; i < bodies . size ( ) ; i + + )
{
world - > removeCollisionObject ( bodies [ i ] ) ; //go through the list of bodies in world for each body b, then remove exactly this body b from world
btMotionState * motionState = bodies [ i ] - > getMotionState ( ) ;
btCollisionShape * shape = bodies [ i ] - > getCollisionShape ( ) ;
delete shape ;
delete motionState ;
delete bodies [ i ] ;
}
btMotionState * motionState = terrainBody - > getMotionState ( ) ; //delete the rest that are not in the array bodies
btCollisionShape * shape = terrainBody - > getCollisionShape ( ) ;
delete shape ;
delete motionState ;
delete terrainBody ;
motionState = cameraBody - > getMotionState ( ) ;
shape = cameraBody - > getCollisionShape ( ) ;
delete shape ;
delete motionState ;
delete cameraBody ; //note: palyerBall is also in the array bodies so we do not need to clean it up
delete dispatcher ; //clean up rest
delete colConfig ;
delete solver ;
delete broadphase ;
delete world ;
//feel like a good little programmer because everything is clean
2014-11-14 13:17:04 +00:00
}
2015-03-13 14:30:21 +00:00
btDynamicsWorld * Physics : : getWorld ( ) {
return world ;
}