2013-11-20 11:09:27 +01:00
|
|
|
#include "PhysicsAPI_Impl.h"
|
2013-11-22 11:52:45 +01:00
|
|
|
#include "OysterPhysics3D.h"
|
2013-11-28 11:58:46 +01:00
|
|
|
#include "SimpleRigidBody.h"
|
|
|
|
#include "SphericalRigidBody.h"
|
2013-11-20 11:09:27 +01:00
|
|
|
|
2013-11-21 17:22:13 +01:00
|
|
|
using namespace ::Oyster::Physics;
|
2013-11-25 16:57:38 +01:00
|
|
|
using namespace ::Oyster::Physics3D;
|
2013-11-21 17:22:13 +01:00
|
|
|
using namespace ::Oyster::Math;
|
|
|
|
using namespace ::Oyster::Collision3D;
|
|
|
|
using namespace ::Utility::DynamicMemory;
|
2013-11-20 11:09:27 +01:00
|
|
|
|
2013-11-25 16:35:56 +01:00
|
|
|
API_Impl API_instance;
|
2013-11-20 11:09:27 +01:00
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
Float4x4 & MomentOfInertia::CreateSphereMatrix( const Float mass, const Float radius)
|
2013-11-22 11:52:45 +01:00
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
return Formula::MomentOfInertia::Sphere(mass, radius);
|
2013-11-22 11:52:45 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
Float4x4 & MomentOfInertia::CreateHollowSphereMatrix( const Float mass, const Float radius)
|
2013-11-22 11:52:45 +01:00
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
return Formula::MomentOfInertia::HollowSphere(mass, radius);
|
2013-11-22 11:52:45 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
Float4x4 & MomentOfInertia::CreateCuboidMatrix( const Float mass, const Float height, const Float width, const Float depth )
|
2013-11-22 11:52:45 +01:00
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
return Formula::MomentOfInertia::Cuboid(mass, height, width, depth);
|
2013-11-22 11:52:45 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
Float4x4 & MomentOfInertia::CreateCylinderMatrix( const Float mass, const Float height, const Float radius )
|
2013-11-22 11:52:45 +01:00
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
return Formula::MomentOfInertia::Cylinder(mass, height, radius);
|
2013-11-22 11:52:45 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
Float4x4 & MomentOfInertia::CreateRodMatrix( const Float mass, const Float length )
|
2013-11-22 11:52:45 +01:00
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
return Formula::MomentOfInertia::RodCenter(mass, length);
|
2013-11-22 11:52:45 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 12:21:44 +01:00
|
|
|
API & API::Instance()
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
2013-11-25 16:35:56 +01:00
|
|
|
return API_instance;
|
2013-11-20 11:09:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
API_Impl::API_Impl()
|
2013-11-28 17:59:12 +01:00
|
|
|
{
|
|
|
|
this->gravityConstant = Constant::gravity_constant;
|
|
|
|
this->updateFrameLength = 1.0f / 120.0f;
|
|
|
|
this->destructionAction = Default::EventAction_Destruction;
|
|
|
|
this->worldScene = Octree();
|
|
|
|
}
|
2013-11-20 11:09:27 +01:00
|
|
|
|
2013-11-25 16:57:38 +01:00
|
|
|
API_Impl::~API_Impl() {}
|
2013-11-20 11:09:27 +01:00
|
|
|
|
2013-11-26 13:27:34 +01:00
|
|
|
void API_Impl::Init( unsigned int numObjects, unsigned int numGravityWells , const Float3 &worldSize )
|
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned char numLayers = 4; //!< @todo TODO: calc numLayers from worldSize
|
|
|
|
this->worldScene = Octree( numObjects, numLayers, worldSize );
|
2013-11-26 13:27:34 +01:00
|
|
|
}
|
|
|
|
|
2013-11-28 17:59:12 +01:00
|
|
|
void API_Impl::SetFrameTimeLength( float deltaTime )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
this->updateFrameLength = deltaTime;
|
2013-11-20 11:09:27 +01:00
|
|
|
}
|
2013-11-28 10:26:29 +01:00
|
|
|
|
2013-11-20 11:09:27 +01:00
|
|
|
void API_Impl::SetGravityConstant( float g )
|
|
|
|
{
|
2013-11-25 16:57:38 +01:00
|
|
|
this->gravityConstant = g;
|
2013-11-20 11:09:27 +01:00
|
|
|
}
|
2013-11-28 10:26:29 +01:00
|
|
|
|
|
|
|
void API_Impl::SetSubscription( API::EventAction_Destruction functionPointer )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
2013-11-28 10:26:29 +01:00
|
|
|
if( functionPointer )
|
|
|
|
{
|
|
|
|
this->destructionAction = functionPointer;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this->destructionAction = Default::EventAction_Destruction;
|
|
|
|
}
|
2013-11-20 11:09:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void API_Impl::Update()
|
|
|
|
{
|
|
|
|
/** @todo TODO: Fix this function.*/
|
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
bool API_Impl::IsInLimbo( const ICustomBody* objRef )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
|
|
|
//! @todo TODO: implement stub
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::MoveToLimbo( const ICustomBody* objRef )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
|
|
|
/** @todo TODO: Fix this function.*/
|
|
|
|
}
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::ReleaseFromLimbo( const ICustomBody* objRef )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
|
|
|
/** @todo TODO: Fix this function.*/
|
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::AddObject( ::Utility::DynamicMemory::UniquePointer<ICustomBody> handle )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
this->worldScene.AddObject( handle );
|
2013-11-20 11:09:27 +01:00
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
|
2013-11-28 17:59:12 +01:00
|
|
|
UniquePointer<ICustomBody> API_Impl::ExtractObject( const ICustomBody* objRef )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
return this->worldScene.Extract( objRef );
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::DestroyObject( const ICustomBody* objRef )
|
2013-11-20 11:09:27 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
UniquePointer<ICustomBody> object = this->worldScene.Extract( objRef );
|
2013-11-28 18:05:19 +01:00
|
|
|
if( object )
|
2013-11-28 17:59:12 +01:00
|
|
|
{
|
|
|
|
this->destructionAction( object );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::ApplyForceAt( const ICustomBody* objRef, const Float3 &worldPos, const Float3 &worldF )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
//this->worldScene.GetCustomBody( tempRef )->Apply //!< @todo TODO: need function
|
|
|
|
this->worldScene.SetAsAltered( tempRef );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::ApplyCollisionResponse( const ICustomBody* objRefA, const ICustomBody* objRefB, Float &deltaWhen, Float3 &worldPointOfContact )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRefA );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
//! @todo TODO: implement stub
|
|
|
|
this->worldScene.SetAsAltered( tempRef );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetMomentOfInertiaTensor_KeepVelocity( const ICustomBody* objRef, const Float4x4 &localI )
|
2013-11-28 17:59:12 +01:00
|
|
|
{ // deprecated
|
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetMomentOfInertiaTensor_KeepVelocity( localI );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetMomentOfInertiaTensor_KeepMomentum( const ICustomBody* objRef, const Float4x4 &localI )
|
2013-11-28 17:59:12 +01:00
|
|
|
{ // deprecated
|
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetMomentOfInertiaTensor_KeepMomentum( localI );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetMass_KeepVelocity( const ICustomBody* objRef, Float m )
|
2013-11-28 17:59:12 +01:00
|
|
|
{ // deprecated
|
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetMass_KeepVelocity( m );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetMass_KeepMomentum( const ICustomBody* objRef, Float m )
|
2013-11-28 17:59:12 +01:00
|
|
|
{ // deprecated
|
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetMass_KeepMomentum( m );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetCenter( const ICustomBody* objRef, const Float3 &worldPos )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
//this->worldScene.GetCustomBody( tempRef )->Set //!< @todo TODO: need function
|
|
|
|
this->worldScene.EvaluatePosition( tempRef );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetRotation( const ICustomBody* objRef, const Float4x4 &rotation )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetRotation( rotation );
|
|
|
|
this->worldScene.EvaluatePosition( tempRef );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-25 10:54:27 +01:00
|
|
|
void API_Impl::SetOrientation( const ICustomBody* objRef, const Float4x4 &orientation )
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetOrientation( orientation );
|
|
|
|
this->worldScene.EvaluatePosition( tempRef );
|
|
|
|
}
|
2013-11-21 17:22:13 +01:00
|
|
|
}
|
|
|
|
|
2013-11-26 13:27:34 +01:00
|
|
|
void API_Impl::SetSize( const ICustomBody* objRef, const Float3 &size )
|
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
unsigned int tempRef = this->worldScene.GetTemporaryReferenceOf( objRef );
|
|
|
|
if( tempRef != this->worldScene.invalid_ref )
|
|
|
|
{
|
|
|
|
this->worldScene.GetCustomBody( tempRef )->SetSize( size );
|
|
|
|
this->worldScene.EvaluatePosition( tempRef );
|
|
|
|
}
|
2013-11-26 13:27:34 +01:00
|
|
|
}
|
|
|
|
|
2013-11-28 11:58:46 +01:00
|
|
|
UniquePointer<ICustomBody> API_Impl::CreateRigidBody( const API::SimpleBodyDescription &desc ) const
|
|
|
|
{
|
|
|
|
return new SimpleRigidBody( desc );
|
|
|
|
}
|
|
|
|
|
|
|
|
UniquePointer<ICustomBody> API_Impl::CreateRigidBody( const API::SphericalBodyDescription &desc ) const
|
2013-11-21 17:22:13 +01:00
|
|
|
{
|
2013-11-28 12:13:14 +01:00
|
|
|
return new SphericalRigidBody( desc );
|
2013-11-28 10:26:29 +01:00
|
|
|
}
|
|
|
|
|
2013-11-28 17:59:12 +01:00
|
|
|
namespace Oyster { namespace Physics
|
2013-11-28 10:26:29 +01:00
|
|
|
{
|
2013-11-28 17:59:12 +01:00
|
|
|
namespace Default
|
|
|
|
{
|
|
|
|
void EventAction_Destruction( ::Utility::DynamicMemory::UniquePointer<::Oyster::Physics::ICustomBody> proto )
|
|
|
|
{ /* Do nothing except allowing the proto uniquePointer destroy itself. */ }
|
2013-11-28 10:26:29 +01:00
|
|
|
|
2013-11-28 17:59:12 +01:00
|
|
|
::Oyster::Physics::ICustomBody::SubscriptMessage EventAction_Collision( const ::Oyster::Physics::ICustomBody *proto, const ::Oyster::Physics::ICustomBody *deuter )
|
|
|
|
{ /* Do nothing except returning business as usual. */
|
|
|
|
return ::Oyster::Physics::ICustomBody::SubscriptMessage_none;
|
|
|
|
}
|
2013-11-28 10:26:29 +01:00
|
|
|
}
|
2013-11-28 17:59:12 +01:00
|
|
|
} }
|