#include "body.h"
#include "aabb.h"
#include "array.h"
#include "contact.h"
#include "core.h"
#include "id_pool.h"
#include "island.h"
#include "joint.h"
#include "physics_world.h"
#include "sensor.h"
#include "shape.h"
#include "solver_set.h"
#include "box2d/box2d.h"
#include "box2d/id.h"
#include <string.h>
B2_ARRAY_SOURCE( b2Body, b2Body )
B2_ARRAY_SOURCE( b2BodySim, b2BodySim )
B2_ARRAY_SOURCE( b2BodyState, b2BodyState )
static void b2LimitVelocity( b2BodyState* state, float maxLinearSpeed )
{
float v2 = b2LengthSquared( state->linearVelocity );
if ( v2 > maxLinearSpeed * maxLinearSpeed )
{
state->linearVelocity = b2MulSV( maxLinearSpeed / sqrtf( v2 ), state->linearVelocity );
}
}
b2Body* b2GetBodyFullId( b2World* world, b2BodyId bodyId )
{
B2_ASSERT( b2Body_IsValid( bodyId ) );
return b2BodyArray_Get( &world->bodies, bodyId.index1 - 1 );
}
b2Transform b2GetBodyTransformQuick( b2World* world, b2Body* body )
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, body->localIndex );
return bodySim->transform;
}
b2Transform b2GetBodyTransform( b2World* world, int bodyId )
{
b2Body* body = b2BodyArray_Get( &world->bodies, bodyId );
return b2GetBodyTransformQuick( world, body );
}
b2BodyId b2MakeBodyId( b2World* world, int bodyId )
{
b2Body* body = b2BodyArray_Get( &world->bodies, bodyId );
return (b2BodyId){ bodyId + 1, world->worldId, body->generation };
}
b2BodySim* b2GetBodySim( b2World* world, b2Body* body )
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, body->localIndex );
return bodySim;
}
b2BodyState* b2GetBodyState( b2World* world, b2Body* body )
{
if ( body->setIndex == b2_awakeSet )
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_awakeSet );
return b2BodyStateArray_Get( &set->bodyStates, body->localIndex );
}
return NULL;
}
static void b2CreateIslandForBody( b2World* world, int setIndex, b2Body* body )
{
B2_ASSERT( body->islandId == B2_NULL_INDEX );
B2_ASSERT( body->islandPrev == B2_NULL_INDEX );
B2_ASSERT( body->islandNext == B2_NULL_INDEX );
B2_ASSERT( setIndex != b2_disabledSet );
b2Island* island = b2CreateIsland( world, setIndex );
body->islandId = island->islandId;
island->headBody = body->id;
island->tailBody = body->id;
island->bodyCount = 1;
}
static void b2RemoveBodyFromIsland( b2World* world, b2Body* body )
{
if ( body->islandId == B2_NULL_INDEX )
{
B2_ASSERT( body->islandPrev == B2_NULL_INDEX );
B2_ASSERT( body->islandNext == B2_NULL_INDEX );
return;
}
int islandId = body->islandId;
b2Island* island = b2IslandArray_Get( &world->islands, islandId );
if ( body->islandPrev != B2_NULL_INDEX )
{
b2Body* prevBody = b2BodyArray_Get( &world->bodies, body->islandPrev );
prevBody->islandNext = body->islandNext;
}
if ( body->islandNext != B2_NULL_INDEX )
{
b2Body* nextBody = b2BodyArray_Get( &world->bodies, body->islandNext );
nextBody->islandPrev = body->islandPrev;
}
B2_ASSERT( island->bodyCount > 0 );
island->bodyCount -= 1;
bool islandDestroyed = false;
if ( island->headBody == body->id )
{
island->headBody = body->islandNext;
if ( island->headBody == B2_NULL_INDEX )
{
B2_ASSERT( island->tailBody == body->id );
B2_ASSERT( island->bodyCount == 0 );
B2_ASSERT( island->contactCount == 0 );
B2_ASSERT( island->jointCount == 0 );
b2DestroyIsland( world, island->islandId );
islandDestroyed = true;
}
}
else if ( island->tailBody == body->id )
{
island->tailBody = body->islandPrev;
}
if ( islandDestroyed == false )
{
b2ValidateIsland( world, islandId );
}
body->islandId = B2_NULL_INDEX;
body->islandPrev = B2_NULL_INDEX;
body->islandNext = B2_NULL_INDEX;
}
static void b2DestroyBodyContacts( b2World* world, b2Body* body, bool wakeBodies )
{
int edgeKey = body->headContactKey;
while ( edgeKey != B2_NULL_INDEX )
{
int contactId = edgeKey >> 1;
int edgeIndex = edgeKey & 1;
b2Contact* contact = b2ContactArray_Get( &world->contacts, contactId );
edgeKey = contact->edges[edgeIndex].nextKey;
b2DestroyContact( world, contact, wakeBodies );
}
b2ValidateSolverSets( world );
}
b2BodyId b2CreateBody( b2WorldId worldId, const b2BodyDef* def )
{
B2_CHECK_DEF( def );
B2_ASSERT( b2IsValidVec2( def->position ) );
B2_ASSERT( b2IsValidRotation( def->rotation ) );
B2_ASSERT( b2IsValidVec2( def->linearVelocity ) );
B2_ASSERT( b2IsValidFloat( def->angularVelocity ) );
B2_ASSERT( b2IsValidFloat( def->linearDamping ) && def->linearDamping >= 0.0f );
B2_ASSERT( b2IsValidFloat( def->angularDamping ) && def->angularDamping >= 0.0f );
B2_ASSERT( b2IsValidFloat( def->sleepThreshold ) && def->sleepThreshold >= 0.0f );
B2_ASSERT( b2IsValidFloat( def->gravityScale ) );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return b2_nullBodyId;
}
bool isAwake = ( def->isAwake || def->enableSleep == false ) && def->isEnabled;
int setId;
if ( def->isEnabled == false )
{
setId = b2_disabledSet;
}
else if ( def->type == b2_staticBody )
{
setId = b2_staticSet;
}
else if ( isAwake == true )
{
setId = b2_awakeSet;
}
else
{
setId = b2AllocId( &world->solverSetIdPool );
if ( setId == world->solverSets.count )
{
b2SolverSetArray_Push( &world->solverSets, (b2SolverSet){ 0 } );
}
else
{
B2_ASSERT( world->solverSets.data[setId].setIndex == B2_NULL_INDEX );
}
world->solverSets.data[setId].setIndex = setId;
}
B2_ASSERT( 0 <= setId && setId < world->solverSets.count );
int bodyId = b2AllocId( &world->bodyIdPool );
uint32_t lockFlags = 0;
lockFlags |= def->motionLocks.linearX ? b2_lockLinearX : 0;
lockFlags |= def->motionLocks.linearY ? b2_lockLinearY : 0;
lockFlags |= def->motionLocks.angularZ ? b2_lockAngularZ : 0;
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, setId );
b2BodySim* bodySim = b2BodySimArray_Add( &set->bodySims );
*bodySim = (b2BodySim){ 0 };
bodySim->transform.p = def->position;
bodySim->transform.q = def->rotation;
bodySim->center = def->position;
bodySim->rotation0 = bodySim->transform.q;
bodySim->center0 = bodySim->center;
bodySim->minExtent = B2_HUGE;
bodySim->maxExtent = 0.0f;
bodySim->linearDamping = def->linearDamping;
bodySim->angularDamping = def->angularDamping;
bodySim->gravityScale = def->gravityScale;
bodySim->bodyId = bodyId;
bodySim->flags = lockFlags;
bodySim->flags |= def->isBullet ? b2_isBullet : 0;
bodySim->flags |= def->allowFastRotation ? b2_allowFastRotation : 0;
bodySim->flags |= def->type == b2_dynamicBody ? b2_dynamicFlag : 0;
if ( setId == b2_awakeSet )
{
b2BodyState* bodyState = b2BodyStateArray_Add( &set->bodyStates );
B2_ASSERT( ( (uintptr_t)bodyState & 0x1F ) == 0 );
*bodyState = (b2BodyState){ 0 };
bodyState->linearVelocity = def->linearVelocity;
bodyState->angularVelocity = def->angularVelocity;
bodyState->deltaRotation = b2Rot_identity;
bodyState->flags = bodySim->flags;
}
if ( bodyId == world->bodies.count )
{
b2BodyArray_Push( &world->bodies, (b2Body){ 0 } );
}
else
{
B2_ASSERT( world->bodies.data[bodyId].id == B2_NULL_INDEX );
}
b2Body* body = b2BodyArray_Get( &world->bodies, bodyId );
if ( def->name )
{
int i = 0;
while ( i < B2_NAME_LENGTH - 1 && def->name[i] != 0 )
{
body->name[i] = def->name[i];
i += 1;
}
while ( i < B2_NAME_LENGTH )
{
body->name[i] = 0;
i += 1;
}
}
else
{
memset( body->name, 0, B2_NAME_LENGTH * sizeof( char ) );
}
body->userData = def->userData;
body->setIndex = setId;
body->localIndex = set->bodySims.count - 1;
body->generation += 1;
body->headShapeId = B2_NULL_INDEX;
body->shapeCount = 0;
body->headChainId = B2_NULL_INDEX;
body->headContactKey = B2_NULL_INDEX;
body->contactCount = 0;
body->headJointKey = B2_NULL_INDEX;
body->jointCount = 0;
body->islandId = B2_NULL_INDEX;
body->islandPrev = B2_NULL_INDEX;
body->islandNext = B2_NULL_INDEX;
body->bodyMoveIndex = B2_NULL_INDEX;
body->id = bodyId;
body->mass = 0.0f;
body->inertia = 0.0f;
body->sleepThreshold = def->sleepThreshold;
body->sleepTime = 0.0f;
body->type = def->type;
body->flags = bodySim->flags;
body->enableSleep = def->enableSleep;
if ( setId >= b2_awakeSet )
{
b2CreateIslandForBody( world, setId, body );
}
b2ValidateSolverSets( world );
b2BodyId id = { bodyId + 1, world->worldId, body->generation };
return id;
}
bool b2WakeBody( b2World* world, b2Body* body )
{
if ( body->setIndex >= b2_firstSleepingSet )
{
b2WakeSolverSet( world, body->setIndex );
b2ValidateSolverSets( world );
return true;
}
return false;
}
void b2DestroyBody( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
bool wakeBodies = true;
int edgeKey = body->headJointKey;
while ( edgeKey != B2_NULL_INDEX )
{
int jointId = edgeKey >> 1;
int edgeIndex = edgeKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
edgeKey = joint->edges[edgeIndex].nextKey;
b2DestroyJointInternal( world, joint, wakeBodies );
}
b2DestroyBodyContacts( world, body, wakeBodies );
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
if ( shape->sensorIndex != B2_NULL_INDEX )
{
b2DestroySensor( world, shape );
}
b2DestroyShapeProxy( shape, &world->broadPhase );
b2FreeId( &world->shapeIdPool, shapeId );
shape->id = B2_NULL_INDEX;
shapeId = shape->nextShapeId;
}
int chainId = body->headChainId;
while ( chainId != B2_NULL_INDEX )
{
b2ChainShape* chain = b2ChainShapeArray_Get( &world->chainShapes, chainId );
b2FreeChainData( chain );
b2FreeId( &world->chainIdPool, chainId );
chain->id = B2_NULL_INDEX;
chainId = chain->nextChainId;
}
b2RemoveBodyFromIsland( world, body );
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
int movedIndex = b2BodySimArray_RemoveSwap( &set->bodySims, body->localIndex );
if ( movedIndex != B2_NULL_INDEX )
{
b2BodySim* movedSim = set->bodySims.data + body->localIndex;
int movedId = movedSim->bodyId;
b2Body* movedBody = b2BodyArray_Get( &world->bodies, movedId );
B2_ASSERT( movedBody->localIndex == movedIndex );
movedBody->localIndex = body->localIndex;
}
if ( body->setIndex == b2_awakeSet )
{
int result = b2BodyStateArray_RemoveSwap( &set->bodyStates, body->localIndex );
B2_ASSERT( result == movedIndex );
B2_UNUSED( result );
}
else if ( set->setIndex >= b2_firstSleepingSet && set->bodySims.count == 0 )
{
b2DestroySolverSet( world, set->setIndex );
}
b2FreeId( &world->bodyIdPool, body->id );
body->setIndex = B2_NULL_INDEX;
body->localIndex = B2_NULL_INDEX;
body->id = B2_NULL_INDEX;
b2ValidateSolverSets( world );
}
int b2Body_GetContactCapacity( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return 0;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->contactCount;
}
int b2Body_GetContactData( b2BodyId bodyId, b2ContactData* contactData, int capacity )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return 0;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
int contactKey = body->headContactKey;
int index = 0;
while ( contactKey != B2_NULL_INDEX && index < capacity )
{
int contactId = contactKey >> 1;
int edgeIndex = contactKey & 1;
b2Contact* contact = b2ContactArray_Get( &world->contacts, contactId );
if ( contact->flags & b2_contactTouchingFlag )
{
b2Shape* shapeA = b2ShapeArray_Get( &world->shapes, contact->shapeIdA );
b2Shape* shapeB = b2ShapeArray_Get( &world->shapes, contact->shapeIdB );
contactData[index].contactId = (b2ContactId){ contact->contactId + 1, bodyId.world0, 0, contact->generation };
contactData[index].shapeIdA = (b2ShapeId){ shapeA->id + 1, bodyId.world0, shapeA->generation };
contactData[index].shapeIdB = (b2ShapeId){ shapeB->id + 1, bodyId.world0, shapeB->generation };
b2ContactSim* contactSim = b2GetContactSim( world, contact );
contactData[index].manifold = contactSim->manifold;
index += 1;
}
contactKey = contact->edges[edgeIndex].nextKey;
}
B2_ASSERT( index <= capacity );
return index;
}
b2AABB b2Body_ComputeAABB( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return (b2AABB){ 0 };
}
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->headShapeId == B2_NULL_INDEX )
{
b2Transform transform = b2GetBodyTransform( world, body->id );
return (b2AABB){ transform.p, transform.p };
}
b2Shape* shape = b2ShapeArray_Get( &world->shapes, body->headShapeId );
b2AABB aabb = shape->aabb;
while ( shape->nextShapeId != B2_NULL_INDEX )
{
shape = b2ShapeArray_Get( &world->shapes, shape->nextShapeId );
aabb = b2AABB_Union( aabb, shape->aabb );
}
return aabb;
}
void b2UpdateBodyMassData( b2World* world, b2Body* body )
{
b2BodySim* bodySim = b2GetBodySim( world, body );
body->flags &= ~b2_dirtyMass;
body->mass = 0.0f;
body->inertia = 0.0f;
bodySim->invMass = 0.0f;
bodySim->invInertia = 0.0f;
bodySim->localCenter = b2Vec2_zero;
bodySim->minExtent = B2_HUGE;
bodySim->maxExtent = 0.0f;
if ( body->type != b2_dynamicBody )
{
bodySim->center = bodySim->transform.p;
bodySim->center0 = bodySim->center;
if ( body->type == b2_kinematicBody )
{
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
const b2Shape* s = b2ShapeArray_Get( &world->shapes, shapeId );
b2ShapeExtent extent = b2ComputeShapeExtent( s, b2Vec2_zero );
bodySim->minExtent = b2MinFloat( bodySim->minExtent, extent.minExtent );
bodySim->maxExtent = b2MaxFloat( bodySim->maxExtent, extent.maxExtent );
shapeId = s->nextShapeId;
}
}
return;
}
int shapeCount = body->shapeCount;
b2MassData* masses = b2AllocateArenaItem( &world->arena, shapeCount * sizeof( b2MassData ), "mass data" );
b2Vec2 localCenter = b2Vec2_zero;
int shapeId = body->headShapeId;
int shapeIndex = 0;
while ( shapeId != B2_NULL_INDEX )
{
const b2Shape* s = b2ShapeArray_Get( &world->shapes, shapeId );
shapeId = s->nextShapeId;
if ( s->density == 0.0f )
{
masses[shapeIndex] = (b2MassData){ 0 };
shapeIndex += 1;
continue;
}
b2MassData massData = b2ComputeShapeMass( s );
body->mass += massData.mass;
localCenter = b2MulAdd( localCenter, massData.mass, massData.center );
masses[shapeIndex] = massData;
shapeIndex += 1;
}
if ( body->mass > 0.0f )
{
bodySim->invMass = 1.0f / body->mass;
localCenter = b2MulSV( bodySim->invMass, localCenter );
}
for ( shapeIndex = 0; shapeIndex < shapeCount; ++shapeIndex )
{
b2MassData massData = masses[shapeIndex];
if ( massData.mass == 0.0f )
{
continue;
}
b2Vec2 offset = b2Sub( localCenter, massData.center );
float inertia = massData.rotationalInertia + massData.mass * b2Dot( offset, offset );
body->inertia += inertia;
}
b2FreeArenaItem( &world->arena, masses );
masses = NULL;
B2_ASSERT( body->inertia >= 0.0f );
if ( body->inertia > 0.0f )
{
bodySim->invInertia = 1.0f / body->inertia;
}
else
{
body->inertia = 0.0f;
bodySim->invInertia = 0.0f;
}
b2Vec2 oldCenter = bodySim->center;
bodySim->localCenter = localCenter;
bodySim->center = b2TransformPoint( bodySim->transform, bodySim->localCenter );
bodySim->center0 = bodySim->center;
b2BodyState* state = b2GetBodyState( world, body );
if ( state != NULL )
{
b2Vec2 deltaLinear = b2CrossSV( state->angularVelocity, b2Sub( bodySim->center, oldCenter ) );
state->linearVelocity = b2Add( state->linearVelocity, deltaLinear );
}
shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
const b2Shape* s = b2ShapeArray_Get( &world->shapes, shapeId );
b2ShapeExtent extent = b2ComputeShapeExtent( s, localCenter );
bodySim->minExtent = b2MinFloat( bodySim->minExtent, extent.minExtent );
bodySim->maxExtent = b2MaxFloat( bodySim->maxExtent, extent.maxExtent );
shapeId = s->nextShapeId;
}
}
b2Vec2 b2Body_GetPosition( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return transform.p;
}
b2Rot b2Body_GetRotation( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return transform.q;
}
b2Transform b2Body_GetTransform( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return b2GetBodyTransformQuick( world, body );
}
b2Vec2 b2Body_GetLocalPoint( b2BodyId bodyId, b2Vec2 worldPoint )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return b2InvTransformPoint( transform, worldPoint );
}
b2Vec2 b2Body_GetWorldPoint( b2BodyId bodyId, b2Vec2 localPoint )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return b2TransformPoint( transform, localPoint );
}
b2Vec2 b2Body_GetLocalVector( b2BodyId bodyId, b2Vec2 worldVector )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return b2InvRotateVector( transform.q, worldVector );
}
b2Vec2 b2Body_GetWorldVector( b2BodyId bodyId, b2Vec2 localVector )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2Transform transform = b2GetBodyTransformQuick( world, body );
return b2RotateVector( transform.q, localVector );
}
void b2Body_SetTransform( b2BodyId bodyId, b2Vec2 position, b2Rot rotation )
{
B2_ASSERT( b2IsValidVec2( position ) );
B2_ASSERT( b2IsValidRotation( rotation ) );
B2_ASSERT( b2Body_IsValid( bodyId ) );
b2World* world = b2GetWorld( bodyId.world0 );
B2_ASSERT( world->locked == false );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->transform.p = position;
bodySim->transform.q = rotation;
bodySim->center = b2TransformPoint( bodySim->transform, bodySim->localCenter );
bodySim->rotation0 = bodySim->transform.q;
bodySim->center0 = bodySim->center;
b2BroadPhase* broadPhase = &world->broadPhase;
b2Transform transform = bodySim->transform;
const float margin = B2_AABB_MARGIN;
const float speculativeDistance = B2_SPECULATIVE_DISTANCE;
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
b2AABB aabb = b2ComputeShapeAABB( shape, transform );
aabb.lowerBound.x -= speculativeDistance;
aabb.lowerBound.y -= speculativeDistance;
aabb.upperBound.x += speculativeDistance;
aabb.upperBound.y += speculativeDistance;
shape->aabb = aabb;
if ( b2AABB_Contains( shape->fatAABB, aabb ) == false )
{
b2AABB fatAABB;
fatAABB.lowerBound.x = aabb.lowerBound.x - margin;
fatAABB.lowerBound.y = aabb.lowerBound.y - margin;
fatAABB.upperBound.x = aabb.upperBound.x + margin;
fatAABB.upperBound.y = aabb.upperBound.y + margin;
shape->fatAABB = fatAABB;
if ( shape->proxyKey != B2_NULL_INDEX )
{
b2BroadPhase_MoveProxy( broadPhase, shape->proxyKey, fatAABB );
}
}
shapeId = shape->nextShapeId;
}
}
b2Vec2 b2Body_GetLinearVelocity( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodyState* state = b2GetBodyState( world, body );
if ( state != NULL )
{
return state->linearVelocity;
}
return b2Vec2_zero;
}
float b2Body_GetAngularVelocity( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodyState* state = b2GetBodyState( world, body );
if ( state != NULL )
{
return state->angularVelocity;
}
return 0.0;
}
void b2Body_SetLinearVelocity( b2BodyId bodyId, b2Vec2 linearVelocity )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type == b2_staticBody )
{
return;
}
if ( b2LengthSquared( linearVelocity ) > 0.0f )
{
b2WakeBody( world, body );
}
b2BodyState* state = b2GetBodyState( world, body );
if ( state == NULL )
{
return;
}
state->linearVelocity = linearVelocity;
}
void b2Body_SetAngularVelocity( b2BodyId bodyId, float angularVelocity )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type == b2_staticBody || ( body->flags & b2_lockAngularZ ) )
{
return;
}
if ( angularVelocity != 0.0f )
{
b2WakeBody( world, body );
}
b2BodyState* state = b2GetBodyState( world, body );
if ( state == NULL )
{
return;
}
state->angularVelocity = angularVelocity;
}
void b2Body_SetTargetTransform( b2BodyId bodyId, b2Transform target, float timeStep, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->setIndex == b2_disabledSet )
{
return;
}
if ( body->type == b2_staticBody || timeStep <= 0.0f )
{
return;
}
if ( body->setIndex != b2_awakeSet && wake == false )
{
return;
}
b2BodySim* sim = b2GetBodySim( world, body );
b2Vec2 center1 = sim->center;
b2Vec2 center2 = b2TransformPoint( target, sim->localCenter );
float invTimeStep = 1.0f / timeStep;
b2Vec2 linearVelocity = b2MulSV( invTimeStep, b2Sub( center2, center1 ) );
b2Rot q1 = sim->transform.q;
b2Rot q2 = target.q;
float deltaAngle = b2RelativeAngle( q1, q2 );
float angularVelocity = invTimeStep * deltaAngle;
if ( body->setIndex != b2_awakeSet )
{
float maxVelocity = b2Length( linearVelocity ) + b2AbsFloat( angularVelocity ) * sim->maxExtent;
if ( maxVelocity < body->sleepThreshold )
{
return;
}
b2WakeBody( world, body );
}
B2_ASSERT( body->setIndex == b2_awakeSet );
b2BodyState* state = b2GetBodyState( world, body );
state->linearVelocity = linearVelocity;
state->angularVelocity = angularVelocity;
}
b2Vec2 b2Body_GetLocalPointVelocity( b2BodyId bodyId, b2Vec2 localPoint )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodyState* state = b2GetBodyState( world, body );
if ( state == NULL )
{
return b2Vec2_zero;
}
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, body->localIndex );
b2Vec2 r = b2RotateVector( bodySim->transform.q, b2Sub( localPoint, bodySim->localCenter ) );
b2Vec2 v = b2Add( state->linearVelocity, b2CrossSV( state->angularVelocity, r ) );
return v;
}
b2Vec2 b2Body_GetWorldPointVelocity( b2BodyId bodyId, b2Vec2 worldPoint )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodyState* state = b2GetBodyState( world, body );
if ( state == NULL )
{
return b2Vec2_zero;
}
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, body->localIndex );
b2Vec2 r = b2Sub( worldPoint, bodySim->center );
b2Vec2 v = b2Add( state->linearVelocity, b2CrossSV( state->angularVelocity, r ) );
return v;
}
void b2Body_ApplyForce( b2BodyId bodyId, b2Vec2 force, b2Vec2 point, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->force = b2Add( bodySim->force, force );
bodySim->torque += b2Cross( b2Sub( point, bodySim->center ), force );
}
}
void b2Body_ApplyForceToCenter( b2BodyId bodyId, b2Vec2 force, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->force = b2Add( bodySim->force, force );
}
}
void b2Body_ApplyTorque( b2BodyId bodyId, float torque, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->torque += torque;
}
}
void b2Body_ClearForces( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->force = b2Vec2_zero;
bodySim->torque = 0.0f;
}
void b2Body_ApplyLinearImpulse( b2BodyId bodyId, b2Vec2 impulse, b2Vec2 point, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
int localIndex = body->localIndex;
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_awakeSet );
b2BodyState* state = b2BodyStateArray_Get( &set->bodyStates, localIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, localIndex );
state->linearVelocity = b2MulAdd( state->linearVelocity, bodySim->invMass, impulse );
state->angularVelocity += bodySim->invInertia * b2Cross( b2Sub( point, bodySim->center ), impulse );
b2LimitVelocity( state, world->maxLinearSpeed );
}
}
void b2Body_ApplyLinearImpulseToCenter( b2BodyId bodyId, b2Vec2 impulse, bool wake )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
int localIndex = body->localIndex;
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_awakeSet );
b2BodyState* state = b2BodyStateArray_Get( &set->bodyStates, localIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, localIndex );
state->linearVelocity = b2MulAdd( state->linearVelocity, bodySim->invMass, impulse );
b2LimitVelocity( state, world->maxLinearSpeed );
}
}
void b2Body_ApplyAngularImpulse( b2BodyId bodyId, float impulse, bool wake )
{
B2_ASSERT( b2Body_IsValid( bodyId ) );
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->type != b2_dynamicBody || body->setIndex == b2_disabledSet )
{
return;
}
if ( wake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
if ( body->setIndex == b2_awakeSet )
{
int localIndex = body->localIndex;
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_awakeSet );
b2BodyState* state = b2BodyStateArray_Get( &set->bodyStates, localIndex );
b2BodySim* bodySim = b2BodySimArray_Get( &set->bodySims, localIndex );
state->angularVelocity += bodySim->invInertia * impulse;
}
}
b2BodyType b2Body_GetType( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->type;
}
void b2Body_SetType( b2BodyId bodyId, b2BodyType type )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodyType originalType = body->type;
if ( originalType == type )
{
return;
}
if ( body->setIndex == b2_disabledSet )
{
body->type = type;
if ( type == b2_dynamicBody )
{
body->flags |= b2_dynamicFlag;
}
else
{
body->flags &= ~b2_dynamicFlag;
}
b2UpdateBodyMassData( world, body );
return;
}
bool wakeBodies = false;
b2DestroyBodyContacts( world, body, wakeBodies );
b2WakeBody( world, body );
b2SolverSet* staticSet = b2SolverSetArray_Get( &world->solverSets, b2_staticSet );
int jointKey = body->headJointKey;
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
jointKey = joint->edges[edgeIndex].nextKey;
if ( joint->setIndex == b2_disabledSet )
{
continue;
}
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
b2WakeBody( world, bodyA );
b2WakeBody( world, bodyB );
b2UnlinkJoint( world, joint );
b2SolverSet* jointSourceSet = b2SolverSetArray_Get( &world->solverSets, joint->setIndex );
b2TransferJoint( world, staticSet, jointSourceSet, joint );
}
body->type = type;
if ( type == b2_dynamicBody )
{
body->flags |= b2_dynamicFlag;
}
else
{
body->flags &= ~b2_dynamicFlag;
}
b2SolverSet* awakeSet = b2SolverSetArray_Get( &world->solverSets, b2_awakeSet );
b2SolverSet* sourceSet = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2SolverSet* targetSet = type == b2_staticBody ? staticSet : awakeSet;
b2TransferBody( world, targetSet, sourceSet, body );
if ( originalType == b2_staticBody )
{
b2CreateIslandForBody( world, b2_awakeSet, body );
}
else if ( type == b2_staticBody )
{
b2RemoveBodyFromIsland( world, body );
}
jointKey = body->headJointKey;
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
jointKey = joint->edges[edgeIndex].nextKey;
if ( joint->setIndex == b2_disabledSet )
{
continue;
}
B2_ASSERT( joint->setIndex == b2_staticSet );
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
B2_ASSERT( bodyA->setIndex == b2_staticSet || bodyA->setIndex == b2_awakeSet );
B2_ASSERT( bodyB->setIndex == b2_staticSet || bodyB->setIndex == b2_awakeSet );
if ( bodyA->type == b2_dynamicBody || bodyB->type == b2_dynamicBody )
{
b2TransferJoint( world, awakeSet, staticSet, joint );
}
}
b2Transform transform = b2GetBodyTransformQuick( world, body );
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
shapeId = shape->nextShapeId;
b2DestroyShapeProxy( shape, &world->broadPhase );
bool forcePairCreation = true;
b2CreateShapeProxy( shape, &world->broadPhase, type, transform, forcePairCreation );
}
jointKey = body->headJointKey;
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
jointKey = joint->edges[edgeIndex].nextKey;
int otherEdgeIndex = edgeIndex ^ 1;
int otherBodyId = joint->edges[otherEdgeIndex].bodyId;
b2Body* otherBody = b2BodyArray_Get( &world->bodies, otherBodyId );
if ( otherBody->setIndex == b2_disabledSet )
{
continue;
}
if ( body->type != b2_dynamicBody && otherBody->type != b2_dynamicBody )
{
continue;
}
b2LinkJoint( world, joint );
}
b2UpdateBodyMassData( world, body );
b2BodyState* state = b2GetBodyState( world, body );
if ( state != NULL )
{
state->flags = body->flags;
}
b2ValidateSolverSets( world );
b2ValidateIsland( world, body->islandId );
}
void b2Body_SetName( b2BodyId bodyId, const char* name )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( name )
{
int i = 0;
while ( i < B2_NAME_LENGTH - 1 && name[i] != 0 )
{
body->name[i] = name[i];
i += 1;
}
while ( i < B2_NAME_LENGTH )
{
body->name[i] = 0;
i += 1;
}
}
else
{
memset( body->name, 0, B2_NAME_LENGTH * sizeof( char ) );
}
}
const char* b2Body_GetName( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->name;
}
void b2Body_SetUserData( b2BodyId bodyId, void* userData )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
body->userData = userData;
}
void* b2Body_GetUserData( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->userData;
}
float b2Body_GetMass( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->mass;
}
float b2Body_GetRotationalInertia( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->inertia;
}
b2Vec2 b2Body_GetLocalCenterOfMass( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return bodySim->localCenter;
}
b2Vec2 b2Body_GetWorldCenterOfMass( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return bodySim->center;
}
void b2Body_SetMassData( b2BodyId bodyId, b2MassData massData )
{
B2_ASSERT( b2IsValidFloat( massData.mass ) && massData.mass >= 0.0f );
B2_ASSERT( b2IsValidFloat( massData.rotationalInertia ) && massData.rotationalInertia >= 0.0f );
B2_ASSERT( b2IsValidVec2( massData.center ) );
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
body->mass = massData.mass;
body->inertia = massData.rotationalInertia;
bodySim->localCenter = massData.center;
b2Vec2 center = b2TransformPoint( bodySim->transform, massData.center );
bodySim->center = center;
bodySim->center0 = center;
bodySim->invMass = body->mass > 0.0f ? 1.0f / body->mass : 0.0f;
bodySim->invInertia = body->inertia > 0.0f ? 1.0f / body->inertia : 0.0f;
}
b2MassData b2Body_GetMassData( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
b2MassData massData = { body->mass, bodySim->localCenter, body->inertia };
return massData;
}
void b2Body_ApplyMassFromShapes( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2UpdateBodyMassData( world, body );
}
void b2Body_SetLinearDamping( b2BodyId bodyId, float linearDamping )
{
B2_ASSERT( b2IsValidFloat( linearDamping ) && linearDamping >= 0.0f );
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->linearDamping = linearDamping;
}
float b2Body_GetLinearDamping( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return bodySim->linearDamping;
}
void b2Body_SetAngularDamping( b2BodyId bodyId, float angularDamping )
{
B2_ASSERT( b2IsValidFloat( angularDamping ) && angularDamping >= 0.0f );
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->angularDamping = angularDamping;
}
float b2Body_GetAngularDamping( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return bodySim->angularDamping;
}
void b2Body_SetGravityScale( b2BodyId bodyId, float gravityScale )
{
B2_ASSERT( b2Body_IsValid( bodyId ) );
B2_ASSERT( b2IsValidFloat( gravityScale ) );
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->gravityScale = gravityScale;
}
float b2Body_GetGravityScale( b2BodyId bodyId )
{
B2_ASSERT( b2Body_IsValid( bodyId ) );
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return bodySim->gravityScale;
}
bool b2Body_IsAwake( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->setIndex == b2_awakeSet;
}
void b2Body_SetAwake( b2BodyId bodyId, bool awake )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( awake && body->setIndex >= b2_firstSleepingSet )
{
b2WakeBody( world, body );
}
else if ( awake == false && body->setIndex == b2_awakeSet )
{
b2Island* island = b2IslandArray_Get( &world->islands, body->islandId );
if ( island->constraintRemoveCount > 0 )
{
b2SplitIsland( world, body->islandId );
}
b2TrySleepIsland( world, body->islandId );
}
}
void b2Body_WakeTouching(b2BodyId bodyId)
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
int contactKey = body->headContactKey;
while ( contactKey != B2_NULL_INDEX )
{
int contactId = contactKey >> 1;
int edgeIndex = contactKey & 1;
b2Contact* contact = b2ContactArray_Get( &world->contacts, contactId );
b2Shape* shapeA = b2ShapeArray_Get( &world->shapes, contact->shapeIdA );
b2Shape* shapeB = b2ShapeArray_Get( &world->shapes, contact->shapeIdB );
if (shapeA->bodyId == bodyId.index1 - 1)
{
b2Body* otherBody = b2BodyArray_Get( &world->bodies, shapeB->bodyId );
b2WakeBody( world, otherBody );
}
else
{
b2Body* otherBody = b2BodyArray_Get( &world->bodies, shapeA->bodyId );
b2WakeBody( world, otherBody );
}
contactKey = contact->edges[edgeIndex].nextKey;
}
}
bool b2Body_IsEnabled( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->setIndex != b2_disabledSet;
}
bool b2Body_IsSleepEnabled( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->enableSleep;
}
void b2Body_SetSleepThreshold( b2BodyId bodyId, float sleepThreshold )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
body->sleepThreshold = sleepThreshold;
}
float b2Body_GetSleepThreshold( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->sleepThreshold;
}
void b2Body_EnableSleep( b2BodyId bodyId, bool enableSleep )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
body->enableSleep = enableSleep;
if ( enableSleep == false )
{
b2WakeBody( world, body );
}
}
void b2Body_Disable( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->setIndex == b2_disabledSet )
{
return;
}
bool wakeBodies = true;
b2DestroyBodyContacts( world, body, wakeBodies );
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, body->setIndex );
b2SolverSet* disabledSet = b2SolverSetArray_Get( &world->solverSets, b2_disabledSet );
int jointKey = body->headJointKey;
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
jointKey = joint->edges[edgeIndex].nextKey;
if ( joint->setIndex == b2_disabledSet )
{
continue;
}
B2_ASSERT( joint->setIndex == set->setIndex || set->setIndex == b2_staticSet );
b2UnlinkJoint( world, joint );
b2SolverSet* jointSet = b2SolverSetArray_Get( &world->solverSets, joint->setIndex );
b2TransferJoint( world, disabledSet, jointSet, joint );
}
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
shapeId = shape->nextShapeId;
b2DestroyShapeProxy( shape, &world->broadPhase );
}
b2RemoveBodyFromIsland( world, body );
b2TransferBody( world, disabledSet, set, body );
b2ValidateConnectivity( world );
b2ValidateSolverSets( world );
}
void b2Body_Enable( b2BodyId bodyId )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( body->setIndex != b2_disabledSet )
{
return;
}
b2SolverSet* disabledSet = b2SolverSetArray_Get( &world->solverSets, b2_disabledSet );
int setId = body->type == b2_staticBody ? b2_staticSet : b2_awakeSet;
b2SolverSet* targetSet = b2SolverSetArray_Get( &world->solverSets, setId );
b2TransferBody( world, targetSet, disabledSet, body );
b2Transform transform = b2GetBodyTransformQuick( world, body );
b2BodyType proxyType = body->type;
bool forcePairCreation = true;
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
shapeId = shape->nextShapeId;
b2CreateShapeProxy( shape, &world->broadPhase, proxyType, transform, forcePairCreation );
}
if ( setId != b2_staticSet )
{
b2CreateIslandForBody( world, setId, body );
}
int jointKey = body->headJointKey;
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
B2_ASSERT( joint->setIndex == b2_disabledSet );
B2_ASSERT( joint->islandId == B2_NULL_INDEX );
jointKey = joint->edges[edgeIndex].nextKey;
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
if ( bodyA->setIndex == b2_disabledSet || bodyB->setIndex == b2_disabledSet )
{
continue;
}
int jointSetId;
if ( bodyA->setIndex == b2_staticSet && bodyB->setIndex == b2_staticSet )
{
jointSetId = b2_staticSet;
}
else if ( bodyA->setIndex == b2_staticSet )
{
jointSetId = bodyB->setIndex;
}
else
{
jointSetId = bodyA->setIndex;
}
b2SolverSet* jointSet = b2SolverSetArray_Get( &world->solverSets, jointSetId );
b2TransferJoint( world, jointSet, disabledSet, joint );
if ( jointSetId != b2_staticSet )
{
b2LinkJoint( world, joint );
}
}
b2ValidateSolverSets( world );
}
void b2Body_SetMotionLocks( b2BodyId bodyId, b2MotionLocks locks )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
uint32_t newFlags = 0;
newFlags |= locks.linearX ? b2_lockLinearX : 0;
newFlags |= locks.linearY ? b2_lockLinearY : 0;
newFlags |= locks.angularZ ? b2_lockAngularZ : 0;
b2Body* body = b2GetBodyFullId( world, bodyId );
if ( ( body->flags & b2_allLocks ) != newFlags )
{
body->flags &= ~b2_allLocks;
body->flags |= newFlags;
b2BodySim* bodySim = b2GetBodySim( world, body );
bodySim->flags &= ~b2_allLocks;
bodySim->flags |= newFlags;
b2BodyState* state = b2GetBodyState( world, body );
if ( state != NULL )
{
state->flags = bodySim->flags;
if ( locks.linearX )
{
state->linearVelocity.x = 0.0f;
}
if ( locks.linearY )
{
state->linearVelocity.y = 0.0f;
}
if ( locks.angularZ )
{
state->angularVelocity = 0.0f;
}
}
}
}
b2MotionLocks b2Body_GetMotionLocks( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2MotionLocks locks;
locks.linearX = ( body->flags & b2_lockLinearX );
locks.linearY = ( body->flags & b2_lockLinearY );
locks.angularZ = ( body->flags & b2_lockAngularZ );
return locks;
}
void b2Body_SetBullet( b2BodyId bodyId, bool flag )
{
b2World* world = b2GetWorldLocked( bodyId.world0 );
if ( world == NULL )
{
return;
}
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
if ( flag )
{
bodySim->flags |= b2_isBullet;
}
else
{
bodySim->flags &= ~b2_isBullet;
}
}
bool b2Body_IsBullet( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
b2BodySim* bodySim = b2GetBodySim( world, body );
return ( bodySim->flags & b2_isBullet ) != 0;
}
void b2Body_EnableContactEvents( b2BodyId bodyId, bool flag )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
shape->enableContactEvents = flag;
shapeId = shape->nextShapeId;
}
}
void b2Body_EnableHitEvents( b2BodyId bodyId, bool flag )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
int shapeId = body->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
shape->enableHitEvents = flag;
shapeId = shape->nextShapeId;
}
}
b2WorldId b2Body_GetWorld( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
return (b2WorldId){ bodyId.world0 + 1, world->generation };
}
int b2Body_GetShapeCount( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->shapeCount;
}
int b2Body_GetShapes( b2BodyId bodyId, b2ShapeId* shapeArray, int capacity )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
int shapeId = body->headShapeId;
int shapeCount = 0;
while ( shapeId != B2_NULL_INDEX && shapeCount < capacity )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
b2ShapeId id = { shape->id + 1, bodyId.world0, shape->generation };
shapeArray[shapeCount] = id;
shapeCount += 1;
shapeId = shape->nextShapeId;
}
return shapeCount;
}
int b2Body_GetJointCount( b2BodyId bodyId )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
return body->jointCount;
}
int b2Body_GetJoints( b2BodyId bodyId, b2JointId* jointArray, int capacity )
{
b2World* world = b2GetWorld( bodyId.world0 );
b2Body* body = b2GetBodyFullId( world, bodyId );
int jointKey = body->headJointKey;
int jointCount = 0;
while ( jointKey != B2_NULL_INDEX && jointCount < capacity )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
b2JointId id = { jointId + 1, bodyId.world0, joint->generation };
jointArray[jointCount] = id;
jointCount += 1;
jointKey = joint->edges[edgeIndex].nextKey;
}
return jointCount;
}
bool b2ShouldBodiesCollide( b2World* world, b2Body* bodyA, b2Body* bodyB )
{
if ( bodyA->type != b2_dynamicBody && bodyB->type != b2_dynamicBody )
{
return false;
}
int jointKey;
int otherBodyId;
if ( bodyA->jointCount < bodyB->jointCount )
{
jointKey = bodyA->headJointKey;
otherBodyId = bodyB->id;
}
else
{
jointKey = bodyB->headJointKey;
otherBodyId = bodyA->id;
}
while ( jointKey != B2_NULL_INDEX )
{
int jointId = jointKey >> 1;
int edgeIndex = jointKey & 1;
int otherEdgeIndex = edgeIndex ^ 1;
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
if ( joint->collideConnected == false && joint->edges[otherEdgeIndex].bodyId == otherBodyId )
{
return false;
}
jointKey = joint->edges[edgeIndex].nextKey;
}
return true;
}