2015-05-09 20:32:36 +00:00
# pragma once
# include "base_vehicle.hpp"
# include "controller.hpp"
# include "btMatrix4x4.hpp"
using namespace ace : : simulation ;
namespace ace {
namespace vehicledamage {
2015-05-11 18:01:53 +00:00
base_vehicle : : base_vehicle ( uint32_t id , ace : : simulation : : object_p object_ , ace : : vector3 < float > position_ ) : id ( id ) , object ( object_ ) {
2015-05-09 20:32:36 +00:00
bt_mesh = std : : make_shared < btTriangleMesh > ( ) ;
fire_lod = - 1 ;
// Find the fire LOD
for ( int x = 0 ; x < object_ - > lods . size ( ) ; x + + ) {
if ( object_ - > lods [ x ] - > type = = LOD_TYPE_GEOMETRY_FIRE ) {
fire_lod = x ;
break ;
}
}
if ( fire_lod = = - 1 ) // @TODO: fallback on geo LOD
2015-05-12 05:33:39 +00:00
fire_lod = 0 ;
2015-05-14 07:21:48 +00:00
//fire_lod = 0;
2015-05-09 20:32:36 +00:00
assert ( fire_lod ! = - 1 ) ;
2015-05-12 05:33:39 +00:00
2015-05-09 20:32:36 +00:00
// Build the mesh from object faces
// TODO: LOD!?
// P3d store in x,z,y format
for ( auto & face : object_ - > lods [ fire_lod ] - > faces ) {
bt_mesh - > addTriangle (
2015-05-11 05:05:25 +00:00
btVector3 ( face - > vertices [ 0 ] - > x ( ) , face - > vertices [ 0 ] - > y ( ) , face - > vertices [ 0 ] - > z ( ) ) ,
btVector3 ( face - > vertices [ 1 ] - > x ( ) , face - > vertices [ 1 ] - > y ( ) , face - > vertices [ 1 ] - > z ( ) ) ,
btVector3 ( face - > vertices [ 2 ] - > x ( ) , face - > vertices [ 2 ] - > y ( ) , face - > vertices [ 2 ] - > z ( ) )
2015-05-09 20:32:36 +00:00
) ;
}
bt_shape = std : : make_shared < btBvhTriangleMeshShape > ( bt_mesh . get ( ) , true , true ) ;
bt_object = std : : make_shared < btCollisionObject > ( ) ;
bt_object - > setCollisionShape ( bt_shape . get ( ) ) ;
bt_object - > setUserIndex ( id ) ;
bt_object - > setUserPointer ( ( void * ) this ) ;
// @TODO: This is moving it in the bullet world for handling multiple collisions, instead our raytests need to ignore ALL but this type. How do we do that?
// For now this only works for single-object collisions then
2015-05-14 06:24:46 +00:00
// Set 3d world bullet position based on game position
btTransform transform = bt_object - > getWorldTransform ( ) ;
transform . setOrigin ( btVector3 ( position_ . x ( ) , position_ . y ( ) , position_ . z ( ) ) ) ;
bt_object - > setWorldTransform ( transform ) ;
2015-05-09 20:32:36 +00:00
controller : : get ( ) . bt_world - > addCollisionObject ( bt_object . get ( ) ) ;
}
base_vehicle : : ~ base_vehicle ( ) {
controller : : get ( ) . bt_world - > removeCollisionObject ( bt_object . get ( ) ) ;
}
2015-05-11 18:01:53 +00:00
bool base_vehicle : : simulate ( ) {
std : : vector < uint32_t > lods ;
lods . push_back ( fire_lod ) ;
2015-05-14 07:21:48 +00:00
object - > animate ( animation_state , lods ) ;
2015-05-11 18:01:53 +00:00
return true ;
}
2015-05-09 20:32:36 +00:00
void base_vehicle : : transform ( ) {
btTransform transform = bt_object - > getWorldTransform ( ) ;
transform . getBasis ( ) . setEulerZYX ( direction . x ( ) , direction . y ( ) , direction . z ( ) ) ;
transform . setOrigin ( btVector3 ( up . x ( ) , up . y ( ) , up . z ( ) ) ) ;
bt_object - > setWorldTransform ( transform ) ;
}
float base_vehicle : : thickness ( const ace : : vector3 < float > & position , const ace : : vector3 < float > & direction ) {
float result = - 1.0f ;
btVector3 bt_from ( position . x ( ) , position . y ( ) , position . z ( ) ) ;
btVector3 bt_dir ( direction . x ( ) , direction . y ( ) , direction . z ( ) ) ;
btVector3 bt_to ( bt_from + ( bt_dir * 100 ) ) ;
btCollisionWorld : : AllHitsRayResultCallback allResults ( bt_from , bt_to ) ;
controller : : get ( ) . bt_world - > rayTest ( bt_from , bt_to , allResults ) ;
// get the first and last result on the target object, give results
assert ( allResults . m_hitPointWorld . size ( ) > 2 ) ;
if ( allResults . m_hitPointWorld . size ( ) = = 2 ) {
// @TODO: This just gets the thickness of hte first 2 hit surfaces
result = allResults . m_hitPointWorld [ 0 ] . distance ( allResults . m_hitPointWorld [ 1 ] ) ;
} else {
// @TODO:
// We dont support multi-surface hits yet
}
return result ;
}
float base_vehicle : : surface_raycast ( const ace : : vector3 < float > & position , const ace : : vector3 < float > & direction , std : : vector < ace : : vector3 < float > > & results ) {
float result = - 1.0f ;
btVector3 bt_from ( position . x ( ) , position . y ( ) , position . z ( ) ) ;
btVector3 bt_dir ( direction . x ( ) , direction . y ( ) , direction . z ( ) ) ;
btVector3 bt_to ( bt_from + ( bt_dir * 100 ) ) ;
btCollisionWorld : : AllHitsRayResultCallback allResults ( bt_from , bt_to ) ;
controller : : get ( ) . bt_world - > rayTest ( bt_from , bt_to , allResults ) ;
if ( allResults . m_hitNormalWorld . size ( ) > 0 ) {
for ( int x = 0 ; x < allResults . m_hitNormalWorld . size ( ) ; x + + ) {
results . push_back ( ace : : vector3 < float > ( allResults . m_hitNormalWorld [ x ] . x ( ) , allResults . m_hitNormalWorld [ x ] . y ( ) , allResults . m_hitNormalWorld [ x ] . z ( ) ) ) ;
}
result = allResults . m_hitNormalWorld [ 0 ] . distance ( allResults . m_hitNormalWorld [ allResults . m_hitNormalWorld . size ( ) - 1 ] ) ;
}
return result ;
}
std : : vector < ace : : vector3 < float > > base_vehicle : : selection_position ( const uint32_t lod , const std : : string & name , const SELECTION_SEARCH_TYPE searchType ) {
named_selection_p selection ;
std : : vector < ace : : vector3 < float > > result ;
if ( ( selection = selection_by_name ( lod , name ) ) = = nullptr )
return result ;
switch ( searchType ) {
case SELECTION_SEARCH_TYPE : : AVERAGE_CENTER : {
ace : : vector3 < float > average ;
std : : vector < ace : : vector3 < float > > results ;
for ( auto & a : selection - > vertices ) {
for ( auto & b : selection - > vertices ) {
if ( a ! = b ) {
average = average + ace : : vector3 < float > : : lerp ( static_cast < ace : : vector3 < float > > ( * a ) , static_cast < ace : : vector3 < float > > ( * b ) , 0.5f ) ;
}
}
}
average = average / results . size ( ) ;
result . push_back ( average ) ;
break ;
}
}
return result ;
}
std : : vector < ace : : vector3 < float > > base_vehicle : : selection_by_name_vertices ( const uint32_t lod , const std : : string & name ) {
std : : vector < ace : : vector3 < float > > result ;
ace : : simulation : : named_selection_p selection = selection_by_name ( lod , name ) ;
for ( auto & vertex : selection - > vertices ) {
result . push_back ( static_cast < ace : : vector3 < float > > ( * vertex ) ) ;
}
return result ;
}
ace : : simulation : : named_selection_p base_vehicle : : selection_by_name ( const uint32_t lod , const std : : string & name ) {
named_selection_p result ;
if ( lod ! = - 1 ) {
auto iter = object - > lods [ lod ] - > selections . find ( name ) ;
if ( iter = = object - > lods [ lod ] - > selections . end ( ) ) {
return nullptr ;
}
result = iter - > second ;
} else {
for ( auto & lod : object - > lods ) {
auto iter = lod - > selections . find ( name ) ;
if ( iter = = lod - > selections . end ( ) ) {
continue ;
} else {
result = iter - > second ;
}
}
}
return result ;
}
}
} ;