Selaa lähdekoodia

fixed 23 files out of 277:

- fixed 1270 CR characters
 - fixed 56 trailing spaces
 - fixed 5085 tabs
legacy
Lolbot lolbot 12 vuotta sitten
vanhempi
commit
23807ea88e
20 muutettua tiedostoa jossa 2190 lisäystä ja 2190 poistoa
  1. +41
    -41
      src/easymesh/easymesh.cpp
  2. +4
    -4
      src/entity.h
  3. +131
    -131
      src/input/input.cpp
  4. +1
    -1
      src/input/input.h
  5. +2
    -2
      src/lol/math/math.h
  6. +2
    -2
      src/numeric.h
  7. +1
    -1
      src/worldentity.cpp
  8. +376
    -376
      test/BtPhysTest.cpp
  9. +12
    -12
      test/BtPhysTest.h
  10. +269
    -269
      test/PhysicObject.h
  11. +245
    -245
      test/Physics/Include/BulletCharacterController.h
  12. +56
    -56
      test/Physics/Include/EasyCharacterController.h
  13. +153
    -153
      test/Physics/Include/EasyConstraint.h
  14. +102
    -102
      test/Physics/Include/EasyPhysics.h
  15. +18
    -18
      test/Physics/Include/LolBtPhysicsIntegration.h
  16. +373
    -373
      test/Physics/Include/LolPhysics.h
  17. +161
    -161
      test/Physics/Src/BulletCharacterController.cpp
  18. +55
    -55
      test/Physics/Src/EasyCharacterController.cpp
  19. +12
    -12
      test/Physics/Src/EasyConstraint.cpp
  20. +176
    -176
      test/Physics/Src/EasyPhysics.cpp

+ 41
- 41
src/easymesh/easymesh.cpp Näytä tiedosto

@@ -240,45 +240,45 @@ void EasyMesh::Rotate(float t, vec3 const &axis)
void EasyMesh::RadialJitter(float r) void EasyMesh::RadialJitter(float r)
{ {
Array<int> Welded; Array<int> Welded;
Welded.Push(-1); Welded.Push(-1);
for (int i = m_cursors.Last().m1 + 1; i < m_vert.Count(); i++) for (int i = m_cursors.Last().m1 + 1; i < m_vert.Count(); i++)
{ {
int j, k; int j, k;
for (j = m_cursors.Last().m1, k = 0; j < i; j++, k++) for (j = m_cursors.Last().m1, k = 0; j < i; j++, k++)
{ {
if(Welded[k] < 0) if(Welded[k] < 0)
{ {
vec3 diff = m_vert[i].m1 - m_vert[j].m1; vec3 diff = m_vert[i].m1 - m_vert[j].m1;
if(diff.x > 0.1f || diff.x < -0.1f)
if(diff.x > 0.1f || diff.x < -0.1f) continue;
continue; if(diff.y > 0.1f || diff.y < -0.1f)
continue;
if(diff.y > 0.1f || diff.y < -0.1f) if(diff.z > 0.1f || diff.z < -0.1f)
continue; continue;
break;
if(diff.z > 0.1f || diff.z < -0.1f) }
continue; }
if(j == i)
break; Welded.Push(-1);
} else
} Welded.Push(j);
}
if(j == i) int i, j;
Welded.Push(-1);
else
Welded.Push(j);
}
int i, j;
for (i = m_cursors.Last().m1, j = 0; i < m_vert.Count(); i++, j++) for (i = m_cursors.Last().m1, j = 0; i < m_vert.Count(); i++, j++)
{ {
if(Welded[j] == -1) if(Welded[j] == -1)
m_vert[i].m1 *= 1.0f + RandF(r); m_vert[i].m1 *= 1.0f + RandF(r);
else else
m_vert[i].m1 = m_vert[Welded[j]].m1; m_vert[i].m1 = m_vert[Welded[j]].m1;
} }


ComputeNormals(m_cursors.Last().m2, m_indices.Count() - m_cursors.Last().m2); ComputeNormals(m_cursors.Last().m2, m_indices.Count() - m_cursors.Last().m2);
} }


void EasyMesh::TaperX(float y, float z, float xoff) void EasyMesh::TaperX(float y, float z, float xoff)
@@ -361,10 +361,10 @@ void EasyMesh::AppendCylinder(int nsides, float h, float r1, float r2,
vec3 p1(r1, -h * .5f, 0.f), p2(r2, h * .5f, 0.f), n; vec3 p1(r1, -h * .5f, 0.f), p2(r2, h * .5f, 0.f), n;


/* Construct normal */ /* Construct normal */
if (r2 != .0f) if (r2 != .0f)
n = vec3(r2, h * .5f, 0.f); n = vec3(r2, h * .5f, 0.f);
else else
n = vec3(r1, h * .5f, 0.f); n = vec3(r1, h * .5f, 0.f);
n.y = r1 * (r1 - r2) / h; n.y = r1 * (r1 - r2) / h;
if (!smooth) if (!smooth)
n = mat3::rotate(180.0f / nsides, 0.f, 1.f, 0.f) * n; n = mat3::rotate(180.0f / nsides, 0.f, 1.f, 0.f) * n;
@@ -850,7 +850,7 @@ void EasyMesh::Chamfer(float f)


for (int i = 0; i < ilen / 3; i++) for (int i = 0; i < ilen / 3; i++)
{ {
} }


/* Fun shit: reduce all triangles */ /* Fun shit: reduce all triangles */


+ 4
- 4
src/entity.h Näytä tiedosto

@@ -45,11 +45,11 @@ protected:
{ {
GAMEGROUP_BEFORE = 0, GAMEGROUP_BEFORE = 0,
GAMEGROUP_DEFAULT, GAMEGROUP_DEFAULT,
GAMEGROUP_AFTER, GAMEGROUP_AFTER,
GAMEGROUP_AFTER_0, GAMEGROUP_AFTER_0,
GAMEGROUP_AFTER_1, GAMEGROUP_AFTER_1,


// Must be the last element // Must be the last element
GAMEGROUP_END GAMEGROUP_END
} }
m_gamegroup; m_gamegroup;


+ 131
- 131
src/input/input.cpp Näytä tiedosto

@@ -31,7 +31,7 @@ namespace lol
* Input implementation class * Input implementation class
*/ */


InputTracker* Input::m_input_tracker = NULL; InputTracker* Input::m_input_tracker = NULL;


static class InputData static class InputData
{ {
@@ -66,10 +66,10 @@ static InputData * const data = &inputdata;


int ButtonSetting::GetActionSettingIdx(ACTION_TYPE SearchAction) int ButtonSetting::GetActionSettingIdx(ACTION_TYPE SearchAction)
{ {
for (int i = 0; i < m_associated_action_list.Count(); i++) for (int i = 0; i < m_associated_action_list.Count(); i++)
if (ACTION_CMP(m_associated_action_list[i].m_action, SearchAction)) if (ACTION_CMP(m_associated_action_list[i].m_action, SearchAction))
return i; return i;
return -1; return -1;
} }


/* /*
@@ -78,37 +78,37 @@ int ButtonSetting::GetActionSettingIdx(ACTION_TYPE SearchAction)


InputTracker::InputTracker() InputTracker::InputTracker()
{ {
m_gamegroup = GAMEGROUP_BEFORE; m_gamegroup = GAMEGROUP_BEFORE;


for (int i = 0; i < Key::Last * 2; ++i) for (int i = 0; i < Key::Last * 2; ++i)
m_input_status << 0; m_input_status << 0;


Ticker::Ref(this); Ticker::Ref(this);
} }


//Internal //Internal
int InputTracker::GetButtonSettingIdx(Key Button) int InputTracker::GetButtonSettingIdx(Key Button)
{ {
for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
if (m_input_assocation_list[i].m_raw_button == Button) if (m_input_assocation_list[i].m_raw_button == Button)
return i; return i;
return -1; return -1;
} }


//----- //-----
int InputTracker::GetCurrentButtonStatus(Key Button) int InputTracker::GetCurrentButtonStatus(Key Button)
{ {
if (Button < m_input_status.Count()) if (Button < m_input_status.Count())
return m_input_status[Button]; return m_input_status[Button];
return 0; return 0;
} }


//----- //-----
int InputTracker::GetPreviousButtonStatus(Key Button) int InputTracker::GetPreviousButtonStatus(Key Button)
{ {
if (Button < m_input_status.Count()) if (Button < m_input_status.Count())
return m_input_status[(int)Button + (int)Key::Last]; return m_input_status[(int)Button + (int)Key::Last];
return 0; return 0;
} }


//Internal : Updates the action status & timers //Internal : Updates the action status & timers
@@ -116,146 +116,146 @@ void InputTracker::UpdateActionStatus(float seconds)
{ {
#if defined USE_SDL #if defined USE_SDL
#if SDL_MAJOR_VERSION == 1 && SDL_MINOR_VERSION >= 3 #if SDL_MAJOR_VERSION == 1 && SDL_MINOR_VERSION >= 3
Uint8 *keystate = SDL_GetKeyboardState(NULL); Uint8 *keystate = SDL_GetKeyboardState(NULL);
#else #else
Uint8 *keystate = SDL_GetKeyState(NULL); Uint8 *keystate = SDL_GetKeyState(NULL);
#endif #endif
//SOOOoooo ugly. //SOOOoooo ugly.
for (int i = 0; i < Key::Last; ++i) for (int i = 0; i < Key::Last; ++i)
{ {
m_input_status[i + Key::Last] = m_input_status[i]; m_input_status[i + Key::Last] = m_input_status[i];
m_input_status[i] = keystate[i]; m_input_status[i] = keystate[i];
} }
#endif #endif


for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
{ {
ButtonSetting &CurIT = m_input_assocation_list[i]; ButtonSetting &CurIT = m_input_assocation_list[i];


for (int j = 0; j < CurIT.m_associated_action_list.Count(); j++) for (int j = 0; j < CurIT.m_associated_action_list.Count(); j++)
{ {
ActionSetting &CurAS = CurIT.m_associated_action_list[j]; ActionSetting &CurAS = CurIT.m_associated_action_list[j];


if (CurAS.m_buffered_since <= CurAS.m_buffering_time) if (CurAS.m_buffered_since <= CurAS.m_buffering_time)
CurAS.m_buffered_since += seconds; CurAS.m_buffered_since += seconds;


if (GetCurrentButtonStatus(CurIT.m_raw_button) && if (GetCurrentButtonStatus(CurIT.m_raw_button) &&
CurAS.m_buffering_time >= .0f) CurAS.m_buffering_time >= .0f)
CurAS.m_buffered_since = .0f; CurAS.m_buffered_since = .0f;
} }
} }
} }


//Helps link a software input Action-Id to an hardware input Button-Id. //Helps link a software input Action-Id to an hardware input Button-Id.
void InputTracker::LinkActionToKey(ACTION_TYPE Action, Key Button) void InputTracker::LinkActionToKey(ACTION_TYPE Action, Key Button)
{ {
int ITIdx = GetButtonSettingIdx(Button); int ITIdx = GetButtonSettingIdx(Button);
if (ITIdx == -1) if (ITIdx == -1)
{ {
ITIdx = m_input_assocation_list.Count(); ITIdx = m_input_assocation_list.Count();
m_input_assocation_list << ButtonSetting(Button); m_input_assocation_list << ButtonSetting(Button);
} }


ButtonSetting &CurIT = m_input_assocation_list[ITIdx]; ButtonSetting &CurIT = m_input_assocation_list[ITIdx];


int ASIdx = CurIT.GetActionSettingIdx(Action); int ASIdx = CurIT.GetActionSettingIdx(Action);
if (ASIdx == -1) if (ASIdx == -1)
{ {
ASIdx = CurIT.m_associated_action_list.Count(); ASIdx = CurIT.m_associated_action_list.Count();
CurIT.m_associated_action_list << ActionSetting(Action); CurIT.m_associated_action_list << ActionSetting(Action);
} }
} }


//Helps unlink a software input Action-Id to an hardware input Button-Id. //Helps unlink a software input Action-Id to an hardware input Button-Id.
void InputTracker::UnlinkAction(ACTION_TYPE Action) void InputTracker::UnlinkAction(ACTION_TYPE Action)
{ {
for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
{ {
ButtonSetting &CurIT = m_input_assocation_list[i]; ButtonSetting &CurIT = m_input_assocation_list[i];
int ASIdx = CurIT.GetActionSettingIdx(Action); int ASIdx = CurIT.GetActionSettingIdx(Action);
if (ASIdx != -1) if (ASIdx != -1)
CurIT.m_associated_action_list.Remove(ASIdx); CurIT.m_associated_action_list.Remove(ASIdx);
} }
} }


//Returns the current status of a given action //Returns the current status of a given action
int InputTracker::GetStatus(ACTION_TYPE Action) int InputTracker::GetStatus(ACTION_TYPE Action)
{ {
for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
{ {
ButtonSetting &CurIT = m_input_assocation_list[i]; ButtonSetting &CurIT = m_input_assocation_list[i];
int ASIdx = CurIT.GetActionSettingIdx(Action); int ASIdx = CurIT.GetActionSettingIdx(Action);
if (ASIdx != -1) if (ASIdx != -1)
{ {
ActionSetting &CurAS = CurIT.m_associated_action_list[ASIdx]; ActionSetting &CurAS = CurIT.m_associated_action_list[ASIdx];


if (CurAS.m_buffering_time >= .0f && CurAS.m_buffered_since <= CurAS.m_buffering_time) if (CurAS.m_buffering_time >= .0f && CurAS.m_buffered_since <= CurAS.m_buffering_time)
return 1; return 1;
return 0; return 0;
} }
} }
return 0; return 0;
} }


//Returns TRUE if action status went from Active to Inactive this frame //Returns TRUE if action status went from Active to Inactive this frame
bool InputTracker::WasReleased(ACTION_TYPE Action) bool InputTracker::WasReleased(ACTION_TYPE Action)
{ {
for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
{ {
ButtonSetting &CurIT = m_input_assocation_list[i]; ButtonSetting &CurIT = m_input_assocation_list[i];
int ASIdx = CurIT.GetActionSettingIdx(Action); int ASIdx = CurIT.GetActionSettingIdx(Action);
if (ASIdx != -1) if (ASIdx != -1)
{ {
if (GetPreviousButtonStatus(CurIT.m_raw_button) &&
if (GetPreviousButtonStatus(CurIT.m_raw_button) && !GetCurrentButtonStatus(CurIT.m_raw_button))
!GetCurrentButtonStatus(CurIT.m_raw_button)) return true;
return true; return false;
return false; }
} }
} return false;
return false;
} }


//Returns TRUE if action status went from Inactive to Active this frame //Returns TRUE if action status went from Inactive to Active this frame
bool InputTracker::WasPressed(ACTION_TYPE Action) bool InputTracker::WasPressed(ACTION_TYPE Action)
{ {
for (int i = 0; i < m_input_assocation_list.Count(); i++) for (int i = 0; i < m_input_assocation_list.Count(); i++)
{ {
ButtonSetting &CurIT = m_input_assocation_list[i]; ButtonSetting &CurIT = m_input_assocation_list[i];
int ASIdx = CurIT.GetActionSettingIdx(Action); int ASIdx = CurIT.GetActionSettingIdx(Action);
if (ASIdx != -1) if (ASIdx != -1)
{ {
if (!GetPreviousButtonStatus(CurIT.m_raw_button) && if (!GetPreviousButtonStatus(CurIT.m_raw_button) &&
GetCurrentButtonStatus(CurIT.m_raw_button)) GetCurrentButtonStatus(CurIT.m_raw_button))
return true; return true;
return false; return false;
} }
} }
return false; return false;
} }


//Returns the current status of a given action //Returns the current status of a given action
int InputTracker::GetStatus(Key Button) int InputTracker::GetStatus(Key Button)
{ {
return GetCurrentButtonStatus(Button); return GetCurrentButtonStatus(Button);
} }


//Returns TRUE if action status went from Active to Inactive this frame //Returns TRUE if action status went from Active to Inactive this frame
bool InputTracker::WasReleased(Key Button) bool InputTracker::WasReleased(Key Button)
{ {
if (GetPreviousButtonStatus(Button) && if (GetPreviousButtonStatus(Button) &&
!GetCurrentButtonStatus(Button)) !GetCurrentButtonStatus(Button))
return true; return true;
return false; return false;
} }


//Returns TRUE if action status went from Inactive to Active this frame //Returns TRUE if action status went from Inactive to Active this frame
bool InputTracker::WasPressed(Key Button) bool InputTracker::WasPressed(Key Button)
{ {
if (!GetPreviousButtonStatus(Button) && if (!GetPreviousButtonStatus(Button) &&
GetCurrentButtonStatus(Button)) GetCurrentButtonStatus(Button))
return true; return true;
return false; return false;
} }


/* /*
@@ -316,63 +316,63 @@ int Input::GetButtonState(int button)
//Helps link a software input Action-Id to an hardware input Button-Id. //Helps link a software input Action-Id to an hardware input Button-Id.
void Input::LinkActionToKey(ACTION_TYPE Action, struct Key Button) void Input::LinkActionToKey(ACTION_TYPE Action, struct Key Button)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
Input::m_input_tracker->LinkActionToKey(Action, Button); Input::m_input_tracker->LinkActionToKey(Action, Button);
} }


//Helps unlink a software input Action-Id to an hardware input Button-Id. //Helps unlink a software input Action-Id to an hardware input Button-Id.
void Input::UnlinkAction(ACTION_TYPE Action) void Input::UnlinkAction(ACTION_TYPE Action)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
Input::m_input_tracker->UnlinkAction(Action); Input::m_input_tracker->UnlinkAction(Action);
} }


//Returns the current status of a given action //Returns the current status of a given action
int Input::GetStatus(ACTION_TYPE Action) int Input::GetStatus(ACTION_TYPE Action)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->GetStatus(Action); return Input::m_input_tracker->GetStatus(Action);
return 0; return 0;
} }


//Returns TRUE if action status when from Active to Inactive this frame //Returns TRUE if action status when from Active to Inactive this frame
bool Input::WasPressed(ACTION_TYPE Action) bool Input::WasPressed(ACTION_TYPE Action)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->WasPressed(Action); return Input::m_input_tracker->WasPressed(Action);
return false; return false;
} }


//Returns TRUE if action status when from Active to Inactive this frame //Returns TRUE if action status when from Active to Inactive this frame
bool Input::WasReleased(ACTION_TYPE Action) bool Input::WasReleased(ACTION_TYPE Action)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->WasReleased(Action); return Input::m_input_tracker->WasReleased(Action);
return false; return false;
} }


//Returns the current status of a given action //Returns the current status of a given action
int Input::GetStatus(Key Button) int Input::GetStatus(Key Button)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->GetStatus(Button); return Input::m_input_tracker->GetStatus(Button);
return 0; return 0;
} }


//Returns TRUE if action status when from Active to Inactive this frame //Returns TRUE if action status when from Active to Inactive this frame
bool Input::WasPressed(Key Button) bool Input::WasPressed(Key Button)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->WasPressed(Button); return Input::m_input_tracker->WasPressed(Button);
return false; return false;
} }


//Returns TRUE if action status when from Active to Inactive this frame //Returns TRUE if action status when from Active to Inactive this frame
bool Input::WasReleased(Key Button) bool Input::WasReleased(Key Button)
{ {
if (CheckInputTrackerInit()) if (CheckInputTrackerInit())
return Input::m_input_tracker->WasReleased(Button); return Input::m_input_tracker->WasReleased(Button);
return false; return false;
} }


//-- //--


+ 1
- 1
src/input/input.h Näytä tiedosto

@@ -409,7 +409,7 @@ public:
static ivec2 GetMousePos(); static ivec2 GetMousePos();
static ivec3 GetMouseButtons(); static ivec3 GetMouseButtons();


//BH : Shouldn't use this //BH : Shouldn't use this
static int GetButtonState(int button); static int GetButtonState(int button);


/* Action management */ /* Action management */


+ 2
- 2
src/lol/math/math.h Näytä tiedosto

@@ -66,11 +66,11 @@ static inline void sincos(float const &x, float *s, float *c)


static inline double lerp(double const &a, double const &b, double const &x) static inline double lerp(double const &a, double const &b, double const &x)
{ {
return a + (b - a) * x; return a + (b - a) * x;
} }
static inline float lerp(float const &a, float const &b, float const &x) static inline float lerp(float const &a, float const &b, float const &x)
{ {
return a + (b - a) * x; return a + (b - a) * x;
} }






+ 2
- 2
src/numeric.h Näytä tiedosto

@@ -55,8 +55,8 @@ template <typename T> static inline T PotUp(T val)
//Lerp for float //Lerp for float
template <typename T1, typename T2, typename Tf> static inline T1 damp(const T1 &a, const T2 &b, const Tf &x, const Tf &dt) template <typename T1, typename T2, typename Tf> static inline T1 damp(const T1 &a, const T2 &b, const Tf &x, const Tf &dt)
{ {
if (dt <= .0f) if (dt <= .0f)
return a; return a;
return lol::lerp(a, b, dt / (dt + x)); return lol::lerp(a, b, dt / (dt + x));
} }




+ 1
- 1
src/worldentity.cpp Näytä tiedosto

@@ -28,7 +28,7 @@ WorldEntity::WorldEntity()
m_position = vec3(0); m_position = vec3(0);
m_rotation = quat(1); m_rotation = quat(1);
m_velocity = vec3(0); m_velocity = vec3(0);
m_rotation_velocity = vec3(0); m_rotation_velocity = vec3(0);
m_bbox[0] = m_bbox[1] = vec3(0); m_bbox[0] = m_bbox[1] = vec3(0);


m_mousepos = ivec2(0); m_mousepos = ivec2(0);


+ 376
- 376
test/BtPhysTest.cpp Näytä tiedosto

@@ -49,25 +49,25 @@ using namespace lol::phys;


int gNumObjects = 64; int gNumObjects = 64;


#define USE_WALL 1 #define USE_WALL 1
#define USE_PLATFORM 1 #define USE_PLATFORM 1
#define USE_ROPE 0 #define USE_ROPE 0
#define USE_BODIES 1 #define USE_BODIES 1
#define USE_ROTATION 0 #define USE_ROTATION 0
#define USE_CHARACTER 1 #define USE_CHARACTER 1
#define USE_STAIRS 1 #define USE_STAIRS 1
#define IPT_MOVE_FORWARD "Move_Forward"
#define IPT_MOVE_FORWARD "Move_Forward" #define IPT_MOVE_BACKWARD "Move_Backward"
#define IPT_MOVE_BACKWARD "Move_Backward" #define IPT_MOVE_LEFT "Move_Left"
#define IPT_MOVE_LEFT "Move_Left" #define IPT_MOVE_RIGHT "Move_Right"
#define IPT_MOVE_RIGHT "Move_Right" #define IPT_MOVE_UP "Move_Up"
#define IPT_MOVE_UP "Move_Up" #define IPT_MOVE_DOWN "Move_Down"
#define IPT_MOVE_DOWN "Move_Down" #define IPT_MOVE_JUMP "Move_Jump"
#define IPT_MOVE_JUMP "Move_Jump"


BtPhysTest::BtPhysTest(bool editor) BtPhysTest::BtPhysTest(bool editor)
{ {
m_loop_value = .0f; m_loop_value = .0f;


/* Create a camera that matches the settings of XNA BtPhysTest */ /* Create a camera that matches the settings of XNA BtPhysTest */
m_camera = new Camera(vec3(0.f, 600.f, 0.f), m_camera = new Camera(vec3(0.f, 600.f, 0.f),
@@ -75,191 +75,191 @@ BtPhysTest::BtPhysTest(bool editor)
vec3(0, 1, 0)); vec3(0, 1, 0));
m_camera->SetRotation(quat::fromeuler_xyz(0.f, 0.f, 0.f)); m_camera->SetRotation(quat::fromeuler_xyz(0.f, 0.f, 0.f));
m_camera->SetPerspective(45.f, 1280.f, 960.f, .1f, 1000.f); m_camera->SetPerspective(45.f, 1280.f, 960.f, .1f, 1000.f);
//m_camera->SetOrtho(1280.f / 6, 960.f / 6, -1000.f, 1000.f); //m_camera->SetOrtho(1280.f / 6, 960.f / 6, -1000.f, 1000.f);
Ticker::Ref(m_camera); Ticker::Ref(m_camera);


m_ready = false; m_ready = false;


m_simulation = new Simulation(); m_simulation = new Simulation();
m_simulation->SetWorldLimit(vec3(-1000.0f, -1000.0f, -1000.0f), vec3(1000.0f, 1000.0f, 1000.0f)); m_simulation->SetWorldLimit(vec3(-1000.0f, -1000.0f, -1000.0f), vec3(1000.0f, 1000.0f, 1000.0f));
m_simulation->Init(); m_simulation->Init();
vec3 NewGravity = vec3(.0f, -10.0f, .0f); vec3 NewGravity = vec3(.0f, -10.0f, .0f);
m_simulation->SetGravity(NewGravity); m_simulation->SetGravity(NewGravity);
m_simulation->SetContinuousDetection(true); m_simulation->SetContinuousDetection(true);
m_simulation->SetTimestep(1.f / 120.f); m_simulation->SetTimestep(1.f / 120.f);
Ticker::Ref(m_simulation); Ticker::Ref(m_simulation);


float offset = 29.5f; float offset = 29.5f;
vec3 pos_offset = vec3(.0f, 30.f, .0f); vec3 pos_offset = vec3(.0f, 30.f, .0f);
if (USE_STAIRS) if (USE_STAIRS)
{ {
vec3 new_offset = vec3(1.0f, .125f, .0f); vec3 new_offset = vec3(1.0f, .125f, .0f);
quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f); quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f);
vec3 NewPosition = pos_offset + vec3(5.0f, -29.f, 15.0f); vec3 NewPosition = pos_offset + vec3(5.0f, -29.f, 15.0f);
{ {
NewRotation = quat::fromeuler_xyz(0.f, 0.f, 30.f); NewRotation = quat::fromeuler_xyz(0.f, 0.f, 30.f);
NewPosition += vec3(4.0f, .0f, -4.0f); NewPosition += vec3(4.0f, .0f, -4.0f);
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3);
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3); Ticker::Ref(NewPhyobj);
Ticker::Ref(NewPhyobj); m_stairs_list << NewPhyobj;
m_stairs_list << NewPhyobj; }
} {
{ NewRotation = quat::fromeuler_xyz(0.f, 0.f, 40.f);
NewRotation = quat::fromeuler_xyz(0.f, 0.f, 40.f); NewPosition += vec3(4.0f, .0f, -4.0f);
NewPosition += vec3(4.0f, .0f, -4.0f); PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3);
Ticker::Ref(NewPhyobj);
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3); m_stairs_list << NewPhyobj;
Ticker::Ref(NewPhyobj); }
m_stairs_list << NewPhyobj; NewPosition = pos_offset + vec3(5.0f, -29.5f, 15.0f);
} NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f);
NewPosition = pos_offset + vec3(5.0f, -29.5f, 15.0f); for (int i=0; i < 15; i++)
NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f); {
for (int i=0; i < 15; i++) NewPosition += new_offset;
{ PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3);
NewPosition += new_offset; Ticker::Ref(NewPhyobj);
m_stairs_list << NewPhyobj;
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 3); }
Ticker::Ref(NewPhyobj); }
m_stairs_list << NewPhyobj; if (USE_WALL)
} {
} for (int i=0; i < 6; i++)
{
if (USE_WALL) vec3 NewPosition = vec3(.0f);
{ quat NewRotation = quat(1.f);
for (int i=0; i < 6; i++) PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation);
{ int idx = i/2;
vec3 NewPosition = vec3(.0f); NewPosition = pos_offset;
quat NewRotation = quat(1.f); NewPosition[idx] += offset;
offset *= -1.f;
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation); if (idx != 1)
{
int idx = i/2; vec3 NewAxis = vec3(.0f);
NewPosition = pos_offset; NewAxis[2 - idx] = 1;
NewPosition[idx] += offset; NewRotation = quat::rotate(90.f, NewAxis);
offset *= -1.f; }
NewPhyobj->SetTransform(NewPosition, NewRotation);
if (idx != 1) Ticker::Ref(NewPhyobj);
{ m_ground_list << NewPhyobj;
vec3 NewAxis = vec3(.0f); }
NewAxis[2 - idx] = 1; }
NewRotation = quat::rotate(90.f, NewAxis); PhysicsObject* BasePhyobj = NULL;
} if (USE_PLATFORM)
{
NewPhyobj->SetTransform(NewPosition, NewRotation); quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f);
Ticker::Ref(NewPhyobj); vec3 NewPosition = pos_offset + vec3(5.0f, -25.0f, -15.0f);
m_ground_list << NewPhyobj; PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1);
} m_platform_list << NewPhyobj;
} Ticker::Ref(NewPhyobj);
NewPosition = pos_offset + vec3(-15.0f, -25.0f, 5.0f);
PhysicsObject* BasePhyobj = NULL; NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1);
if (USE_PLATFORM) BasePhyobj = NewPhyobj;
{ m_platform_list << NewPhyobj;
quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f); Ticker::Ref(NewPhyobj);
vec3 NewPosition = pos_offset + vec3(5.0f, -25.0f, -15.0f); NewRotation = quat::fromeuler_xyz(0.f, 0.f, 90.f);
NewPosition = pos_offset + vec3(-20.0f, -25.0f, 5.0f);
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1); NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1);
NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), true, true);
m_platform_list << NewPhyobj; m_platform_list << NewPhyobj;
Ticker::Ref(NewPhyobj); Ticker::Ref(NewPhyobj);
//NewPosition += vec3(-0.0f, .0f, .0f);
NewPosition = pos_offset + vec3(-15.0f, -25.0f, 5.0f); //NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1);
//NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), true, false);
NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1); //m_platform_list << NewPhyobj;
BasePhyobj = NewPhyobj; //Ticker::Ref(NewPhyobj);
//NewPosition += vec3(-2.0f, .0f, .0f);
m_platform_list << NewPhyobj; //NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1);
Ticker::Ref(NewPhyobj); //NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), false, false);
//m_platform_list << NewPhyobj;
NewRotation = quat::fromeuler_xyz(0.f, 0.f, 90.f); //Ticker::Ref(NewPhyobj);
NewPosition = pos_offset + vec3(-20.0f, -25.0f, 5.0f); }
if (USE_CHARACTER)
NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1); {
quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f);
NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), true, true); vec3 NewPosition = pos_offset + vec3(-5.0f, -10.0f, 15.0f);
m_platform_list << NewPhyobj; PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 2);
Ticker::Ref(NewPhyobj); m_character_list << NewPhyobj;
Ticker::Ref(NewPhyobj);
//NewPosition += vec3(-0.0f, .0f, .0f); Input::LinkActionToKey(IPT_MOVE_FORWARD, Key::Up);
//NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1); Input::LinkActionToKey(IPT_MOVE_BACKWARD, Key::Down);
Input::LinkActionToKey(IPT_MOVE_LEFT, Key::Left);
//NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), true, false); Input::LinkActionToKey(IPT_MOVE_RIGHT, Key::Right);
//m_platform_list << NewPhyobj; Input::LinkActionToKey(IPT_MOVE_JUMP, Key::Space);
//Ticker::Ref(NewPhyobj); Input::LinkActionToKey(IPT_MOVE_UP, Key::PageUp);
Input::LinkActionToKey(IPT_MOVE_DOWN, Key::PageDown);
//NewPosition += vec3(-2.0f, .0f, .0f); //NewPhyobj->GetCharacter()->AttachTo(BasePhyobj->GetPhysic(), true, true);
//NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 1); }
if (USE_BODIES)
//NewPhyobj->GetPhysic()->AttachTo(BasePhyobj->GetPhysic(), false, false); {
//m_platform_list << NewPhyobj; for (int x=0; x < 6; x++)
//Ticker::Ref(NewPhyobj); {
} for (int y=0; y < 6; y++)
{
if (USE_CHARACTER) for (int z=0; z < 5; z++)
{ {
quat NewRotation = quat::fromeuler_xyz(0.f, 0.f, 0.f); PhysicsObject* new_physobj = new PhysicsObject(m_simulation, 1000.f,
vec3 NewPosition = pos_offset + vec3(-5.0f, -10.0f, 15.0f); vec3(-20.f, 15.f, -20.f) +
vec3(8.f * (float)x, 8.f * (float)y, 8.f * (float)z));
PhysicsObject* NewPhyobj = new PhysicsObject(m_simulation, NewPosition, NewRotation, 2); m_physobj_list << new_physobj;
Ticker::Ref(new_physobj);
m_character_list << NewPhyobj; }
Ticker::Ref(NewPhyobj); }
}
}
Input::LinkActionToKey(IPT_MOVE_FORWARD, Key::Up); if (USE_ROPE)
Input::LinkActionToKey(IPT_MOVE_BACKWARD, Key::Down); {
Input::LinkActionToKey(IPT_MOVE_LEFT, Key::Left); Array<PhysicsObject*> RopeElements;
Input::LinkActionToKey(IPT_MOVE_RIGHT, Key::Right); for (int i = 0; i < 14; i++)
Input::LinkActionToKey(IPT_MOVE_JUMP, Key::Space); {
Input::LinkActionToKey(IPT_MOVE_UP, Key::PageUp); PhysicsObject* new_physobj = new PhysicsObject(m_simulation, 1000.f,
Input::LinkActionToKey(IPT_MOVE_DOWN, Key::PageDown); vec3(0.f, 15.f, -20.f) +
vec3(0.f, 0.f, 2.f * (float)i), 1);
//NewPhyobj->GetCharacter()->AttachTo(BasePhyobj->GetPhysic(), true, true); RopeElements << new_physobj;
} m_physobj_list << new_physobj;
Ticker::Ref(new_physobj);
if (USE_BODIES) if (RopeElements.Count() > 1)
{ {
for (int x=0; x < 6; x++) EasyConstraint* new_constraint = new EasyConstraint();
{ vec3 A2B = .5f * (RopeElements[i]->GetPhysic()->GetTransform().v3.xyz -
for (int y=0; y < 6; y++) RopeElements[i - 1]->GetPhysic()->GetTransform().v3.xyz);
{ new_constraint->SetPhysObjA(RopeElements[i - 1]->GetPhysic(), lol::mat4::translate(A2B));
for (int z=0; z < 5; z++) new_constraint->SetPhysObjB(RopeElements[i]->GetPhysic(), lol::mat4::translate(-A2B));
{ new_constraint->InitConstraintToPoint2Point();
PhysicsObject* new_physobj = new PhysicsObject(m_simulation, 1000.f, new_constraint->DisableCollisionBetweenObjs(true);
vec3(-20.f, 15.f, -20.f) + new_constraint->AddToSimulation(m_simulation);
vec3(8.f * (float)x, 8.f * (float)y, 8.f * (float)z)); m_constraint_list << new_constraint;
m_physobj_list << new_physobj; }
Ticker::Ref(new_physobj); }
} }
}
}
}
if (USE_ROPE)
{
Array<PhysicsObject*> RopeElements;
for (int i = 0; i < 14; i++)
{
PhysicsObject* new_physobj = new PhysicsObject(m_simulation, 1000.f,
vec3(0.f, 15.f, -20.f) +
vec3(0.f, 0.f, 2.f * (float)i), 1);
RopeElements << new_physobj;
m_physobj_list << new_physobj;
Ticker::Ref(new_physobj);
if (RopeElements.Count() > 1)
{
EasyConstraint* new_constraint = new EasyConstraint();
vec3 A2B = .5f * (RopeElements[i]->GetPhysic()->GetTransform().v3.xyz -
RopeElements[i - 1]->GetPhysic()->GetTransform().v3.xyz);
new_constraint->SetPhysObjA(RopeElements[i - 1]->GetPhysic(), lol::mat4::translate(A2B));
new_constraint->SetPhysObjB(RopeElements[i]->GetPhysic(), lol::mat4::translate(-A2B));
new_constraint->InitConstraintToPoint2Point();
new_constraint->DisableCollisionBetweenObjs(true);
new_constraint->AddToSimulation(m_simulation);
m_constraint_list << new_constraint;
}
}
}
} }


void BtPhysTest::TickGame(float seconds) void BtPhysTest::TickGame(float seconds)
@@ -269,143 +269,143 @@ void BtPhysTest::TickGame(float seconds)
if (Input::WasReleased(Key::Escape)) if (Input::WasReleased(Key::Escape))
Ticker::Shutdown(); Ticker::Shutdown();


m_loop_value += seconds; m_loop_value += seconds;
if (m_loop_value > M_PI * 2.0f) if (m_loop_value > M_PI * 2.0f)
m_loop_value -= M_PI * 2.0f; m_loop_value -= M_PI * 2.0f;
vec3 GroundBarycenter = vec3(.0f);
vec3 GroundBarycenter = vec3(.0f); vec3 PhysObjBarycenter = vec3(.0f);
vec3 PhysObjBarycenter = vec3(.0f); float factor = .0f;
float factor = .0f; if (USE_WALL)
{
if (USE_WALL) for (int i = 0; i < m_ground_list.Count(); i++)
{ {
for (int i = 0; i < m_ground_list.Count(); i++) PhysicsObject* PhysObj = m_ground_list[i];
{ mat4 GroundMat = PhysObj->GetTransform();
PhysicsObject* PhysObj = m_ground_list[i]; GroundBarycenter += GroundMat.v3.xyz;
mat4 GroundMat = PhysObj->GetTransform(); factor += 1.f;
}
GroundBarycenter += GroundMat.v3.xyz; GroundBarycenter /= factor;
factor += 1.f; for (int i = 0; i < m_ground_list.Count(); i++)
} {
PhysicsObject* PhysObj = m_ground_list[i];
GroundBarycenter /= factor; mat4 GroundMat = PhysObj->GetTransform();
vec3 CenterToGround = GroundMat.v3.xyz - GroundBarycenter;
for (int i = 0; i < m_ground_list.Count(); i++) vec3 CenterToCam = m_camera->m_position - GroundBarycenter;
{ if (dot(normalize(CenterToCam - CenterToGround),
PhysicsObject* PhysObj = m_ground_list[i]; normalize(CenterToGround)) > 0.f)
PhysObj->SetRender(false);
mat4 GroundMat = PhysObj->GetTransform(); else
vec3 CenterToGround = GroundMat.v3.xyz - GroundBarycenter; PhysObj->SetRender(true);
vec3 CenterToCam = m_camera->m_position - GroundBarycenter; }
}
if (dot(normalize(CenterToCam - CenterToGround), if (USE_ROTATION)
normalize(CenterToGround)) > 0.f) {
PhysObj->SetRender(false); for (int i = 0; i < m_ground_list.Count(); i++)
else {
PhysObj->SetRender(true); PhysicsObject* PhysObj = m_ground_list[i];
} mat4 GroundMat = PhysObj->GetTransform();
} mat4 CenterMx = mat4::translate(GroundBarycenter);
GroundMat = inverse(CenterMx) * GroundMat;
if (USE_ROTATION) GroundMat = CenterMx *
{ mat4(quat::fromeuler_xyz(vec3(.0f, 20.f, 20.0f) * seconds))
for (int i = 0; i < m_ground_list.Count(); i++) * GroundMat;
{ PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat));
PhysicsObject* PhysObj = m_ground_list[i]; }
}
mat4 GroundMat = PhysObj->GetTransform(); if (USE_PLATFORM)
mat4 CenterMx = mat4::translate(GroundBarycenter); {
GroundMat = inverse(CenterMx) * GroundMat; for (int i = 0; i < m_platform_list.Count(); i++)
GroundMat = CenterMx * {
mat4(quat::fromeuler_xyz(vec3(.0f, 20.f, 20.0f) * seconds)) PhysicsObject* PhysObj = m_platform_list[i];
* GroundMat; mat4 GroundMat = PhysObj->GetTransform();
PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat)); if (i == 0)
} {
} GroundMat = GroundMat * mat4(quat::fromeuler_xyz(vec3(20.f, .0f, .0f) * seconds));
PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat));
if (USE_PLATFORM) }
{ else if (i == 1)
for (int i = 0; i < m_platform_list.Count(); i++) {
{ GroundMat =
PhysicsObject* PhysObj = m_platform_list[i]; mat4::translate(vec3(-15.0f, 5.0f, lol::cos(m_loop_value) * 8.f)) *
mat4(quat::fromeuler_xyz(vec3(.0f, lol::cos(m_loop_value) * 20.f, .0f)));
mat4 GroundMat = PhysObj->GetTransform(); PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat));
if (i == 0) }
{ }
GroundMat = GroundMat * mat4(quat::fromeuler_xyz(vec3(20.f, .0f, .0f) * seconds)); }
PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat)); if (USE_CHARACTER)
} {
else if (i == 1) for (int i = 0; i < m_character_list.Count(); i++)
{ {
GroundMat = PhysicsObject* PhysObj = m_character_list[i];
mat4::translate(vec3(-15.0f, 5.0f, lol::cos(m_loop_value) * 8.f)) * EasyCharacterController* Character = (EasyCharacterController*)PhysObj->GetCharacter();
mat4(quat::fromeuler_xyz(vec3(.0f, lol::cos(m_loop_value) * 20.f, .0f))); mat4 CtlrMx = Character->GetTransform();
PhysObj->SetTransform(GroundMat.v3.xyz, quat(GroundMat)); int HMovement = Input::GetStatus(IPT_MOVE_RIGHT) - Input::GetStatus(IPT_MOVE_LEFT);
} int VMovement = Input::GetStatus(IPT_MOVE_FORWARD) - Input::GetStatus(IPT_MOVE_BACKWARD);
} int RMovement = Input::GetStatus(IPT_MOVE_UP) - Input::GetStatus(IPT_MOVE_DOWN);
} vec3 CharMove = vec3((float)VMovement * seconds * 4.f, (float)RMovement * seconds * 10.f, (float)HMovement * seconds * 4.f);
if (Input::WasReleased(IPT_MOVE_JUMP))
if (USE_CHARACTER) Character->Jump();
{ Character->SetMovementForFrame(CharMove);
for (int i = 0; i < m_character_list.Count(); i++) RayCastResult HitResult;
{ if (m_simulation->RayHits(HitResult, ERT_Closest, Character->GetTransform().v3.xyz, (Character->GetTransform().v3.xyz + vec3(.0f, -1.f, .0f)), Character))
PhysicsObject* PhysObj = m_character_list[i]; Character->AttachTo(HitResult.m_collider_list[0], true, true);
EasyCharacterController* Character = (EasyCharacterController*)PhysObj->GetCharacter(); else
mat4 CtlrMx = Character->GetTransform(); Character->AttachTo(NULL);
}
int HMovement = Input::GetStatus(IPT_MOVE_RIGHT) - Input::GetStatus(IPT_MOVE_LEFT); }
int VMovement = Input::GetStatus(IPT_MOVE_FORWARD) - Input::GetStatus(IPT_MOVE_BACKWARD); if (USE_CHARACTER)
int RMovement = Input::GetStatus(IPT_MOVE_UP) - Input::GetStatus(IPT_MOVE_DOWN); {
vec3 CharMove = vec3((float)VMovement * seconds * 4.f, (float)RMovement * seconds * 10.f, (float)HMovement * seconds * 4.f); PhysObjBarycenter = vec3(.0f);
factor = .0f;
if (Input::WasReleased(IPT_MOVE_JUMP)) for (int i = 0; i < m_character_list.Count(); i++)
Character->Jump(); {
Character->SetMovementForFrame(CharMove); PhysicsObject* PhysObj = m_character_list[i];
mat4 GroundMat = PhysObj->GetTransform();
RayCastResult HitResult; PhysObjBarycenter += GroundMat.v3.xyz;
if (m_simulation->RayHits(HitResult, ERT_Closest, Character->GetTransform().v3.xyz, (Character->GetTransform().v3.xyz + vec3(.0f, -1.f, .0f)), Character)) factor += 1.f;
Character->AttachTo(HitResult.m_collider_list[0], true, true); }
else PhysObjBarycenter /= factor;
Character->AttachTo(NULL); m_camera->SetTarget(m_camera->GetTarget() + (seconds / (seconds + 0.18f)) * (PhysObjBarycenter - m_camera->GetTarget()));
} vec3 CamPosCenter = m_camera->GetTarget() + vec3(.0f, 5.0f, .0f);
} m_camera->SetPosition(CamPosCenter + normalize(m_camera->GetPosition() - CamPosCenter) * 20.0f);
}
if (USE_CHARACTER) else
{ {
PhysObjBarycenter = vec3(.0f); PhysObjBarycenter = vec3(.0f);
factor = .0f; for (int i = 0; i < m_physobj_list.Count(); i++)
{
for (int i = 0; i < m_character_list.Count(); i++) PhysicsObject* PhysObj = m_physobj_list[i];
{ mat4 GroundMat = PhysObj->GetTransform();
PhysicsObject* PhysObj = m_character_list[i]; PhysObjBarycenter += GroundMat.v3.xyz;
mat4 GroundMat = PhysObj->GetTransform(); factor += 1.f;
}
PhysObjBarycenter += GroundMat.v3.xyz; PhysObjBarycenter /= factor;
factor += 1.f; m_camera->SetTarget(PhysObjBarycenter);
} m_camera->SetPosition(GroundBarycenter + normalize(GroundBarycenter - PhysObjBarycenter) * 60.0f);
}
PhysObjBarycenter /= factor;
m_camera->SetTarget(m_camera->GetTarget() + (seconds / (seconds + 0.18f)) * (PhysObjBarycenter - m_camera->GetTarget()));
vec3 CamPosCenter = m_camera->GetTarget() + vec3(.0f, 5.0f, .0f);
m_camera->SetPosition(CamPosCenter + normalize(m_camera->GetPosition() - CamPosCenter) * 20.0f);
}
else
{
PhysObjBarycenter = vec3(.0f);
for (int i = 0; i < m_physobj_list.Count(); i++)
{
PhysicsObject* PhysObj = m_physobj_list[i];
mat4 GroundMat = PhysObj->GetTransform();
PhysObjBarycenter += GroundMat.v3.xyz;
factor += 1.f;
}
PhysObjBarycenter /= factor;
m_camera->SetTarget(PhysObjBarycenter);
m_camera->SetPosition(GroundBarycenter + normalize(GroundBarycenter - PhysObjBarycenter) * 60.0f);
}


} }


@@ -425,50 +425,50 @@ void BtPhysTest::TickDraw(float seconds)


BtPhysTest::~BtPhysTest() BtPhysTest::~BtPhysTest()
{ {
Ticker::Unref(m_camera); Ticker::Unref(m_camera);
while (m_constraint_list.Count())
while (m_constraint_list.Count()) {
{ EasyConstraint* CurPop = m_constraint_list.Last();
EasyConstraint* CurPop = m_constraint_list.Last(); m_constraint_list.Pop();
m_constraint_list.Pop(); CurPop->RemoveFromSimulation(m_simulation);
CurPop->RemoveFromSimulation(m_simulation); delete CurPop;
delete CurPop; }
} while (m_ground_list.Count())
while (m_ground_list.Count()) {
{ PhysicsObject* CurPop = m_ground_list.Last();
PhysicsObject* CurPop = m_ground_list.Last(); m_ground_list.Pop();
m_ground_list.Pop(); CurPop->GetPhysic()->RemoveFromSimulation(m_simulation);
CurPop->GetPhysic()->RemoveFromSimulation(m_simulation); Ticker::Unref(CurPop);
Ticker::Unref(CurPop); }
} while (m_stairs_list.Count())
while (m_stairs_list.Count()) {
{ PhysicsObject* CurPop = m_stairs_list.Last();
PhysicsObject* CurPop = m_stairs_list.Last(); m_stairs_list.Pop();
m_stairs_list.Pop(); CurPop->GetPhysic()->RemoveFromSimulation(m_simulation);
CurPop->GetPhysic()->RemoveFromSimulation(m_simulation); Ticker::Unref(CurPop);
Ticker::Unref(CurPop); }
} while (m_character_list.Count())
while (m_character_list.Count()) {
{ PhysicsObject* CurPop = m_character_list.Last();
PhysicsObject* CurPop = m_character_list.Last(); m_character_list.Pop();
m_character_list.Pop(); CurPop->GetCharacter()->RemoveFromSimulation(m_simulation);
CurPop->GetCharacter()->RemoveFromSimulation(m_simulation); Ticker::Unref(CurPop);
Ticker::Unref(CurPop); }
} while (m_platform_list.Count())
while (m_platform_list.Count()) {
{ PhysicsObject* CurPop = m_platform_list.Last();
PhysicsObject* CurPop = m_platform_list.Last(); m_platform_list.Pop();
m_platform_list.Pop(); CurPop->GetPhysic()->RemoveFromSimulation(m_simulation);
CurPop->GetPhysic()->RemoveFromSimulation(m_simulation); Ticker::Unref(CurPop);
Ticker::Unref(CurPop); }
} while (m_physobj_list.Count())
while (m_physobj_list.Count()) {
{ PhysicsObject* CurPop = m_physobj_list.Last();
PhysicsObject* CurPop = m_physobj_list.Last(); m_physobj_list.Pop();
m_physobj_list.Pop(); CurPop->GetPhysic()->RemoveFromSimulation(m_simulation);
CurPop->GetPhysic()->RemoveFromSimulation(m_simulation); Ticker::Unref(CurPop);
Ticker::Unref(CurPop); }
}
Ticker::Unref(m_simulation); Ticker::Unref(m_simulation);


} }


+ 12
- 12
test/BtPhysTest.h Näytä tiedosto

@@ -21,18 +21,18 @@ protected:
virtual void TickDraw(float seconds); virtual void TickDraw(float seconds);


private: private:
Camera* m_camera; Camera* m_camera;
bool m_ready; bool m_ready;
lol::phys::Simulation* m_simulation;
lol::phys::Simulation* m_simulation; Array<EasyConstraint*> m_constraint_list;
Array<EasyConstraint*> m_constraint_list; Array<PhysicsObject*> m_physobj_list;
Array<PhysicsObject*> m_physobj_list; Array<PhysicsObject*> m_ground_list;
Array<PhysicsObject*> m_ground_list; Array<PhysicsObject*> m_platform_list;
Array<PhysicsObject*> m_platform_list; Array<PhysicsObject*> m_character_list;
Array<PhysicsObject*> m_character_list; Array<PhysicsObject*> m_stairs_list;
Array<PhysicsObject*> m_stairs_list; float m_loop_value;
float m_loop_value;
}; };


#endif // __BTPHYSTEST_H__ #endif // __BTPHYSTEST_H__


+ 269
- 269
test/PhysicObject.h Näytä tiedosto

@@ -23,279 +23,279 @@ using namespace lol::phys;
class PhysicsObject : public WorldEntity class PhysicsObject : public WorldEntity
{ {
public: public:
PhysicsObject(Simulation* new_sim, const vec3 &base_location, const quat &base_rotation) PhysicsObject(Simulation* new_sim, const vec3 &base_location, const quat &base_rotation)
: m_ready(false), m_should_render(true), m_is_character(false) : m_ready(false), m_should_render(true), m_is_character(false)
{ {
m_physics = new EasyPhysic(this); m_physics = new EasyPhysic(this);
m_mesh.Compile("[sc#ddd afcb60 1 60 -.1]");
m_mesh.Compile("[sc#ddd afcb60 1 60 -.1]"); vec3 BoxSize = vec3(60.f, 1.f, 60.f);
vec3 BoxSize = vec3(60.f, 1.f, 60.f); m_physics->SetCollisionChannel(0, 0xFF);
m_physics->SetCollisionChannel(0, 0xFF); m_physics->SetShapeToBox(BoxSize);
m_physics->SetShapeToBox(BoxSize); m_physics->SetMass(.0f);
m_physics->SetMass(.0f); m_physics->SetTransform(base_location, base_rotation);
m_physics->SetTransform(base_location, base_rotation); m_physics->InitBodyToRigid(true);
m_physics->InitBodyToRigid(true); m_physics->AddToSimulation(new_sim);
m_physics->AddToSimulation(new_sim); }
} PhysicsObject(Simulation* new_sim, const vec3 &base_location, const quat &base_rotation, int dummy)
: m_ready(false), m_should_render(true), m_is_character(false)
PhysicsObject(Simulation* new_sim, const vec3 &base_location, const quat &base_rotation, int dummy) {
: m_ready(false), m_should_render(true), m_is_character(false) if (dummy == 1) //for platform purpose
{ {
if (dummy == 1) //for platform purpose m_physics = new EasyPhysic(this);
{ m_mesh.Compile("[sc#ddd afcb20 1 20 -.1]");
m_physics = new EasyPhysic(this); vec3 BoxSize = vec3(20.f, 1.f, 20.f);
m_physics->SetCollisionChannel(0, 0xFF);
m_mesh.Compile("[sc#ddd afcb20 1 20 -.1]"); m_physics->SetShapeToBox(BoxSize);
vec3 BoxSize = vec3(20.f, 1.f, 20.f); m_physics->SetMass(.0f);
m_physics->SetCollisionChannel(0, 0xFF); m_physics->SetTransform(base_location, base_rotation);
m_physics->SetShapeToBox(BoxSize); m_physics->InitBodyToRigid(true);
m_physics->SetMass(.0f); m_physics->AddToSimulation(new_sim);
m_physics->SetTransform(base_location, base_rotation); }
m_physics->InitBodyToRigid(true); else if (dummy == 2) //for character purpose
m_physics->AddToSimulation(new_sim); {
} m_character = new EasyCharacterController(this);
else if (dummy == 2) //for character purpose m_is_character = true;
{ //m_mesh.Compile("[sc#f00 afcb10 10 10 -.1]");
m_character = new EasyCharacterController(this); m_mesh.Compile(
m_is_character = true; "[sc#000 scb#000"
//m_mesh.Compile("[sc#f00 afcb10 10 10 -.1]"); //"[sc#aaa scb#aaa"
m_mesh.Compile( "[ad8 2 0 rx180 ty-1]"
"[sc#000 scb#000" "[asph8 .5 .5 .5 ty1]"
//"[sc#aaa scb#aaa" "[ac32 2 .5 .5 0 0]"
"[ad8 2 0 rx180 ty-1]" "[asph6 .1 .1 .1 ty.9 tx.5 tz.15]"
"[asph8 .5 .5 .5 ty1]" "[asph6 .1 .1 .1 ty.9 tx.5 tz-.15]"
"[ac32 2 .5 .5 0 0]" "[asph8 .05 .5 .05 ty.6 tz.5]"
"[asph6 .1 .1 .1 ty.9 tx.5 tz.15]" "[asph8 .05 .5 .05 ty.6 tz-.5]"
"[asph6 .1 .1 .1 ty.9 tx.5 tz-.15]" "]"
"[asph8 .05 .5 .05 ty.6 tz.5]" "[sc#fd0 scb#fd0"
"[asph8 .05 .5 .05 ty.6 tz-.5]" "[ac8 .4 .1 0 0 0 ty.25 rz-90 ty.7 tx.5]"
"]" "]"
"[sc#fd0 scb#fd0" "["
"[ac8 .4 .1 0 0 0 ty.25 rz-90 ty.7 tx.5]" "[sc#fff scb#fff"
"]" "[ad8 2 0 rx180 ty-1]"
"[" "[asph8 .5 .5 .5 ty1]"
"[sc#fff scb#fff" "[ac32 1.9 .5 .5 0 0]"
"[ad8 2 0 rx180 ty-1]" "]"
"[asph8 .5 .5 .5 ty1]" " ty-.1 tx.05]"
"[ac32 1.9 .5 .5 0 0]" );
"]" vec3 BoxSize = vec3(1.f, 2.f, 1.f);
" ty-.1 tx.05]" m_character->SetCollisionChannel(0, 0xFF);
); m_character->SetShapeToCapsule(BoxSize.x, BoxSize.y);
vec3 BoxSize = vec3(1.f, 2.f, 1.f); m_character->SetMass(.0f);
m_character->SetCollisionChannel(0, 0xFF); //m_character->SetStepHeight(1.f);
m_character->SetShapeToCapsule(BoxSize.x, BoxSize.y); m_character->SetTransform(base_location, base_rotation);
m_character->SetMass(.0f); m_character->InitBodyToGhost();
//m_character->SetStepHeight(1.f); m_character->AddToSimulation(new_sim);
m_character->SetTransform(base_location, base_rotation); }
m_character->InitBodyToGhost(); else if (dummy == 3) //for Stairs purpose
m_character->AddToSimulation(new_sim); {
} m_physics = new EasyPhysic(this);
else if (dummy == 3) //for Stairs purpose m_mesh.Compile("[sc#aae afcb4 .25 4 -.01]");
{ vec3 BoxSize = vec3(4.f, .25f, 4.f);
m_physics = new EasyPhysic(this); m_physics->SetCollisionChannel(0, 0xFF);
m_physics->SetShapeToBox(BoxSize);
m_mesh.Compile("[sc#aae afcb4 .25 4 -.01]"); m_physics->SetMass(.0f);
vec3 BoxSize = vec3(4.f, .25f, 4.f); m_physics->SetTransform(base_location, base_rotation);
m_physics->SetCollisionChannel(0, 0xFF); m_physics->InitBodyToRigid(true);
m_physics->SetShapeToBox(BoxSize); m_physics->AddToSimulation(new_sim);
m_physics->SetMass(.0f); }
m_physics->SetTransform(base_location, base_rotation); }
m_physics->InitBodyToRigid(true); PhysicsObject(Simulation* new_sim, float base_mass, const vec3 &base_location, int RandValue = -1)
m_physics->AddToSimulation(new_sim); : m_ready(false), m_should_render(true), m_is_character(false)
} {
} Array<char const *> MeshRand;
//MeshRand << "[sc#add afcb2 2 2 -.1]";
PhysicsObject(Simulation* new_sim, float base_mass, const vec3 &base_location, int RandValue = -1) //MeshRand << "[sc#dad afcb2 2 2 -.1]";
: m_ready(false), m_should_render(true), m_is_character(false) //MeshRand << "[sc#dda afcb2 2 2 -.1]";
{ //MeshRand << "[sc#daa afcb2 2 2 -.1]";
Array<char const *> MeshRand; //MeshRand << "[sc#ada afcb2 2 2 -.1]";
//MeshRand << "[sc#aad afcb2 2 2 -.1]";
//MeshRand << "[sc#add afcb2 2 2 -.1]"; MeshRand << "[sc#add afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
//MeshRand << "[sc#dad afcb2 2 2 -.1]"; MeshRand << "[sc#dad afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
//MeshRand << "[sc#dda afcb2 2 2 -.1]"; MeshRand << "[sc#dda afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
//MeshRand << "[sc#daa afcb2 2 2 -.1]"; MeshRand << "[sc#daa afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
//MeshRand << "[sc#ada afcb2 2 2 -.1]"; MeshRand << "[sc#ada afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
//MeshRand << "[sc#aad afcb2 2 2 -.1]"; MeshRand << "[sc#aad afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]";
MeshRand << "[sc#add afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; int SphereLimit = MeshRand.Count();
MeshRand << "[sc#dad afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; MeshRand << "[sc#add asph1 2 2 2]";
MeshRand << "[sc#dda afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; MeshRand << "[sc#dad asph1 2 2 2]";
MeshRand << "[sc#daa afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; MeshRand << "[sc#dda asph1 2 2 2]";
MeshRand << "[sc#ada afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; MeshRand << "[sc#daa asph1 2 2 2]";
MeshRand << "[sc#aad afcb1.7 1.7 1.7 0][sc#000 afcb1.9 1.9 1.9 0 sx-1 sy-1 sz-1]"; MeshRand << "[sc#ada asph1 2 2 2]";
MeshRand << "[sc#aad asph1 2 2 2]";
int SphereLimit = MeshRand.Count(); int ConeLimit = MeshRand.Count();
MeshRand << "[sc#add scb#add ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#add asph1 2 2 2]"; MeshRand << "[sc#dad scb#dad ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#dad asph1 2 2 2]"; MeshRand << "[sc#dda scb#dda ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#dda asph1 2 2 2]"; MeshRand << "[sc#daa scb#daa ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#daa asph1 2 2 2]"; MeshRand << "[sc#ada scb#ada ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#ada asph1 2 2 2]"; MeshRand << "[sc#aad scb#aad ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]";
MeshRand << "[sc#aad asph1 2 2 2]"; int CylLimit = MeshRand.Count();
MeshRand << "[sc#add scb#add ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
int ConeLimit = MeshRand.Count(); MeshRand << "[sc#dad scb#dad ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
MeshRand << "[sc#dda scb#dda ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
MeshRand << "[sc#add scb#add ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; MeshRand << "[sc#daa scb#daa ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
MeshRand << "[sc#dad scb#dad ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; MeshRand << "[sc#ada scb#ada ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
MeshRand << "[sc#dda scb#dda ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; MeshRand << "[sc#aad scb#aad ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]";
MeshRand << "[sc#daa scb#daa ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; int CapsLimit = MeshRand.Count();
MeshRand << "[sc#ada scb#ada ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; MeshRand << "[sc#add scb#add acap1 2 1]";
MeshRand << "[sc#aad scb#aad ad1 2 0 rx180 ty-1 ac4 2 2 0 0 0]"; MeshRand << "[sc#dad scb#dad acap1 2 1]";
MeshRand << "[sc#dda scb#dda acap1 2 1]";
int CylLimit = MeshRand.Count(); MeshRand << "[sc#daa scb#daa acap1 2 1]";
MeshRand << "[sc#ada scb#ada acap1 2 1]";
MeshRand << "[sc#add scb#add ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; MeshRand << "[sc#aad scb#aad acap1 2 1]";
MeshRand << "[sc#dad scb#dad ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; switch (RandValue)
MeshRand << "[sc#dda scb#dda ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; {
MeshRand << "[sc#daa scb#daa ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; case 0:
MeshRand << "[sc#ada scb#ada ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; {
MeshRand << "[sc#aad scb#aad ad1 2 0 rx180 ty-1 my ac4 2 2 2 0 0]"; RandValue = (int)(lol::RandF() * (SphereLimit - 1));
break;
int CapsLimit = MeshRand.Count(); }
case 1:
MeshRand << "[sc#add scb#add acap1 2 1]"; {
MeshRand << "[sc#dad scb#dad acap1 2 1]"; RandValue = SphereLimit + (int)(lol::RandF() * ((ConeLimit - SphereLimit) - 1));
MeshRand << "[sc#dda scb#dda acap1 2 1]"; break;
MeshRand << "[sc#daa scb#daa acap1 2 1]"; }
MeshRand << "[sc#ada scb#ada acap1 2 1]"; case 2:
MeshRand << "[sc#aad scb#aad acap1 2 1]"; {
RandValue = ConeLimit + (int)(lol::RandF() * ((CylLimit - ConeLimit) - 1));
switch (RandValue) break;
{ }
case 0: case 3:
{ {
RandValue = (int)(lol::RandF() * (SphereLimit - 1)); RandValue = CylLimit + (int)(lol::RandF() * ((CapsLimit - CylLimit) - 1));
break; break;
} }
case 1: case 4:
{ {
RandValue = SphereLimit + (int)(lol::RandF() * ((ConeLimit - SphereLimit) - 1)); RandValue = CapsLimit + (int)(lol::RandF() * ((MeshRand.Count() - CapsLimit) - 1));
break; break;
} }
case 2: default:
{ {
RandValue = ConeLimit + (int)(lol::RandF() * ((CylLimit - ConeLimit) - 1)); RandValue = (int)(lol::RandF() * (MeshRand.Count() - 1));
break; }
} }
case 3: m_physics = new EasyPhysic(this);
{ m_mesh.Compile(MeshRand[RandValue]);
RandValue = CylLimit + (int)(lol::RandF() * ((CapsLimit - CylLimit) - 1)); vec3 BoxSize = vec3(2.0f);
break; int ColGroup = 1;
} if (RandValue < SphereLimit)
case 4: {
{ m_physics->SetShapeToBox(BoxSize);
RandValue = CapsLimit + (int)(lol::RandF() * ((MeshRand.Count() - CapsLimit) - 1)); ColGroup += 0;
break; }
} else if (RandValue < ConeLimit)
default: {
{ m_physics->SetShapeToSphere(BoxSize.x * 2.f);
RandValue = (int)(lol::RandF() * (MeshRand.Count() - 1)); ColGroup += 1;
} }
} else if (RandValue < CylLimit)
{
m_physics = new EasyPhysic(this); m_physics->SetShapeToCone(BoxSize.x, BoxSize.y);
ColGroup += 2;
m_mesh.Compile(MeshRand[RandValue]); }
vec3 BoxSize = vec3(2.0f); else if (RandValue < CapsLimit)
int ColGroup = 1; {
if (RandValue < SphereLimit) m_physics->SetShapeToCylinder(BoxSize);
{ ColGroup += 3;
m_physics->SetShapeToBox(BoxSize); }
ColGroup += 0; else
} {
else if (RandValue < ConeLimit) m_physics->SetShapeToCapsule(BoxSize.x, BoxSize.y);
{ ColGroup += 4;
m_physics->SetShapeToSphere(BoxSize.x * 2.f); }
ColGroup += 1; m_physics->SetCollisionChannel(0, 0xFF);
} //m_physics->SetCollisionChannel(ColGroup, (1<<ColGroup)|(1));
else if (RandValue < CylLimit) m_physics->SetMass(base_mass);
{ m_physics->SetTransform(base_location);
m_physics->SetShapeToCone(BoxSize.x, BoxSize.y); m_physics->InitBodyToRigid();
ColGroup += 2; m_physics->AddToSimulation(new_sim);
} }
else if (RandValue < CapsLimit) void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f)))
{ {
m_physics->SetShapeToCylinder(BoxSize); if (m_is_character)
ColGroup += 3; m_character->SetTransform(base_location, base_rotation);
} else
else m_physics->SetTransform(base_location, base_rotation);
{ }
m_physics->SetShapeToCapsule(BoxSize.x, BoxSize.y); lol::mat4 GetTransform()
ColGroup += 4; {
} if (m_is_character)
return m_character->GetTransform();
m_physics->SetCollisionChannel(0, 0xFF); else
//m_physics->SetCollisionChannel(ColGroup, (1<<ColGroup)|(1)); return m_physics->GetTransform();
m_physics->SetMass(base_mass); }
m_physics->SetTransform(base_location); void SetRender(bool should_render)
m_physics->InitBodyToRigid(); {
m_physics->AddToSimulation(new_sim); m_should_render = should_render;
} }
EasyMesh *GetMesh() { return &m_mesh; }
void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f))) EasyPhysic *GetPhysic() { return m_physics; }
{ EasyCharacterController *GetCharacter() { return m_character; }
if (m_is_character) ~PhysicsObject()
m_character->SetTransform(base_location, base_rotation); {
else }
m_physics->SetTransform(base_location, base_rotation); char const *GetName() { return "<PhysicsObject>"; }
}
lol::mat4 GetTransform()
{
if (m_is_character)
return m_character->GetTransform();
else
return m_physics->GetTransform();
}
void SetRender(bool should_render)
{
m_should_render = should_render;
}
EasyMesh *GetMesh() { return &m_mesh; }
EasyPhysic *GetPhysic() { return m_physics; }
EasyCharacterController *GetCharacter() { return m_character; }
~PhysicsObject()
{
}
char const *GetName() { return "<PhysicsObject>"; }


protected: protected:
virtual void TickGame(float seconds) virtual void TickGame(float seconds)
{ {
WorldEntity::TickGame(seconds); WorldEntity::TickGame(seconds);
} }
virtual void TickDraw(float seconds)
virtual void TickDraw(float seconds) {
{ WorldEntity::TickDraw(seconds);
WorldEntity::TickDraw(seconds); if (!m_ready)
{
if (!m_ready) m_mesh.MeshConvert();
{ m_ready = true;
m_mesh.MeshConvert(); }
m_ready = true; if (m_should_render)
} {
if (m_is_character)
if (m_should_render) m_mesh.Render(m_character->GetTransform());
{ else
if (m_is_character) m_mesh.Render(m_physics->GetTransform());
m_mesh.Render(m_character->GetTransform()); }
else }
m_mesh.Render(m_physics->GetTransform());
}
}


private: private:
//Base datas //Base datas
EasyMesh m_mesh; EasyMesh m_mesh;
EasyPhysic* m_physics; EasyPhysic* m_physics;
EasyCharacterController* m_character; EasyCharacterController* m_character;
bool m_ready;
bool m_ready; bool m_should_render;
bool m_should_render; bool m_is_character;
bool m_is_character;
}; };


#endif /* __PHYSICOBJECT_H__ */ #endif /* __PHYSICOBJECT_H__ */


+ 245
- 245
test/Physics/Include/BulletCharacterController.h Näytä tiedosto

@@ -29,259 +29,259 @@
namespace lol namespace lol
{ {


namespace phys namespace phys
{ {


#ifdef USE_LOL_CTRLR_CHARAC #ifdef USE_LOL_CTRLR_CHARAC
#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET


//SweepCallback used for Swweep Tests. //SweepCallback used for Swweep Tests.
class ClosestNotMeConvexResultCallback : public btCollisionWorld::ClosestConvexResultCallback class ClosestNotMeConvexResultCallback : public btCollisionWorld::ClosestConvexResultCallback
{ {
public: public:
ClosestNotMeConvexResultCallback(btCollisionObject* NewMe, const vec3& NewUp, float MinSlopeDot) : ClosestNotMeConvexResultCallback(btCollisionObject* NewMe, const vec3& NewUp, float MinSlopeDot) :
btCollisionWorld::ClosestConvexResultCallback(LOL2BTU_VEC3(vec3(.0f)), LOL2BTU_VEC3(vec3(.0f))), btCollisionWorld::ClosestConvexResultCallback(LOL2BTU_VEC3(vec3(.0f)), LOL2BTU_VEC3(vec3(.0f))),
m_me(NewMe), m_me(NewMe),
m_up(NewUp), m_up(NewUp),
m_min_slope_dot(MinSlopeDot) { } m_min_slope_dot(MinSlopeDot) { }
virtual btScalar addSingleResult(btCollisionWorld::LocalConvexResult& ConvexResult, bool NormalInWorld)
virtual btScalar addSingleResult(btCollisionWorld::LocalConvexResult& ConvexResult, bool NormalInWorld) {
{ //We hit ourselves, FAIL
//We hit ourselves, FAIL if (ConvexResult.m_hitCollisionObject == m_me)
if (ConvexResult.m_hitCollisionObject == m_me) return btScalar(1.f);
return btScalar(1.f); vec3 WorldHitNomal(.0f);
if (NormalInWorld)
vec3 WorldHitNomal(.0f); WorldHitNomal = BT2LOL_VEC3(ConvexResult.m_hitNormalLocal);
if (NormalInWorld) else //need to transform Normal into worldspace
WorldHitNomal = BT2LOL_VEC3(ConvexResult.m_hitNormalLocal); {
else //need to transform Normal into worldspace btVector3 TmpWorldHitNormal = ConvexResult.m_hitCollisionObject->getWorldTransform().getBasis() * ConvexResult.m_hitNormalLocal;
{ WorldHitNomal = BT2LOL_VEC3(TmpWorldHitNormal);
btVector3 TmpWorldHitNormal = ConvexResult.m_hitCollisionObject->getWorldTransform().getBasis() * ConvexResult.m_hitNormalLocal; }
WorldHitNomal = BT2LOL_VEC3(TmpWorldHitNormal); float DotUp = dot(m_up, WorldHitNomal);
} //We hit below the accepted slope_dot, FAIL
if (DotUp < m_min_slope_dot)
float DotUp = dot(m_up, WorldHitNomal); return btScalar(1.f);
//We hit below the accepted slope_dot, FAIL //Continue to next.
if (DotUp < m_min_slope_dot) return ClosestConvexResultCallback::addSingleResult(ConvexResult, NormalInWorld);
return btScalar(1.f); }
protected:
//Continue to next. btCollisionObject* m_me;
return ClosestConvexResultCallback::addSingleResult(ConvexResult, NormalInWorld); const vec3 m_up;
} float m_min_slope_dot;
protected: };
btCollisionObject* m_me; ///BulletKinematicCharacterController is an object that supports a sliding motion in a world.
const vec3 m_up; ///It uses a ghost object and convex sweep test to test for upcoming collisions. This is combined with discrete collision detection to recover from penetrations.
float m_min_slope_dot; ///Interaction between btKinematicCharacterController and dynamic rigid bodies needs to be explicity implemented by the user.
}; class BulletKinematicCharacterController : public btActionInterface
{
///BulletKinematicCharacterController is an object that supports a sliding motion in a world. public:
///It uses a ghost object and convex sweep test to test for upcoming collisions. This is combined with discrete collision detection to recover from penetrations. BulletKinematicCharacterController(btPairCachingGhostObject* NewGhostObject, btConvexShape* NewConvexShape, float NewStepHeight, int NewUpAxis=1)
///Interaction between btKinematicCharacterController and dynamic rigid bodies needs to be explicity implemented by the user. {
class BulletKinematicCharacterController : public btActionInterface m_convex_shape = NewConvexShape;
{ m_i_up_axis = NewUpAxis;
public: m_ghost_object = NewGhostObject;
BulletKinematicCharacterController(btPairCachingGhostObject* NewGhostObject, btConvexShape* NewConvexShape, float NewStepHeight, int NewUpAxis=1) m_step_height = NewStepHeight;
{ m_added_margin = 0.02f;
m_convex_shape = NewConvexShape; m_walk_direction = vec3(.0f, .0f, .0f);
m_i_up_axis = NewUpAxis; m_do_gobject_sweep_test = true;
m_ghost_object = NewGhostObject; m_turn_angle = .0f;
m_step_height = NewStepHeight; m_use_walk_direction = false; // Should remove walk direction, this doesn't work correctly.
m_velocity_time_interval = .0f;
m_added_margin = 0.02f; m_vertical_velocity = .0f;
m_walk_direction = vec3(.0f, .0f, .0f); m_vertical_offset = .0f;
m_do_gobject_sweep_test = true; m_f_gravity = 9.8f * 3.f; // 3G acceleration.
m_turn_angle = .0f; m_fall_speed = 55.f; // Terminal velocity of a sky diver in m/s.
m_use_walk_direction = false; // Should remove walk direction, this doesn't work correctly. m_jump_speed = 10.f; // ?
m_velocity_time_interval = .0f; m_was_on_ground = false;
m_vertical_velocity = .0f; m_was_jumping = false;
m_vertical_offset = .0f; SetMaxSlope(45.f);
m_f_gravity = 9.8f * 3.f; // 3G acceleration. }
m_fall_speed = 55.f; // Terminal velocity of a sky diver in m/s. ~BulletKinematicCharacterController() { }
m_jump_speed = 10.f; // ? protected:
m_was_on_ground = false; static vec3* GetUpAxisDirections()
m_was_jumping = false; {
SetMaxSlope(45.f); static vec3 sUpAxisDirection[3] = { vec3(1.0f, 0.0f, 0.0f), vec3(0.0f, 1.0f, 0.0f), vec3(0.0f, 0.0f, 1.0f) };
} return sUpAxisDirection;
~BulletKinematicCharacterController() { } }
//--------------------------
protected: //CONVENIENCE FUNCTIONS
//--
static vec3* GetUpAxisDirections() //Returns the reflection Direction of a ray going 'Direction' hitting a surface with Normal 'Normal' from: http://www-cs-students.stanford.edu/~adityagp/final/node3.html
{ vec3 GetReflectedDir(const vec3& Direction, const vec3& Normal)
static vec3 sUpAxisDirection[3] = { vec3(1.0f, 0.0f, 0.0f), vec3(0.0f, 1.0f, 0.0f), vec3(0.0f, 0.0f, 1.0f) }; {
return Direction - (2.f * dot(Direction, Normal) * Normal);
return sUpAxisDirection; }
} //Returns the portion of 'direction' that is parallel to 'normal'
vec3 ProjectDirOnNorm(const vec3& Direction, const vec3& Normal)
//-------------------------- {
//CONVENIENCE FUNCTIONS return Normal * dot(Direction, Normal);
//-- }
//Returns the portion of 'Direction' that is perpindicular to 'Normal'
//Returns the reflection Direction of a ray going 'Direction' hitting a surface with Normal 'Normal' from: http://www-cs-students.stanford.edu/~adityagp/final/node3.html vec3 ProjectDirOnNormPerpindicular(const vec3& Direction, const vec3& Normal)
vec3 GetReflectedDir(const vec3& Direction, const vec3& Normal) {
{ return Direction - ProjectDirOnNorm(Direction, Normal);
return Direction - (2.f * dot(Direction, Normal) * Normal); }
} //Returns Ghost Object. -duh-
//Returns the portion of 'direction' that is parallel to 'normal' btPairCachingGhostObject* GetGhostObject()
vec3 ProjectDirOnNorm(const vec3& Direction, const vec3& Normal) {
{ return m_ghost_object;
return Normal * dot(Direction, Normal); }
} //"Real" war functions
//Returns the portion of 'Direction' that is perpindicular to 'Normal' bool RecoverFromPenetration(btCollisionWorld* CollisionWorld);
vec3 ProjectDirOnNormPerpindicular(const vec3& Direction, const vec3& Normal) void UpdateTargetOnHit(const vec3& hit_normal, float TangentMag = .0f, float NormalMag = 1.f);
{ void DoMove(btCollisionWorld* CollisionWorld, const vec3& MoveStep, float DeltaTime);
return Direction - ProjectDirOnNorm(Direction, Normal); public:
} ///btActionInterface interface : KEEP IN camelCase
//Returns Ghost Object. -duh- virtual void updateAction(btCollisionWorld* CollisionWorld, float deltaTime)
btPairCachingGhostObject* GetGhostObject() {
{ PreStep(CollisionWorld);
return m_ghost_object; PlayerStep(CollisionWorld, deltaTime);
} }
//not in the interface, but called above
//"Real" war functions void PreStep(btCollisionWorld* CollisionWorld);
bool RecoverFromPenetration(btCollisionWorld* CollisionWorld); void PlayerStep(btCollisionWorld* CollisionWorld, float DeltaTime);
void UpdateTargetOnHit(const vec3& hit_normal, float TangentMag = .0f, float NormalMag = 1.f); ///btActionInterface interface : KEEP IN camelCase
void DoMove(btCollisionWorld* CollisionWorld, const vec3& MoveStep, float DeltaTime); void debugDraw(btIDebugDraw* debugDrawer) { }
void SetUpAxis(int NewAxis)
public: {
///btActionInterface interface : KEEP IN camelCase if (NewAxis < 0)
virtual void updateAction(btCollisionWorld* CollisionWorld, float deltaTime) NewAxis = 0;
{ if (NewAxis > 2)
PreStep(CollisionWorld); NewAxis = 2;
PlayerStep(CollisionWorld, deltaTime); m_i_up_axis = NewAxis;
} }
//!!!!!! SHOULD DITCH THAT !!!!!!
//not in the interface, but called above //This should probably be called setPositionIncrementPerSimulatorStep.
void PreStep(btCollisionWorld* CollisionWorld); //This is neither a Direction nor a velocity, but the amount to
void PlayerStep(btCollisionWorld* CollisionWorld, float DeltaTime); //increment the position each simulation iteration, regardless
//of DeltaTime.
///btActionInterface interface : KEEP IN camelCase //This call will Reset any velocity set by SetVelocityForTimeInterval().
void debugDraw(btIDebugDraw* debugDrawer) { } virtual void SetWalkDirection(const vec3& walkDirection)
{
void SetUpAxis(int NewAxis) m_use_walk_direction = true;
{ m_walk_direction = walkDirection;
if (NewAxis < 0) m_normalized_direction = normalize(m_walk_direction);
NewAxis = 0; }
if (NewAxis > 2) //Caller provides a velocity with which the character should MoveStep for
NewAxis = 2; //the given time period. After the time period, velocity is Reset
m_i_up_axis = NewAxis; //to zero.
} //This call will Reset any walk Direction set by SetWalkDirection().
//Negative time intervals will result in no motion.
//!!!!!! SHOULD DITCH THAT !!!!!! virtual void SetVelocityForTimeInterval(const vec3& velocity, float timeInterval)
//This should probably be called setPositionIncrementPerSimulatorStep. {
//This is neither a Direction nor a velocity, but the amount to m_use_walk_direction = false;
//increment the position each simulation iteration, regardless m_walk_direction = velocity;
//of DeltaTime. m_normalized_direction = normalize(m_walk_direction);
//This call will Reset any velocity set by SetVelocityForTimeInterval(). m_velocity_time_interval = timeInterval;
virtual void SetWalkDirection(const vec3& walkDirection) }
{ //Usefulness ?
m_use_walk_direction = true; void Reset() { }
m_walk_direction = walkDirection; void Warp(const vec3& NewOrigin)
m_normalized_direction = normalize(m_walk_direction); {
} btTransform NewTransform;
NewTransform.setIdentity();
//Caller provides a velocity with which the character should MoveStep for NewTransform.setOrigin(LOL2BTU_VEC3(NewOrigin));
//the given time period. After the time period, velocity is Reset m_ghost_object->setWorldTransform(NewTransform);
//to zero. }
//This call will Reset any walk Direction set by SetWalkDirection(). //External Setup
//Negative time intervals will result in no motion. //--
virtual void SetVelocityForTimeInterval(const vec3& velocity, float timeInterval) void SetFallSpeed(float NewFallSpeed) { m_fall_speed = NewFallSpeed; }
{ void SetJumpSpeed(float NewJumpSpeed) { m_jump_speed = NewJumpSpeed; }
m_use_walk_direction = false; void SetMaxJumpHeight(float NewMaxJumpHeight) { m_max_jump_height = NewMaxJumpHeight; }
m_walk_direction = velocity; //Jump logic will go in EasyCC
m_normalized_direction = normalize(m_walk_direction); bool CanJump() const { return OnGround(); }
m_velocity_time_interval = timeInterval; void Jump();
} //NewGravity functions
void SetGravity(float NewGravity) { m_f_gravity = NewGravity; }
//Usefulness ? float GetGravity() const { return m_f_gravity; }
void Reset() { } //The max slope determines the maximum angle that the controller can walk up.
void Warp(const vec3& NewOrigin) //The slope angle is measured in radians.
{ void SetMaxSlope(float NewSlopeRadians) { m_max_slope_radians = NewSlopeRadians; m_max_slope_cosine = lol::cos(NewSlopeRadians); }
btTransform NewTransform; float GetMaxSlope() const { return m_max_slope_radians; }
NewTransform.setIdentity(); void SetUseGhostSweepTest(bool UseGObjectSweepTest) { m_do_gobject_sweep_test = UseGObjectSweepTest; }
NewTransform.setOrigin(LOL2BTU_VEC3(NewOrigin)); bool OnGround() const { return m_vertical_velocity == .0f && m_vertical_offset == .0f; }
m_ghost_object->setWorldTransform(NewTransform); private:
} btPairCachingGhostObject* m_ghost_object;
btConvexShape* m_convex_shape; //is also in m_ghost_object, but it needs to be convex, so we store it here to avoid upcast
//External Setup //keep track of the contact manifolds
//-- btManifoldArray m_manifold_array;
float m_half_height;
void SetFallSpeed(float NewFallSpeed) { m_fall_speed = NewFallSpeed; } float m_velocity_time_interval;
void SetJumpSpeed(float NewJumpSpeed) { m_jump_speed = NewJumpSpeed; } float m_vertical_velocity;
void SetMaxJumpHeight(float NewMaxJumpHeight) { m_max_jump_height = NewMaxJumpHeight; } float m_vertical_offset;
float m_fall_speed;
//Jump logic will go in EasyCC float m_jump_speed;
bool CanJump() const { return OnGround(); } float m_max_jump_height;
void Jump(); float m_max_slope_radians; // Slope angle that is set (used for returning the exact value)
float m_max_slope_cosine; // Cosine equivalent of m_max_slope_radians (calculated once when set, for optimization)
//NewGravity functions float m_f_gravity;
void SetGravity(float NewGravity) { m_f_gravity = NewGravity; } float m_turn_angle;
float GetGravity() const { return m_f_gravity; } float m_step_height;
float m_added_margin;//@todo: remove this and fix the code
//The max slope determines the maximum angle that the controller can walk up. ///this is the desired walk Direction, set by the user
//The slope angle is measured in radians. vec3 m_walk_direction;
void SetMaxSlope(float NewSlopeRadians) { m_max_slope_radians = NewSlopeRadians; m_max_slope_cosine = lol::cos(NewSlopeRadians); } vec3 m_normalized_direction;
float GetMaxSlope() const { return m_max_slope_radians; } //some internal variables
vec3 m_current_position;
void SetUseGhostSweepTest(bool UseGObjectSweepTest) { m_do_gobject_sweep_test = UseGObjectSweepTest; } float m_current_step_offset;
vec3 m_target_position;
bool OnGround() const { return m_vertical_velocity == .0f && m_vertical_offset == .0f; } vec3 m_touching_normal;
bool m_touching_contact;
private: bool m_was_on_ground;
bool m_was_jumping;
btPairCachingGhostObject* m_ghost_object; bool m_do_gobject_sweep_test;
btConvexShape* m_convex_shape; //is also in m_ghost_object, but it needs to be convex, so we store it here to avoid upcast bool m_use_walk_direction;
int m_i_up_axis;
//keep track of the contact manifolds //---------------------------------------------------------------------
btManifoldArray m_manifold_array; //NEW INTERNAL VARS
//---------------------------------------------------------------------
float m_half_height; //Gravity in vec3
float m_velocity_time_interval; vec3 m_gravity;
float m_vertical_velocity; //Current Velocity
float m_vertical_offset; vec3 m_velocity;
float m_fall_speed; };
float m_jump_speed;
float m_max_jump_height;
float m_max_slope_radians; // Slope angle that is set (used for returning the exact value)
float m_max_slope_cosine; // Cosine equivalent of m_max_slope_radians (calculated once when set, for optimization)
float m_f_gravity;
float m_turn_angle;
float m_step_height;
float m_added_margin;//@todo: remove this and fix the code
///this is the desired walk Direction, set by the user
vec3 m_walk_direction;
vec3 m_normalized_direction;
//some internal variables
vec3 m_current_position;
float m_current_step_offset;
vec3 m_target_position;
vec3 m_touching_normal;
bool m_touching_contact;
bool m_was_on_ground;
bool m_was_jumping;
bool m_do_gobject_sweep_test;
bool m_use_walk_direction;
int m_i_up_axis;
//---------------------------------------------------------------------
//NEW INTERNAL VARS
//---------------------------------------------------------------------
//Gravity in vec3
vec3 m_gravity;
//Current Velocity
vec3 m_velocity;
};


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET
#endif // USE_LOL_CTRLR_CHARAC #endif // USE_LOL_CTRLR_CHARAC


} /* namespace phys */ } /* namespace phys */


} /* namespace lol */ } /* namespace lol */




+ 56
- 56
test/Physics/Include/EasyCharacterController.h Näytä tiedosto

@@ -34,81 +34,81 @@ namespace phys
{ {


class EasyCharacterController : public EasyPhysic, class EasyCharacterController : public EasyPhysic,
public Entity public Entity
{ {


friend class Simulation; friend class Simulation;
friend class EasyPhysic; friend class EasyPhysic;


#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET


public: public:
EasyCharacterController(WorldEntity* NewOwnerEntity) : EasyCharacterController(WorldEntity* NewOwnerEntity) :
EasyPhysic(NewOwnerEntity), EasyPhysic(NewOwnerEntity),
m_pair_caching_object(NULL), m_pair_caching_object(NULL),
m_character(NULL), m_character(NULL),
m_step_height(.0f), m_step_height(.0f),
m_base_is_updating(false), m_base_is_updating(false),
m_base_cached_movement(vec3(0.f)), m_base_cached_movement(vec3(0.f)),
m_frame_cached_movement(vec3(0.f)), m_frame_cached_movement(vec3(0.f)),
m_walk_velocity(vec3(0.f)), m_walk_velocity(vec3(0.f)),
m_current_velocity(vec3(0.f)) m_current_velocity(vec3(0.f))
{ {
m_gamegroup = GAMEGROUP_EZP_CHAR_CTRLR; m_gamegroup = GAMEGROUP_EZP_CHAR_CTRLR;
m_up_axis = 1; m_up_axis = 1;
m_gravity = vec3(.0f, -9.81f, .0f); m_gravity = vec3(.0f, -9.81f, .0f);
m_walk_velocity_damping = 0.2f; m_walk_velocity_damping = 0.2f;
} }
~EasyCharacterController() ~EasyCharacterController()
{ {
delete m_character; delete m_character;
} }
virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false);
virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false); virtual void InitBodyToGhost();
virtual void InitBodyToGhost(); virtual void AddToSimulation(class Simulation* current_simulation);
virtual void AddToSimulation(class Simulation* current_simulation); virtual void RemoveFromSimulation(class Simulation* current_simulation);
virtual void RemoveFromSimulation(class Simulation* current_simulation); virtual void SetMovementForFrame(vec3 const &MoveQuantity);
virtual void SetMovementForFrame(vec3 const &MoveQuantity); virtual void Jump();
virtual void Jump(); virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation);
virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation);
protected: protected:
virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix); virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix);
virtual char const *GetName(); virtual char const *GetName();
public: public:
virtual void TickGame(float seconds); virtual void TickGame(float seconds);


protected: protected:


virtual btGhostObject* GetGhostObjectInstance(); virtual btGhostObject* GetGhostObjectInstance();


btPairCachingGhostObject* m_pair_caching_object; btPairCachingGhostObject* m_pair_caching_object;
//btKinematicCharacterController* m_character; //btKinematicCharacterController* m_character;
BulletKinematicCharacterController* m_character; BulletKinematicCharacterController* m_character;


float m_step_height; float m_step_height;
int m_up_axis; int m_up_axis;
bool m_base_is_updating; bool m_base_is_updating;
vec3 m_base_cached_movement; vec3 m_base_cached_movement;
vec3 m_frame_cached_movement; vec3 m_frame_cached_movement;


//---- //----
float m_walk_velocity_damping; float m_walk_velocity_damping;


//---- //----
vec3 m_gravity; vec3 m_gravity;


//---- //----
vec3 m_walk_velocity; vec3 m_walk_velocity;
vec3 m_current_velocity; vec3 m_current_velocity;


#else // NO PHYSIC IMPLEMENTATION #else // NO PHYSIC IMPLEMENTATION


virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false) { } virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false) { }
virtual void InitBodyToGhost() { } virtual void InitBodyToGhost() { }
virtual void AddToSimulation(class Simulation* current_simulation) { } virtual void AddToSimulation(class Simulation* current_simulation) { }
virtual void RemoveFromSimulation(class Simulation* current_simulation) { } virtual void RemoveFromSimulation(class Simulation* current_simulation) { }
virtual void SetMovementForFrame(vec3 const &MoveQuantity) { } virtual void SetMovementForFrame(vec3 const &MoveQuantity) { }


#endif // PHYSIC IMPLEMENTATION #endif // PHYSIC IMPLEMENTATION




+ 153
- 153
test/Physics/Include/EasyConstraint.h Näytä tiedosto

@@ -31,180 +31,180 @@ namespace phys
class EasyConstraint class EasyConstraint
{ {


friend class Simulation; friend class Simulation;
friend class EasyPhysic; friend class EasyPhysic;


#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET


public: public:
EasyConstraint() : EasyConstraint() :
m_typed_constraint(NULL), m_typed_constraint(NULL),
m_p2p_constraint(NULL), m_p2p_constraint(NULL),
m_hinge_constraint(NULL), m_hinge_constraint(NULL),
m_slider_constraint(NULL), m_slider_constraint(NULL),
m_cone_twist_constraint(NULL), m_cone_twist_constraint(NULL),
m_6dof_constraint(NULL), m_6dof_constraint(NULL),
m_owner_simulation(NULL), m_owner_simulation(NULL),
m_a_physobj(NULL), m_a_physobj(NULL),
m_b_physobj(NULL), m_b_physobj(NULL),
m_a_transform(lol::mat4(1.f)), m_a_transform(lol::mat4(1.f)),
m_b_transform(lol::mat4(1.f)), m_b_transform(lol::mat4(1.f)),
m_using_ref_a(false), m_using_ref_a(false),
m_disable_a2b_collision(false) m_disable_a2b_collision(false)
{
{ }
} ~EasyConstraint()
~EasyConstraint() {
{ delete m_typed_constraint;
delete m_typed_constraint; m_p2p_constraint = NULL;
m_p2p_constraint = NULL; m_hinge_constraint = NULL;
m_hinge_constraint = NULL; m_slider_constraint = NULL;
m_slider_constraint = NULL; m_cone_twist_constraint = NULL;
m_cone_twist_constraint = NULL; m_6dof_constraint = NULL;
m_6dof_constraint = NULL; }
} void AddToSimulation(class Simulation* current_simulation);
void RemoveFromSimulation(class Simulation* current_simulation);
void AddToSimulation(class Simulation* current_simulation);
void RemoveFromSimulation(class Simulation* current_simulation);


private: private:


//check if Init can be done //check if Init can be done
bool CanProceedWithInit() bool CanProceedWithInit()
{ {
if (!m_a_physobj || !m_b_physobj) if (!m_a_physobj || !m_b_physobj)
return false; return false;
if (!m_a_physobj->m_rigid_body || !m_b_physobj->m_rigid_body)
if (!m_a_physobj->m_rigid_body || !m_b_physobj->m_rigid_body) return false;
return false; return true;
}
return true; //-------------------------------------------------------------------------
} //Init constraint functions
//--
//------------------------------------------------------------------------- void CustomInitConstraintToPoint2Point()
//Init constraint functions {
//-- m_p2p_constraint = new btPoint2PointConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body,
void CustomInitConstraintToPoint2Point() LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT));
{ m_typed_constraint = m_p2p_constraint;
m_p2p_constraint = new btPoint2PointConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body, }
LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)); void CustomInitConstraintToHinge()
m_typed_constraint = m_p2p_constraint; {
} m_hinge_constraint = new btHingeConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body,
btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)),
void CustomInitConstraintToHinge() btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)),
{ m_using_ref_a);
m_hinge_constraint = new btHingeConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body, m_typed_constraint = m_hinge_constraint;
btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)), }
btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)), void CustomInitConstraintToSlider()
m_using_ref_a); {
m_typed_constraint = m_hinge_constraint; m_slider_constraint = new btSliderConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body,
} btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)),
btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)),
void CustomInitConstraintToSlider() m_using_ref_a);
{ m_typed_constraint = m_slider_constraint;
m_slider_constraint = new btSliderConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body, }
btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)), void CustomInitConstraintToConeTwist()
btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)), {
m_using_ref_a); m_cone_twist_constraint = new btConeTwistConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body,
m_typed_constraint = m_slider_constraint; btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)),
} btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)));
m_typed_constraint = m_cone_twist_constraint;
void CustomInitConstraintToConeTwist() }
{ void CustomInitConstraintTo6Dof()
m_cone_twist_constraint = new btConeTwistConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body, {
btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)), m_6dof_constraint = new btGeneric6DofConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body,
btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT))); btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)),
m_typed_constraint = m_cone_twist_constraint; btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)),
} m_using_ref_a);
m_typed_constraint = m_6dof_constraint;
void CustomInitConstraintTo6Dof() }
{ btTypedConstraint* m_typed_constraint;
m_6dof_constraint = new btGeneric6DofConstraint(*m_a_physobj->m_rigid_body, *m_b_physobj->m_rigid_body, btPoint2PointConstraint* m_p2p_constraint;
btTransform(LOL2BT_QUAT(quat(m_a_transform)), LOL2BT_VEC3(m_a_transform.v3.xyz * LOL2BT_UNIT)), btHingeConstraint* m_hinge_constraint;
btTransform(LOL2BT_QUAT(quat(m_b_transform)), LOL2BT_VEC3(m_b_transform.v3.xyz * LOL2BT_UNIT)), btSliderConstraint* m_slider_constraint;
m_using_ref_a); btConeTwistConstraint* m_cone_twist_constraint;
m_typed_constraint = m_6dof_constraint; btGeneric6DofConstraint* m_6dof_constraint;
}
btTypedConstraint* m_typed_constraint;
btPoint2PointConstraint* m_p2p_constraint;
btHingeConstraint* m_hinge_constraint;
btSliderConstraint* m_slider_constraint;
btConeTwistConstraint* m_cone_twist_constraint;
btGeneric6DofConstraint* m_6dof_constraint;


#else // NO PHYSIC IMPLEMENTATION #else // NO PHYSIC IMPLEMENTATION


public: public:
EasyConstraint() : EasyConstraint() :
m_a_physobj(NULL), m_a_physobj(NULL),
m_b_physobj(NULL), m_b_physobj(NULL),
m_a_transform(lol::mat4(1.f)), m_a_transform(lol::mat4(1.f)),
m_b_transform(lol::mat4(1.f)), m_b_transform(lol::mat4(1.f)),
m_using_ref_a(false), m_using_ref_a(false),
m_disable_a2b_collision(false) m_disable_a2b_collision(false)
{ {
} }


private: private:


void AddToSimulation(class Simulation* current_simulation) { } void AddToSimulation(class Simulation* current_simulation) { }
void RemoveFromSimulation(class Simulation* current_simulation) { } void RemoveFromSimulation(class Simulation* current_simulation) { }


//check if Init can be done //check if Init can be done
bool CanProceedWithInit() { return false; } bool CanProceedWithInit() { return false; }
void CustomInitConstraintToPoint2Point() { } void CustomInitConstraintToPoint2Point() { }
void CustomInitConstraintToHinge() { } void CustomInitConstraintToHinge() { }
void CustomInitConstraintToSlider() { } void CustomInitConstraintToSlider() { }
void CustomInitConstraintToConeTwist() { } void CustomInitConstraintToConeTwist() { }
void CustomInitConstraintTo6Dof() { } void CustomInitConstraintTo6Dof() { }


#endif // PHYSIC IMPLEMENTATION #endif // PHYSIC IMPLEMENTATION


public: public:
void InitConstraintToPoint2Point() { if (CanProceedWithInit()) CustomInitConstraintToPoint2Point(); } void InitConstraintToPoint2Point() { if (CanProceedWithInit()) CustomInitConstraintToPoint2Point(); }
void InitConstraintToHinge() { if (CanProceedWithInit()) CustomInitConstraintToHinge(); } void InitConstraintToHinge() { if (CanProceedWithInit()) CustomInitConstraintToHinge(); }
void InitConstraintToSlider() { if (CanProceedWithInit()) CustomInitConstraintToSlider(); } void InitConstraintToSlider() { if (CanProceedWithInit()) CustomInitConstraintToSlider(); }
void InitConstraintToConeTwist() { if (CanProceedWithInit()) CustomInitConstraintToConeTwist(); } void InitConstraintToConeTwist() { if (CanProceedWithInit()) CustomInitConstraintToConeTwist(); }
void InitConstraintTo6Dof() { if (CanProceedWithInit()) CustomInitConstraintTo6Dof(); } void InitConstraintTo6Dof() { if (CanProceedWithInit()) CustomInitConstraintTo6Dof(); }
//Set given physic object to the proper slot.
//Set given physic object to the proper slot. void SetPhysObjA(EasyPhysic* NewPhysObj, lol::mat4 NewTransform) { SetPhysObj(false, NewPhysObj, NewTransform); }
void SetPhysObjA(EasyPhysic* NewPhysObj, lol::mat4 NewTransform) { SetPhysObj(false, NewPhysObj, NewTransform); } void SetPhysObjB(EasyPhysic* NewPhysObj, lol::mat4 NewTransform) { SetPhysObj(true, NewPhysObj, NewTransform); }
void SetPhysObjB(EasyPhysic* NewPhysObj, lol::mat4 NewTransform) { SetPhysObj(true, NewPhysObj, NewTransform); } void SetPhysObj(bool SetToB, EasyPhysic* NewPhysObj, lol::mat4 NewTransform)
void SetPhysObj(bool SetToB, EasyPhysic* NewPhysObj, lol::mat4 NewTransform) {
{ if (SetToB)
if (SetToB) {
{ m_b_physobj = NewPhysObj;
m_b_physobj = NewPhysObj; m_b_transform = NewTransform;
m_b_transform = NewTransform; }
} else
else {
{ m_a_physobj = NewPhysObj;
m_a_physobj = NewPhysObj; m_a_transform = NewTransform;
m_a_transform = NewTransform; }
} }
} //Set whether or not the physic engine should use the A object as the reference (most constraint transform are local).
void SetRefAsA(bool NewUseRefA)
//Set whether or not the physic engine should use the A object as the reference (most constraint transform are local). {
void SetRefAsA(bool NewUseRefA) m_using_ref_a = NewUseRefA;
{ }
m_using_ref_a = NewUseRefA; //Set whether or not to disable the collision between the bodies
} void DisableCollisionBetweenObjs(bool DisableCollision)
{
//Set whether or not to disable the collision between the bodies m_disable_a2b_collision = DisableCollision;
void DisableCollisionBetweenObjs(bool DisableCollision) }
{
m_disable_a2b_collision = DisableCollision;
}


private: private:
Simulation* m_owner_simulation; Simulation* m_owner_simulation;
EasyPhysic* m_a_physobj; EasyPhysic* m_a_physobj;
EasyPhysic* m_b_physobj; EasyPhysic* m_b_physobj;
lol::mat4 m_a_transform; lol::mat4 m_a_transform;
lol::mat4 m_b_transform; lol::mat4 m_b_transform;
bool m_using_ref_a; bool m_using_ref_a;
bool m_disable_a2b_collision; bool m_disable_a2b_collision;


}; };




+ 102
- 102
test/Physics/Include/EasyPhysics.h Näytä tiedosto

@@ -33,136 +33,136 @@ namespace phys
class EasyPhysic class EasyPhysic
{ {


friend class Simulation; friend class Simulation;
friend class EasyConstraint; friend class EasyConstraint;


#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET


public: public:
EasyPhysic(WorldEntity* NewOwnerEntity); EasyPhysic(WorldEntity* NewOwnerEntity);
~EasyPhysic(); ~EasyPhysic();
virtual void SetShapeToBox(lol::vec3& box_size);
virtual void SetShapeToBox(lol::vec3& box_size); virtual void SetShapeToSphere(float radius);
virtual void SetShapeToSphere(float radius); virtual void SetShapeToCone(float radius, float height);
virtual void SetShapeToCone(float radius, float height); virtual void SetShapeToCylinder(lol::vec3& cyl_size);
virtual void SetShapeToCylinder(lol::vec3& cyl_size); virtual void SetShapeToCapsule(float radius, float height);
virtual void SetShapeToCapsule(float radius, float height); virtual bool CanChangeCollisionChannel() { return (m_rigid_body == NULL); }
virtual mat4 GetTransform();
virtual bool CanChangeCollisionChannel() { return (m_rigid_body == NULL); } virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f)));
virtual mat4 GetTransform();
virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f)));
protected: protected:
virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix); virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix);
public: public:
virtual void SetMass(float mass); virtual void SetMass(float mass);
virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false); virtual void InitBodyToRigid(bool ZeroMassIsKinematic=false);
virtual void InitBodyToGhost(); virtual void InitBodyToGhost();
virtual void AddToSimulation(class Simulation* current_simulation); virtual void AddToSimulation(class Simulation* current_simulation);
virtual void RemoveFromSimulation(class Simulation* current_simulation); virtual void RemoveFromSimulation(class Simulation* current_simulation);


protected: protected:
virtual void SetLocalInertia(float mass); virtual void SetLocalInertia(float mass);
virtual void SetShapeTo(btCollisionShape* collision_shape); virtual void SetShapeTo(btCollisionShape* collision_shape);


virtual btGhostObject* GetGhostObjectInstance(); virtual btGhostObject* GetGhostObjectInstance();


btCollisionObject* m_collision_object; btCollisionObject* m_collision_object;


btGhostObject* m_ghost_object; btGhostObject* m_ghost_object;


btRigidBody* m_rigid_body; btRigidBody* m_rigid_body;
btVector3 m_local_inertia; btVector3 m_local_inertia;


btCollisionShape* m_collision_shape; btCollisionShape* m_collision_shape;
btConvexShape* m_convex_shape; btConvexShape* m_convex_shape;
btMotionState* m_motion_state; btMotionState* m_motion_state;


#else // NO PHYSIC IMPLEMENTATION #else // NO PHYSIC IMPLEMENTATION


public: public:
EasyPhysic(WorldEntity* NewOwnerEntity) { m_owner_entity = NewOwnerEntity; } EasyPhysic(WorldEntity* NewOwnerEntity) { m_owner_entity = NewOwnerEntity; }


virtual void SetShapeToBox(lol::vec3& BoxSize) { } virtual void SetShapeToBox(lol::vec3& BoxSize) { }
virtual void SetShapeToSphere(float radius) { } virtual void SetShapeToSphere(float radius) { }
virtual void SetShapeToCone(float radius, float height) { } virtual void SetShapeToCone(float radius, float height) { }
virtual void SetShapeToCylinder(lol::vec3& cyl_size) { } virtual void SetShapeToCylinder(lol::vec3& cyl_size) { }
virtual void SetShapeToCapsule(float radius, float height) { } virtual void SetShapeToCapsule(float radius, float height) { }


virtual bool CanChangeCollisionChannel() { return true; } virtual bool CanChangeCollisionChannel() { return true; }
virtual mat4 GetTransform() { return mat4(1.0f); } virtual mat4 GetTransform() { return mat4(1.0f); }
virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f))) { } virtual void SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation=lol::quat(lol::mat4(1.0f))) { }
private: private:
virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix) { } virtual void BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix) { }
public: public:
virtual void SetMass(float mass) { } virtual void SetMass(float mass) { }
virtual void InitBodyToRigid() { } virtual void InitBodyToRigid() { }
virtual void InitBodyToGhost() { } virtual void InitBodyToGhost() { }
virtual void AddToSimulation(class Simulation* current_simulation) { } virtual void AddToSimulation(class Simulation* current_simulation) { }
virtual void RemoveFromSimulation(class Simulation* current_simulation) { } virtual void RemoveFromSimulation(class Simulation* current_simulation) { }


virtual void InitBodyToGhost() { } virtual void InitBodyToGhost() { }


#endif // PHYSIC IMPLEMENTATION #endif // PHYSIC IMPLEMENTATION


public: public:
//Sets the collision Group & Mask. //Sets the collision Group & Mask.
//Mask can change at runtime, not group ! //Mask can change at runtime, not group !
virtual bool SetCollisionChannel(int NewGroup, int NewMask) virtual bool SetCollisionChannel(int NewGroup, int NewMask)
{ {
if (CanChangeCollisionChannel()) if (CanChangeCollisionChannel())
{ {
m_collision_group = (1<<NewGroup); m_collision_group = (1<<NewGroup);
m_collision_mask = NewMask; m_collision_mask = NewMask;
return true; return true;
} }
return false; return false;
} }
int GetCollisionGroup() { return m_collision_group; } int GetCollisionGroup() { return m_collision_group; }
int GetCollisionMask() { return m_collision_mask; } int GetCollisionMask() { return m_collision_mask; }
//Base/Attachment logic
//Base/Attachment logic virtual void AttachTo(EasyPhysic* NewBase, bool NewBaseLockLocation = true, bool NewBaseLockRotation = true)
virtual void AttachTo(EasyPhysic* NewBase, bool NewBaseLockLocation = true, bool NewBaseLockRotation = true) {
{ if (NewBase == this || (NewBase && NewBase->m_base_physic == this))
if (NewBase == this || (NewBase && NewBase->m_base_physic == this)) return;
return; if (NewBase)
{
if (NewBase) bool bAlreadyExists = false;
{ for (int i = 0; i < NewBase->m_based_physic_list.Count(); ++i)
bool bAlreadyExists = false; if (NewBase->m_based_physic_list[i] == this)
for (int i = 0; i < NewBase->m_based_physic_list.Count(); ++i) bAlreadyExists = true;
if (NewBase->m_based_physic_list[i] == this) if (!bAlreadyExists)
bAlreadyExists = true; NewBase->m_based_physic_list << this;
if (!bAlreadyExists) m_base_physic = NewBase;
NewBase->m_based_physic_list << this; m_base_lock_location = NewBaseLockLocation;
m_base_physic = NewBase; m_base_lock_rotation = NewBaseLockRotation;
m_base_lock_location = NewBaseLockLocation; }
m_base_lock_rotation = NewBaseLockRotation; else if (m_base_physic)
} {
else if (m_base_physic) for (int i = 0; i < m_base_physic->m_based_physic_list.Count(); ++i)
{ if (m_base_physic->m_based_physic_list[i] == this)
for (int i = 0; i < m_base_physic->m_based_physic_list.Count(); ++i) m_base_physic->m_based_physic_list.Remove(i--);
if (m_base_physic->m_based_physic_list[i] == this) m_base_physic = NULL;
m_base_physic->m_based_physic_list.Remove(i--); }
m_base_physic = NULL; }
}
}


protected: protected:
lol::mat4 m_local_to_world; lol::mat4 m_local_to_world;
float m_mass; float m_mass;
int m_collision_group; int m_collision_group;
int m_collision_mask; int m_collision_mask;
WorldEntity* m_owner_entity; WorldEntity* m_owner_entity;
Simulation* m_owner_simulation; Simulation* m_owner_simulation;
//Base/Attachment logic
//Base/Attachment logic Array<EasyPhysic*> m_based_physic_list; //List of objects based on this : this object moves, its based object MoveStep with it.
Array<EasyPhysic*> m_based_physic_list; //List of objects based on this : this object moves, its based object MoveStep with it. EasyPhysic* m_base_physic; //Base for this object : The base moves, the object moves with it.
EasyPhysic* m_base_physic; //Base for this object : The base moves, the object moves with it. bool m_base_lock_location; //when this is TRUE, location moves with rotation change.
bool m_base_lock_location; //when this is TRUE, location moves with rotation change. bool m_base_lock_rotation; //when this is TRUE, rotation moves with rotation change.
bool m_base_lock_rotation; //when this is TRUE, rotation moves with rotation change. //Touch logic
Array<EasyPhysic*> m_touching_physic; //Maintained by ghost objects
//Touch logic
Array<EasyPhysic*> m_touching_physic; //Maintained by ghost objects
}; };


} /* namespace phys */ } /* namespace phys */


+ 18
- 18
test/Physics/Include/LolBtPhysicsIntegration.h Näytä tiedosto

@@ -19,32 +19,32 @@


namespace lol namespace lol
{ {
//Override Gamegroups names for Physic-useage //Override Gamegroups names for Physic-useage
//"_ENT_" means that this is a group for Entities that use EasyPhysic primitives. //"_ENT_" means that this is a group for Entities that use EasyPhysic primitives.
//"_EZP_" means that this is a group for EasyPhysic primitives. //"_EZP_" means that this is a group for EasyPhysic primitives.
#define GAMEGROUP_ENT_INPUT GAMEGROUP_BEFORE #define GAMEGROUP_ENT_INPUT GAMEGROUP_BEFORE
#define GAMEGROUP_ENT_PLATFORM GAMEGROUP_DEFAULT #define GAMEGROUP_ENT_PLATFORM GAMEGROUP_DEFAULT
#define GAMEGROUP_ENT_MAIN GAMEGROUP_AFTER #define GAMEGROUP_ENT_MAIN GAMEGROUP_AFTER
#define GAMEGROUP_EZP_CHAR_CTRLR GAMEGROUP_AFTER_0 #define GAMEGROUP_EZP_CHAR_CTRLR GAMEGROUP_AFTER_0
#define GAMEGROUP_SIMULATION GAMEGROUP_AFTER_1 #define GAMEGROUP_SIMULATION GAMEGROUP_AFTER_1


#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET


#define LOL2BT_UNIT 1.0f #define LOL2BT_UNIT 1.0f
#define BT2LOL_UNIT 1.0f #define BT2LOL_UNIT 1.0f


#define LOL2BT_SIZE 0.5f #define LOL2BT_SIZE 0.5f
#define BT2LOL_SIZE 2.0f #define BT2LOL_SIZE 2.0f


#define LOL2BT_VEC3(ELEMENT) btVector3((ELEMENT).x, (ELEMENT).y, (ELEMENT).z) #define LOL2BT_VEC3(ELEMENT) btVector3((ELEMENT).x, (ELEMENT).y, (ELEMENT).z)
#define BT2LOL_VEC3(ELEMENT) (*(lol::vec3*)(&(ELEMENT))) #define BT2LOL_VEC3(ELEMENT) (*(lol::vec3*)(&(ELEMENT)))


//Same as above with Unit taken into account //Same as above with Unit taken into account
#define LOL2BTU_VEC3(ELEMENT) btVector3((ELEMENT).x * LOL2BT_UNIT, (ELEMENT).y * LOL2BT_UNIT, (ELEMENT).z * LOL2BT_UNIT) #define LOL2BTU_VEC3(ELEMENT) btVector3((ELEMENT).x * LOL2BT_UNIT, (ELEMENT).y * LOL2BT_UNIT, (ELEMENT).z * LOL2BT_UNIT)
#define BT2LOLU_VEC3(ELEMENT) (*(lol::vec3*)(&(ELEMENT))) * BT2LOL_UNIT #define BT2LOLU_VEC3(ELEMENT) (*(lol::vec3*)(&(ELEMENT))) * BT2LOL_UNIT


#define LOL2BT_QUAT(ELEMENT) btQuaternion((ELEMENT).x, (ELEMENT).y, (ELEMENT).z, (ELEMENT).w) #define LOL2BT_QUAT(ELEMENT) btQuaternion((ELEMENT).x, (ELEMENT).y, (ELEMENT).z, (ELEMENT).w)
#define BT2LOL_QUAT(ELEMENT) lol::quat((ELEMENT).getW(), BT2LOL_VEC3((ELEMENT).getAxis()) #define BT2LOL_QUAT(ELEMENT) lol::quat((ELEMENT).getW(), BT2LOL_VEC3((ELEMENT).getAxis())


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET




+ 373
- 373
test/Physics/Include/LolPhysics.h Näytä tiedosto

@@ -26,406 +26,406 @@ namespace phys


enum eRaycastType enum eRaycastType
{ {
ERT_Closest, ERT_Closest,
ERT_AllHit, ERT_AllHit,
ERT_AnyHit, //Will stop at the first hit. Hit data are supposed to be irrelevant ERT_AnyHit, //Will stop at the first hit. Hit data are supposed to be irrelevant


ERT_MAX ERT_MAX
}; };


struct RayCastResult struct RayCastResult
{ {
RayCastResult(int CollisionFilterGroup=1, int CollisionFilterMask=(0xFF)) RayCastResult(int CollisionFilterGroup=1, int CollisionFilterMask=(0xFF))
{ {
memset(this, 0, sizeof(RayCastResult)); memset(this, 0, sizeof(RayCastResult));
m_collision_filter_group = CollisionFilterGroup;
m_collision_filter_group = CollisionFilterGroup; m_collision_filter_mask = CollisionFilterMask;
m_collision_filter_mask = CollisionFilterMask; }
} void Reset()
void Reset() {
{ m_collider_list.Empty();
m_collider_list.Empty(); m_hit_normal_list.Empty();
m_hit_normal_list.Empty(); m_hit_point_list.Empty();
m_hit_point_list.Empty(); m_hit_fraction_list.Empty();
m_hit_fraction_list.Empty(); }
} Array<EasyPhysic*> m_collider_list;
Array<vec3> m_hit_normal_list;
Array<EasyPhysic*> m_collider_list; Array<vec3> m_hit_point_list;
Array<vec3> m_hit_normal_list; Array<float> m_hit_fraction_list;
Array<vec3> m_hit_point_list; short int m_collision_filter_group;
Array<float> m_hit_fraction_list; short int m_collision_filter_mask;
unsigned int m_flags; //???
short int m_collision_filter_group;
short int m_collision_filter_mask;
unsigned int m_flags; //???
}; };


class Simulation : public Entity class Simulation : public Entity
{ {
public: public:
Simulation() : Simulation() :
m_broadphase(0), m_broadphase(0),
m_collision_configuration(0), m_collision_configuration(0),
m_dispatcher(0), m_dispatcher(0),
m_solver(0), m_solver(0),
m_dynamics_world(0), m_dynamics_world(0),
m_timestep(1.f/60.f) m_timestep(1.f/60.f)
{ {
m_gamegroup = GAMEGROUP_SIMULATION; m_gamegroup = GAMEGROUP_SIMULATION;
} }
~Simulation() ~Simulation()
{ {
Exit(); Exit();
} }
char const *GetName() { return "<Simulation>"; }
char const *GetName() { return "<Simulation>"; }


#ifdef HAVE_PHYS_USE_BULLET #ifdef HAVE_PHYS_USE_BULLET
public: public:
void Init() void Init()
{ {
// Build the broadphase // Build the broadphase
if (1) if (1)
{ {
m_Sweep_broadphase = new btAxisSweep3(LOL2BT_VEC3(m_world_min), LOL2BT_VEC3(m_world_max)); m_Sweep_broadphase = new btAxisSweep3(LOL2BT_VEC3(m_world_min), LOL2BT_VEC3(m_world_max));
m_Sweep_broadphase->getOverlappingPairCache()->setInternalGhostPairCallback(new btGhostPairCallback()); m_Sweep_broadphase->getOverlappingPairCache()->setInternalGhostPairCallback(new btGhostPairCallback());
m_broadphase = m_Sweep_broadphase; m_broadphase = m_Sweep_broadphase;
} }
else else
m_broadphase = new btDbvtBroadphase(); m_broadphase = new btDbvtBroadphase();
// Set up the collision configuration and dispatcher
// Set up the collision configuration and dispatcher m_collision_configuration = new btDefaultCollisionConfiguration();
m_collision_configuration = new btDefaultCollisionConfiguration(); m_dispatcher = new btCollisionDispatcher(m_collision_configuration);
m_dispatcher = new btCollisionDispatcher(m_collision_configuration); // The actual physics solver
m_solver = new btSequentialImpulseConstraintSolver;
// The actual physics solver // The world.
m_solver = new btSequentialImpulseConstraintSolver; m_dynamics_world = new btDiscreteDynamicsWorld(m_dispatcher, m_broadphase, m_solver, m_collision_configuration);
}
// The world. virtual void TickGame(float seconds)
m_dynamics_world = new btDiscreteDynamicsWorld(m_dispatcher, m_broadphase, m_solver, m_collision_configuration); {
} Entity::TickGame(seconds);
//step the simulation
virtual void TickGame(float seconds) if (m_dynamics_world)
{ {
Entity::TickGame(seconds); //the "+1" is to have at least one Timestep and to ensure float to int .5f conversion.
int steps = (int)(seconds / m_timestep) + 1;
//step the simulation m_dynamics_world->stepSimulation(seconds, steps, m_timestep);
if (m_dynamics_world) }
{ }
//the "+1" is to have at least one Timestep and to ensure float to int .5f conversion. //Rip-Off of the btKinematicClosestNotMeRayResultCallback
int steps = (int)(seconds / m_timestep) + 1; class ClosestNotMeRayResultCallback : public btCollisionWorld::ClosestRayResultCallback
m_dynamics_world->stepSimulation(seconds, steps, m_timestep); {
} public:
} ClosestNotMeRayResultCallback(btCollisionObject* Me, const btVector3& rayFromWorld, const btVector3& rayToWorld) :
btCollisionWorld::ClosestRayResultCallback(rayFromWorld, rayToWorld)
//Rip-Off of the btKinematicClosestNotMeRayResultCallback {
class ClosestNotMeRayResultCallback : public btCollisionWorld::ClosestRayResultCallback m_me = Me;
{ }
public: virtual btScalar addSingleResult(btCollisionWorld::LocalRayResult& rayResult,bool normalInWorldSpace)
ClosestNotMeRayResultCallback(btCollisionObject* Me, const btVector3& rayFromWorld, const btVector3& rayToWorld) : {
btCollisionWorld::ClosestRayResultCallback(rayFromWorld, rayToWorld) if (rayResult.m_collisionObject == m_me)
{ return 1.0;
m_me = Me; return ClosestRayResultCallback::addSingleResult(rayResult, normalInWorldSpace);
} }
protected:
virtual btScalar addSingleResult(btCollisionWorld::LocalRayResult& rayResult,bool normalInWorldSpace) btCollisionObject* m_me;
{ };
if (rayResult.m_collisionObject == m_me) //Will stop at the first hit. Hit data are supposed to be irrelevant
return 1.0; class AnyHitRayResultCallback : public btCollisionWorld::ClosestRayResultCallback
{
return ClosestRayResultCallback::addSingleResult(rayResult, normalInWorldSpace); public:
} AnyHitRayResultCallback(const btVector3& rayFromWorld, const btVector3& rayToWorld) :
protected: btCollisionWorld::ClosestRayResultCallback(rayFromWorld, rayToWorld)
btCollisionObject* m_me; {
}; }
virtual btScalar addSingleResult(btCollisionWorld::LocalRayResult& rayResult,bool normalInWorldSpace)
//Will stop at the first hit. Hit data are supposed to be irrelevant {
class AnyHitRayResultCallback : public btCollisionWorld::ClosestRayResultCallback return .0f;
{ }
public: };
AnyHitRayResultCallback(const btVector3& rayFromWorld, const btVector3& rayToWorld) : //Returns true when hitting something. If SourceCaster is set, it will be ignored by Raycast.
btCollisionWorld::ClosestRayResultCallback(rayFromWorld, rayToWorld) bool RayHits(RayCastResult& HitResult, eRaycastType RaycastType, const vec3& RayFrom, const vec3& RayTo, EasyPhysic* SourceCaster=NULL)
{ {
} bool bResult = false;
btCollisionWorld::RayResultCallback* BtRayResult = NULL;
virtual btScalar addSingleResult(btCollisionWorld::LocalRayResult& rayResult,bool normalInWorldSpace) btCollisionWorld::ClosestRayResultCallback* BtRayResult_Closest;
{ btCollisionWorld::AllHitsRayResultCallback* BtRayResult_AllHits;
return .0f; switch (RaycastType)
} {
}; case ERT_Closest:
{
//Returns true when hitting something. If SourceCaster is set, it will be ignored by Raycast. if (SourceCaster)
bool RayHits(RayCastResult& HitResult, eRaycastType RaycastType, const vec3& RayFrom, const vec3& RayTo, EasyPhysic* SourceCaster=NULL) BtRayResult_Closest = new ClosestNotMeRayResultCallback(SourceCaster->m_collision_object, LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo));
{ else
bool bResult = false; BtRayResult_Closest = new btCollisionWorld::ClosestRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo));
BtRayResult = BtRayResult_Closest;
btCollisionWorld::RayResultCallback* BtRayResult = NULL; break;
btCollisionWorld::ClosestRayResultCallback* BtRayResult_Closest; }
btCollisionWorld::AllHitsRayResultCallback* BtRayResult_AllHits; case ERT_AllHit:
{
switch (RaycastType) BtRayResult_AllHits = new btCollisionWorld::AllHitsRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo));
{ BtRayResult = BtRayResult_AllHits;
case ERT_Closest: break;
{ }
if (SourceCaster) case ERT_AnyHit:
BtRayResult_Closest = new ClosestNotMeRayResultCallback(SourceCaster->m_collision_object, LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo)); {
else BtRayResult_Closest = new AnyHitRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo));
BtRayResult_Closest = new btCollisionWorld::ClosestRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo)); BtRayResult = BtRayResult_Closest;
BtRayResult = BtRayResult_Closest; break;
break; }
} }
case ERT_AllHit: m_dynamics_world->rayTest(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo), *BtRayResult);
{ if (BtRayResult->hasHit())
BtRayResult_AllHits = new btCollisionWorld::AllHitsRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo)); {
BtRayResult = BtRayResult_AllHits; bResult = true;
break; switch (RaycastType)
} {
case ERT_AnyHit: case ERT_Closest:
{ {
BtRayResult_Closest = new AnyHitRayResultCallback(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo)); HitResult.m_collider_list << (EasyPhysic*)BtRayResult_Closest->m_collisionObject->getUserPointer();
BtRayResult = BtRayResult_Closest; HitResult.m_hit_normal_list << BT2LOLU_VEC3(BtRayResult_Closest->m_hitNormalWorld);
break; HitResult.m_hit_point_list << BT2LOLU_VEC3(BtRayResult_Closest->m_hitPointWorld);
} HitResult.m_hit_fraction_list << BtRayResult_Closest->m_closestHitFraction;
} break;
}
m_dynamics_world->rayTest(LOL2BTU_VEC3(RayFrom), LOL2BTU_VEC3(RayTo), *BtRayResult); case ERT_AllHit:
if (BtRayResult->hasHit()) {
{ for (int i = 0; i < BtRayResult_AllHits->m_collisionObjects.size(); i++)
bResult = true; {
HitResult.m_collider_list << (EasyPhysic*)BtRayResult_AllHits->m_collisionObjects[i]->getUserPointer();
switch (RaycastType) HitResult.m_hit_normal_list << BT2LOLU_VEC3(BtRayResult_AllHits->m_hitNormalWorld[i]);
{ HitResult.m_hit_point_list << BT2LOLU_VEC3(BtRayResult_AllHits->m_hitPointWorld[i]);
case ERT_Closest: HitResult.m_hit_fraction_list << BtRayResult_AllHits->m_hitFractions[i];
{ }
HitResult.m_collider_list << (EasyPhysic*)BtRayResult_Closest->m_collisionObject->getUserPointer(); break;
HitResult.m_hit_normal_list << BT2LOLU_VEC3(BtRayResult_Closest->m_hitNormalWorld); }
HitResult.m_hit_point_list << BT2LOLU_VEC3(BtRayResult_Closest->m_hitPointWorld); }
HitResult.m_hit_fraction_list << BtRayResult_Closest->m_closestHitFraction; }
break; delete BtRayResult;
} return bResult;
case ERT_AllHit: }
{ void Exit()
for (int i = 0; i < BtRayResult_AllHits->m_collisionObjects.size(); i++) {
{ delete m_dynamics_world;
HitResult.m_collider_list << (EasyPhysic*)BtRayResult_AllHits->m_collisionObjects[i]->getUserPointer(); delete m_solver;
HitResult.m_hit_normal_list << BT2LOLU_VEC3(BtRayResult_AllHits->m_hitNormalWorld[i]); delete m_dispatcher;
HitResult.m_hit_point_list << BT2LOLU_VEC3(BtRayResult_AllHits->m_hitPointWorld[i]); delete m_collision_configuration;
HitResult.m_hit_fraction_list << BtRayResult_AllHits->m_hitFractions[i]; delete m_broadphase;
} }
break; btDiscreteDynamicsWorld* GetWorld()
} {
} return m_dynamics_world;
} }
delete BtRayResult;
return bResult;
}
void Exit()
{
delete m_dynamics_world;
delete m_solver;
delete m_dispatcher;
delete m_collision_configuration;
delete m_broadphase;
}
btDiscreteDynamicsWorld* GetWorld()
{
return m_dynamics_world;
}


private: private:
void CustomSetContinuousDetection(bool ShouldUseCCD) void CustomSetContinuousDetection(bool ShouldUseCCD)
{ {
if (m_dynamics_world) if (m_dynamics_world)
m_dynamics_world->getDispatchInfo().m_useContinuous = ShouldUseCCD; m_dynamics_world->getDispatchInfo().m_useContinuous = ShouldUseCCD;
} }
void CustomSetGravity(vec3 &NewGravity)
void CustomSetGravity(vec3 &NewGravity) {
{ if (m_dynamics_world)
if (m_dynamics_world) m_dynamics_world->setGravity(LOL2BT_VEC3(NewGravity * LOL2BT_UNIT));
m_dynamics_world->setGravity(LOL2BT_VEC3(NewGravity * LOL2BT_UNIT)); }
} void CustomSetWorldLimit(vec3 const &NewWorldMin, vec3 const &NewWorldMax)
{
void CustomSetWorldLimit(vec3 const &NewWorldMin, vec3 const &NewWorldMax) }
{ void CustomSetTimestep(float NewTimestep) { }
} //broadphase
btBroadphaseInterface* m_broadphase;
void CustomSetTimestep(float NewTimestep) { } btAxisSweep3* m_Sweep_broadphase;
// Set up the collision configuration and dispatc
//broadphase btDefaultCollisionConfiguration* m_collision_configuration;
btBroadphaseInterface* m_broadphase; btCollisionDispatcher* m_dispatcher;
btAxisSweep3* m_Sweep_broadphase; // The actual physics solver
// Set up the collision configuration and dispatc btSequentialImpulseConstraintSolver* m_solver;
btDefaultCollisionConfiguration* m_collision_configuration; // The world.
btCollisionDispatcher* m_dispatcher; btDiscreteDynamicsWorld* m_dynamics_world;
// The actual physics solver
btSequentialImpulseConstraintSolver* m_solver;
// The world.
btDiscreteDynamicsWorld* m_dynamics_world;


#else // NO PHYSIC IMPLEMENTATION #else // NO PHYSIC IMPLEMENTATION


public: public:
void Init() { } void Init() { }
void TickGame(float seconds) { } void TickGame(float seconds) { }
bool RayHits(RayCastResult& HitResult, eRaycastType RaycastType, const vec3& RayFrom, const vec3& RayTo, EasyPhysic* SourceCaster=NULL) { return false; } bool RayHits(RayCastResult& HitResult, eRaycastType RaycastType, const vec3& RayFrom, const vec3& RayTo, EasyPhysic* SourceCaster=NULL) { return false; }
void Exit() { } void Exit() { }
private: private:
void CustomSetContinuousDetection(bool ShouldUseCCD) { } void CustomSetContinuousDetection(bool ShouldUseCCD) { }
void CustomSetGravity(vec3 &NewGravity) { } void CustomSetGravity(vec3 &NewGravity) { }
void CustomSetWorldLimit(vec3 &NewWorldMin, vec3 &NewWorldMax) { } void CustomSetWorldLimit(vec3 &NewWorldMin, vec3 &NewWorldMax) { }
void CustomSetTimestep(float NewTimestep) { } void CustomSetTimestep(float NewTimestep) { }


#endif // PHYSIC IMPLEMENTATION #endif // PHYSIC IMPLEMENTATION


public: public:
//Main logic : //Main logic :
//The Set*() functions do the all-lib-independent data storage. //The Set*() functions do the all-lib-independent data storage.
//And then it calls the CustomSet*() which are the specialized versions. //And then it calls the CustomSet*() which are the specialized versions.
//Sets the continuous collision detection flag.
//Sets the continuous collision detection flag. void SetContinuousDetection(bool ShouldUseCCD)
void SetContinuousDetection(bool ShouldUseCCD) {
{ m_using_CCD = ShouldUseCCD;
m_using_CCD = ShouldUseCCD; CustomSetContinuousDetection(ShouldUseCCD);
CustomSetContinuousDetection(ShouldUseCCD); }
} //Sets the simulation gravity.
void SetGravity(vec3 &NewGravity)
//Sets the simulation gravity. {
void SetGravity(vec3 &NewGravity) m_gravity = NewGravity;
{ CustomSetGravity(NewGravity);
m_gravity = NewGravity; }
CustomSetGravity(NewGravity); //Sets the simulation gravity.
} void SetWorldLimit(vec3 const &NewWorldMin, vec3 const &NewWorldMax)
{
//Sets the simulation gravity. m_world_min = NewWorldMin;
void SetWorldLimit(vec3 const &NewWorldMin, vec3 const &NewWorldMax) m_world_max = NewWorldMax;
{ CustomSetWorldLimit(NewWorldMin, NewWorldMax);
m_world_min = NewWorldMin; }
m_world_max = NewWorldMax; //Sets the simulation fixed timestep.
CustomSetWorldLimit(NewWorldMin, NewWorldMax); void SetTimestep(float NewTimestep)
} {
if (NewTimestep > .0f)
//Sets the simulation fixed timestep. {
void SetTimestep(float NewTimestep) m_timestep = NewTimestep;
{ CustomSetTimestep(NewTimestep);
if (NewTimestep > .0f) }
{ }
m_timestep = NewTimestep;
CustomSetTimestep(NewTimestep);
}
}


private: private:


friend class EasyPhysic; friend class EasyPhysic;
friend class EasyCharacterController; friend class EasyCharacterController;
friend class EasyConstraint; friend class EasyConstraint;
enum eEasyPhysicType
enum eEasyPhysicType {
{ EEPT_Dynamic,
EEPT_Dynamic, EEPT_Static,
EEPT_Static, EEPT_Ghost,
EEPT_Ghost, EEPT_CollisionObject,
EEPT_CollisionObject, EEPT_CharacterController,
EEPT_CharacterController, EEPT_MAX
};
EEPT_MAX //m_owner_simulation
}; //Adds the given EasyPhysic to the correct list.
void ObjectRegistration(bool AddObject, EasyPhysic* NewEP, eEasyPhysicType CurType)
//m_owner_simulation {
//Adds the given EasyPhysic to the correct list. Array<EasyPhysic*>* SearchList = NULL;
void ObjectRegistration(bool AddObject, EasyPhysic* NewEP, eEasyPhysicType CurType) switch(CurType)
{ {
Array<EasyPhysic*>* SearchList = NULL; case EEPT_Dynamic:
switch(CurType) {
{ SearchList = &m_dynamic_list;
case EEPT_Dynamic: break;
{ }
SearchList = &m_dynamic_list; case EEPT_Static:
break; {
} SearchList = &m_static_list;
case EEPT_Static: break;
{ }
SearchList = &m_static_list; case EEPT_Ghost:
break; {
} SearchList = &m_ghost_list;
case EEPT_Ghost: break;
{ }
SearchList = &m_ghost_list; case EEPT_CollisionObject:
break; {
} SearchList = &m_collision_object_list;
case EEPT_CollisionObject: break;
{ }
SearchList = &m_collision_object_list; case EEPT_CharacterController:
break; {
} SearchList = &m_character_controller_list;
case EEPT_CharacterController: break;
{ }
SearchList = &m_character_controller_list; }
break; if (AddObject)
} {
} NewEP->m_owner_simulation = this;
(*SearchList) << NewEP;
if (AddObject) }
{ else
NewEP->m_owner_simulation = this; {
(*SearchList) << NewEP; NewEP->m_owner_simulation = NULL;
} for (int i = 0; i < SearchList->Count(); ++i)
else {
{ if ((*SearchList)[i] == NewEP)
NewEP->m_owner_simulation = NULL; {
for (int i = 0; i < SearchList->Count(); ++i) SearchList->Remove(i--);
{ break;
if ((*SearchList)[i] == NewEP) }
{ }
SearchList->Remove(i--); }
break; }
} void ObjectRegistration(bool AddObject, EasyConstraint* NewEC)
} {
} Array<EasyConstraint*>* SearchList = NULL;
} SearchList = &m_constraint_list;
void ObjectRegistration(bool AddObject, EasyConstraint* NewEC) if (AddObject)
{ {
Array<EasyConstraint*>* SearchList = NULL; NewEC->m_owner_simulation = this;
SearchList = &m_constraint_list; (*SearchList) << NewEC;
}
if (AddObject) else
{ {
NewEC->m_owner_simulation = this; NewEC->m_owner_simulation = NULL;
(*SearchList) << NewEC; for (int i = 0; i < SearchList->Count(); ++i)
} {
else if ((*SearchList)[i] == NewEC)
{ {
NewEC->m_owner_simulation = NULL; SearchList->Remove(i--);
for (int i = 0; i < SearchList->Count(); ++i) break;
{ }
if ((*SearchList)[i] == NewEC) }
{ }
SearchList->Remove(i--); }
break; //Easy Physics body List
} Array<EasyPhysic*> m_dynamic_list;
} Array<EasyPhysic*> m_static_list;
} Array<EasyPhysic*> m_ghost_list;
} Array<EasyPhysic*> m_collision_object_list;
Array<EasyPhysic*> m_character_controller_list;
//Easy Physics body List Array<EasyConstraint*> m_constraint_list;
Array<EasyPhysic*> m_dynamic_list; //Easy Physics data storage
Array<EasyPhysic*> m_static_list; float m_timestep;
Array<EasyPhysic*> m_ghost_list; bool m_using_CCD;
Array<EasyPhysic*> m_collision_object_list; vec3 m_gravity;
Array<EasyPhysic*> m_character_controller_list; vec3 m_world_min;
Array<EasyConstraint*> m_constraint_list; vec3 m_world_max;
//Easy Physics data storage
float m_timestep;
bool m_using_CCD;
vec3 m_gravity;
vec3 m_world_min;
vec3 m_world_max;
}; };


} /* namespace phys */ } /* namespace phys */


+ 161
- 161
test/Physics/Src/BulletCharacterController.cpp Näytä tiedosto

@@ -45,202 +45,202 @@ namespace phys
//When called, will try to remove Character controller from its collision. //When called, will try to remove Character controller from its collision.
bool BulletKinematicCharacterController::RecoverFromPenetration(btCollisionWorld* CollisionWorld) bool BulletKinematicCharacterController::RecoverFromPenetration(btCollisionWorld* CollisionWorld)
{ {
bool HasPenetration = false; bool HasPenetration = false;
//Retrieve all pair with us colliding.
//Retrieve all pair with us colliding. CollisionWorld->getDispatcher()->dispatchAllCollisionPairs(m_ghost_object->getOverlappingPairCache(), CollisionWorld->getDispatchInfo(), CollisionWorld->getDispatcher());
CollisionWorld->getDispatcher()->dispatchAllCollisionPairs(m_ghost_object->getOverlappingPairCache(), CollisionWorld->getDispatchInfo(), CollisionWorld->getDispatcher()); m_current_position = BT2LOLU_VEC3(m_ghost_object->getWorldTransform().getOrigin());
m_current_position = BT2LOLU_VEC3(m_ghost_object->getWorldTransform().getOrigin()); float MaxPen = .0f;
for (int i = 0; i < m_ghost_object->getOverlappingPairCache()->getNumOverlappingPairs(); i++)
float MaxPen = .0f; {
for (int i = 0; i < m_ghost_object->getOverlappingPairCache()->getNumOverlappingPairs(); i++) m_manifold_array.resize(0);
{ //this is the equivalent of the "Touch algorithm". Maybe refactor ?
m_manifold_array.resize(0); btBroadphasePair* CollisionPair = &m_ghost_object->getOverlappingPairCache()->getOverlappingPairArray()[i];
if (CollisionPair->m_algorithm)
//this is the equivalent of the "Touch algorithm". Maybe refactor ? CollisionPair->m_algorithm->getAllContactManifolds(m_manifold_array);
btBroadphasePair* CollisionPair = &m_ghost_object->getOverlappingPairCache()->getOverlappingPairArray()[i]; for (int j = 0; j < m_manifold_array.size(); ++j)
if (CollisionPair->m_algorithm) {
CollisionPair->m_algorithm->getAllContactManifolds(m_manifold_array); btPersistentManifold* CurMfold = m_manifold_array[j];
//Normal direction differs if we're Body0
for (int j = 0; j < m_manifold_array.size(); ++j) float DirSign = CurMfold->getBody0() == m_ghost_object ? -1.f : 1.f;
{ for (int k = 0; k < CurMfold->getNumContacts(); k++)
btPersistentManifold* CurMfold = m_manifold_array[j]; {
//Normal direction differs if we're Body0 const btManifoldPoint& MfPoint = CurMfold->getContactPoint(k);
float DirSign = CurMfold->getBody0() == m_ghost_object ? -1.f : 1.f; float Dist = MfPoint.getDistance();
if (Dist < .0f)
for (int k = 0; k < CurMfold->getNumContacts(); k++) {
{ if (Dist < MaxPen)
const btManifoldPoint& MfPoint = CurMfold->getContactPoint(k); {
float Dist = MfPoint.getDistance(); MaxPen = Dist;
if (Dist < .0f) m_touching_normal = BT2LOL_VEC3(MfPoint.m_normalWorldOnB) * DirSign;
{ }
if (Dist < MaxPen) m_current_position += BT2LOL_VEC3(MfPoint.m_normalWorldOnB) * DirSign * Dist * .2f;
{ HasPenetration = true;
MaxPen = Dist; }
m_touching_normal = BT2LOL_VEC3(MfPoint.m_normalWorldOnB) * DirSign; }
} }
m_current_position += BT2LOL_VEC3(MfPoint.m_normalWorldOnB) * DirSign * Dist * .2f; }
HasPenetration = true; btTransform GObjMx = m_ghost_object->getWorldTransform();
} GObjMx.setOrigin(LOL2BTU_VEC3(m_current_position));
} m_ghost_object->setWorldTransform(GObjMx);
} return HasPenetration;
}
btTransform GObjMx = m_ghost_object->getWorldTransform();
GObjMx.setOrigin(LOL2BTU_VEC3(m_current_position));
m_ghost_object->setWorldTransform(GObjMx);
return HasPenetration;
} }


//When the Controller hits a wall, we modify the target so the controller will MoveStep along the wall. //When the Controller hits a wall, we modify the target so the controller will MoveStep along the wall.
void BulletKinematicCharacterController::UpdateTargetOnHit(const vec3& HitNormal, float TangentMag, float NormalMag) void BulletKinematicCharacterController::UpdateTargetOnHit(const vec3& HitNormal, float TangentMag, float NormalMag)
{ {
vec3 Movedir = m_target_position - m_current_position; vec3 Movedir = m_target_position - m_current_position;
float MoveLength = (float)length(Movedir); float MoveLength = (float)length(Movedir);


if (MoveLength > SIMD_EPSILON) if (MoveLength > SIMD_EPSILON)
{ {
Movedir = normalize(Movedir); Movedir = normalize(Movedir);


vec3 ReflectDir = normalize(GetReflectedDir(Movedir, HitNormal)); vec3 ReflectDir = normalize(GetReflectedDir(Movedir, HitNormal));
vec3 ParallelDir = ProjectDirOnNorm(ReflectDir, HitNormal); vec3 ParallelDir = ProjectDirOnNorm(ReflectDir, HitNormal);
vec3 PerpindicularDir = ProjectDirOnNormPerpindicular(ReflectDir, HitNormal); vec3 PerpindicularDir = ProjectDirOnNormPerpindicular(ReflectDir, HitNormal);


m_target_position = m_current_position; m_target_position = m_current_position;


if (NormalMag != .0f) if (NormalMag != .0f)
m_target_position += PerpindicularDir * NormalMag * MoveLength; m_target_position += PerpindicularDir * NormalMag * MoveLength;
} }
} }


//Handles the actual Movement. It actually moves in the 3 dimensions, function name is confusing. //Handles the actual Movement. It actually moves in the 3 dimensions, function name is confusing.
void BulletKinematicCharacterController::DoMove(btCollisionWorld* CollisionWorld, const vec3& MoveStep, float DeltaTime) void BulletKinematicCharacterController::DoMove(btCollisionWorld* CollisionWorld, const vec3& MoveStep, float DeltaTime)
{ {
// phase 2: forward and strafe // phase 2: forward and strafe
m_target_position = m_current_position + MoveStep; m_target_position = m_current_position + MoveStep;
btTransform SweepStart, SweepEnd; btTransform SweepStart, SweepEnd;
SweepStart.setIdentity(); SweepStart.setIdentity();
SweepEnd.setIdentity(); SweepEnd.setIdentity();
float Fraction = 1.f;
float Fraction = 1.f; float SqDist = .0f;
float SqDist = .0f; if (m_touching_contact && dot(m_normalized_direction, m_touching_normal) > .0f)
UpdateTargetOnHit(m_touching_normal);
if (m_touching_contact && dot(m_normalized_direction, m_touching_normal) > .0f) //Let's loop on movement, until Movement fraction if below 0.01, which means we've reached our destination.
UpdateTargetOnHit(m_touching_normal); //Or until we'tried 10 times.
int MaxMoveLoop = 10;
//Let's loop on movement, until Movement fraction if below 0.01, which means we've reached our destination. while (Fraction > .01f && MaxMoveLoop-- > 0)
//Or until we'tried 10 times. {
int MaxMoveLoop = 10; SweepStart.setOrigin(LOL2BTU_VEC3(m_current_position));
while (Fraction > .01f && MaxMoveLoop-- > 0) SweepEnd.setOrigin(LOL2BTU_VEC3(m_target_position));
{ vec3 SweepDirNeg(m_current_position - m_target_position);
SweepStart.setOrigin(LOL2BTU_VEC3(m_current_position)); ClosestNotMeConvexResultCallback SweepCallback(m_ghost_object, SweepDirNeg, .0f);
SweepEnd.setOrigin(LOL2BTU_VEC3(m_target_position)); SweepCallback.m_collisionFilterGroup = GetGhostObject()->getBroadphaseHandle()->m_collisionFilterGroup;
vec3 SweepDirNeg(m_current_position - m_target_position); SweepCallback.m_collisionFilterMask = GetGhostObject()->getBroadphaseHandle()->m_collisionFilterMask;
//The sweep test is done with an added margin, so we use it and then discard it
ClosestNotMeConvexResultCallback SweepCallback(m_ghost_object, SweepDirNeg, .0f); float SavedMargin = m_convex_shape->getMargin();
SweepCallback.m_collisionFilterGroup = GetGhostObject()->getBroadphaseHandle()->m_collisionFilterGroup; m_convex_shape->setMargin(SavedMargin + m_added_margin); //Apply Added Margin
SweepCallback.m_collisionFilterMask = GetGhostObject()->getBroadphaseHandle()->m_collisionFilterMask; if (m_do_gobject_sweep_test)
m_ghost_object->convexSweepTest (m_convex_shape, SweepStart, SweepEnd, SweepCallback, CollisionWorld->getDispatchInfo().m_allowedCcdPenetration);
//The sweep test is done with an added margin, so we use it and then discard it else
float SavedMargin = m_convex_shape->getMargin(); CollisionWorld->convexSweepTest (m_convex_shape, SweepStart, SweepEnd, SweepCallback, CollisionWorld->getDispatchInfo().m_allowedCcdPenetration);
m_convex_shape->setMargin(SavedMargin + m_added_margin); //Apply Added Margin m_convex_shape->setMargin(SavedMargin); //Restore saved margin
if (m_do_gobject_sweep_test) Fraction -= SweepCallback.m_closestHitFraction;
m_ghost_object->convexSweepTest (m_convex_shape, SweepStart, SweepEnd, SweepCallback, CollisionWorld->getDispatchInfo().m_allowedCcdPenetration); if (SweepCallback.hasHit())
else {
CollisionWorld->convexSweepTest (m_convex_shape, SweepStart, SweepEnd, SweepCallback, CollisionWorld->getDispatchInfo().m_allowedCcdPenetration); //We moved only a Fraction
m_convex_shape->setMargin(SavedMargin); //Restore saved margin float HitDist = (float)length(BT2LOLU_VEC3(SweepCallback.m_hitPointWorld) - m_current_position);
UpdateTargetOnHit(BT2LOL_VEC3(SweepCallback.m_hitNormalWorld));
Fraction -= SweepCallback.m_closestHitFraction; vec3 NewDir = m_target_position - m_current_position;
SqDist = sqlength(NewDir);
if (SweepCallback.hasHit()) if (SqDist > SIMD_EPSILON)
{ {
//We moved only a Fraction NewDir = normalize(NewDir);
float HitDist = (float)length(BT2LOLU_VEC3(SweepCallback.m_hitPointWorld) - m_current_position); //See Quake2: "If velocity is against original velocity, stop ead to avoid tiny oscilations in sloping corners."
if (dot(NewDir, m_normalized_direction) <= .0f)
UpdateTargetOnHit(BT2LOL_VEC3(SweepCallback.m_hitNormalWorld)); break;
vec3 NewDir = m_target_position - m_current_position; }
SqDist = sqlength(NewDir); else
if (SqDist > SIMD_EPSILON) break;
{ }
NewDir = normalize(NewDir); else //We moved whole way
//See Quake2: "If velocity is against original velocity, stop ead to avoid tiny oscilations in sloping corners." m_current_position = m_target_position;
if (dot(NewDir, m_normalized_direction) <= .0f) }
break;
}
else
break;
}
else //We moved whole way
m_current_position = m_target_position;
}
} }


//The PreStepis done in order to recover from any HasPenetration. //The PreStepis done in order to recover from any HasPenetration.
void BulletKinematicCharacterController::PreStep(btCollisionWorld* CollisionWorld) void BulletKinematicCharacterController::PreStep(btCollisionWorld* CollisionWorld)
{ {
int MaxPenetrationLoop = 0; int MaxPenetrationLoop = 0;
m_touching_contact = false; m_touching_contact = false;
while (RecoverFromPenetration(CollisionWorld))
while (RecoverFromPenetration(CollisionWorld)) {
{ MaxPenetrationLoop++;
MaxPenetrationLoop++; m_touching_contact = true;
m_touching_contact = true; if (MaxPenetrationLoop > 4)
if (MaxPenetrationLoop > 4) break;
break; }
} m_current_position = BT2LOLU_VEC3(m_ghost_object->getWorldTransform().getOrigin());
m_target_position = m_current_position;
m_current_position = BT2LOLU_VEC3(m_ghost_object->getWorldTransform().getOrigin());
m_target_position = m_current_position;
} }


//And so we step : //And so we step :
//StepUpfirst, then movement, then StepDownon the ground. //StepUpfirst, then movement, then StepDownon the ground.
void BulletKinematicCharacterController::PlayerStep(btCollisionWorld* CollisionWorld, float DeltaTime) void BulletKinematicCharacterController::PlayerStep(btCollisionWorld* CollisionWorld, float DeltaTime)
{ {
// quick check... // quick check...
if (!m_use_walk_direction && m_velocity_time_interval <= .0f) if (!m_use_walk_direction && m_velocity_time_interval <= .0f)
return; // no motion return; // no motion
// Update fall velocity.
// Update fall velocity. //m_velocity -= m_gravity * DeltaTime;
//m_velocity -= m_gravity * DeltaTime; btTransform NewTransform;
NewTransform = m_ghost_object->getWorldTransform();
btTransform NewTransform; vec3 MoveStep(.0f);
NewTransform = m_ghost_object->getWorldTransform(); if (m_use_walk_direction)
MoveStep = m_walk_direction;
vec3 MoveStep(.0f); else
if (m_use_walk_direction) {
MoveStep = m_walk_direction; //Still have some time left for moving!
else float dtMoving = (DeltaTime < m_velocity_time_interval) ? DeltaTime : m_velocity_time_interval;
{ m_velocity_time_interval -= DeltaTime;
//Still have some time left for moving! // how far will we MoveStep while we are moving?
float dtMoving = (DeltaTime < m_velocity_time_interval) ? DeltaTime : m_velocity_time_interval; MoveStep = m_walk_direction * dtMoving;
m_velocity_time_interval -= DeltaTime; }
//Okay, step !
// how far will we MoveStep while we are moving? DoMove(CollisionWorld, MoveStep, DeltaTime);
MoveStep = m_walk_direction * dtMoving; //Movement finished, update World transform
} NewTransform.setOrigin(LOL2BTU_VEC3(m_current_position));
m_ghost_object->setWorldTransform(NewTransform);
//Okay, step !
DoMove(CollisionWorld, MoveStep, DeltaTime);
//Movement finished, update World transform
NewTransform.setOrigin(LOL2BTU_VEC3(m_current_position));
m_ghost_object->setWorldTransform(NewTransform);
} }


//should MoveStep Jump logic in EasyCC //should MoveStep Jump logic in EasyCC
void BulletKinematicCharacterController::Jump() void BulletKinematicCharacterController::Jump()
{ {
if (!CanJump()) if (!CanJump())
return; return;


m_vertical_velocity = m_jump_speed; m_vertical_velocity = m_jump_speed;
m_was_jumping = true; m_was_jumping = true;
} }


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET
#endif // USE_LOL_CTRLR_CHARAC #endif // USE_LOL_CTRLR_CHARAC


} /* namespace phys */ } /* namespace phys */


} /* namespace lol */ } /* namespace lol */

+ 55
- 55
test/Physics/Src/EasyCharacterController.cpp Näytä tiedosto

@@ -38,68 +38,68 @@ void EasyCharacterController::InitBodyToRigid(bool ZeroMassIsKinematic)
//Return correct Ghost Object //Return correct Ghost Object
btGhostObject* EasyCharacterController::GetGhostObjectInstance() btGhostObject* EasyCharacterController::GetGhostObjectInstance()
{ {
return new btPairCachingGhostObject(); return new btPairCachingGhostObject();
} }


//Init to Pair caching ghost object, since Character uses that one. //Init to Pair caching ghost object, since Character uses that one.
void EasyCharacterController::InitBodyToGhost() void EasyCharacterController::InitBodyToGhost()
{ {
EasyPhysic::InitBodyToGhost(); EasyPhysic::InitBodyToGhost();


m_pair_caching_object = (btPairCachingGhostObject*)m_ghost_object; m_pair_caching_object = (btPairCachingGhostObject*)m_ghost_object;
m_ghost_object->setCollisionFlags(btCollisionObject::CF_CHARACTER_OBJECT | m_ghost_object->getCollisionFlags()); m_ghost_object->setCollisionFlags(btCollisionObject::CF_CHARACTER_OBJECT | m_ghost_object->getCollisionFlags());
} }


//Add Physic object to the simulation //Add Physic object to the simulation
void EasyCharacterController::AddToSimulation(class Simulation* current_simulation) void EasyCharacterController::AddToSimulation(class Simulation* current_simulation)
{ {
EasyPhysic::AddToSimulation(current_simulation); EasyPhysic::AddToSimulation(current_simulation);


btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
if (dynamics_world) if (dynamics_world)
{ {
if (m_character) if (m_character)
delete m_character; delete m_character;


//m_character = new btKinematicCharacterController(m_pair_caching_object, m_convex_shape, m_step_height, m_up_axis); //m_character = new btKinematicCharacterController(m_pair_caching_object, m_convex_shape, m_step_height, m_up_axis);
m_character = new BulletKinematicCharacterController(m_pair_caching_object, m_convex_shape, m_step_height, m_up_axis); m_character = new BulletKinematicCharacterController(m_pair_caching_object, m_convex_shape, m_step_height, m_up_axis);


//Deactivate Character controller basic behaviour. //Deactivate Character controller basic behaviour.
//m_character->setGravity(.0f); //m_character->setGravity(.0f);
//m_character->setFallSpeed(.0f); //m_character->setFallSpeed(.0f);


dynamics_world->addAction(m_character); dynamics_world->addAction(m_character);
current_simulation->ObjectRegistration(true, this, Simulation::EEPT_CharacterController); current_simulation->ObjectRegistration(true, this, Simulation::EEPT_CharacterController);
Ticker::Ref(this); Ticker::Ref(this);
} }
} }


//Remove Physic object to the simulation //Remove Physic object to the simulation
void EasyCharacterController::RemoveFromSimulation(class Simulation* current_simulation) void EasyCharacterController::RemoveFromSimulation(class Simulation* current_simulation)
{ {
EasyPhysic::RemoveFromSimulation(current_simulation); EasyPhysic::RemoveFromSimulation(current_simulation);
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); if (dynamics_world)
if (dynamics_world) {
{ if (m_character)
if (m_character) {
{ dynamics_world->removeAction(m_character);
dynamics_world->removeAction(m_character); current_simulation->ObjectRegistration(false, this, Simulation::EEPT_CharacterController);
current_simulation->ObjectRegistration(false, this, Simulation::EEPT_CharacterController); Ticker::Unref(this);
Ticker::Unref(this); }
} }
}
} }


void EasyCharacterController::Jump() void EasyCharacterController::Jump()
{ {
m_character->Jump(); m_character->Jump();
} }


//Set movement for this frame //Set movement for this frame
void EasyCharacterController::SetMovementForFrame(vec3 const &MoveQuantity) void EasyCharacterController::SetMovementForFrame(vec3 const &MoveQuantity)
{ {
m_frame_cached_movement = MoveQuantity; m_frame_cached_movement = MoveQuantity;
} }


//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@@ -107,43 +107,43 @@ void EasyCharacterController::SetMovementForFrame(vec3 const &MoveQuantity)
//-- //--
void EasyCharacterController::SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation) void EasyCharacterController::SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation)
{ {
if (m_base_is_updating) if (m_base_is_updating)
{ {
m_base_cached_movement = base_location - m_local_to_world.v3.xyz; m_base_cached_movement = base_location - m_local_to_world.v3.xyz;
m_local_to_world = lol::mat4::translate(m_local_to_world.v3.xyz) * lol::mat4(base_rotation); m_local_to_world = lol::mat4::translate(m_local_to_world.v3.xyz) * lol::mat4(base_rotation);
if (m_ghost_object) if (m_ghost_object)
m_ghost_object->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * m_local_to_world.v3.xyz))); m_ghost_object->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * m_local_to_world.v3.xyz)));
} }
else else
EasyPhysic::SetTransform(base_location, base_rotation); EasyPhysic::SetTransform(base_location, base_rotation);
} }


//Internal callback when Base transform has changed. //Internal callback when Base transform has changed.
void EasyCharacterController::BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix) void EasyCharacterController::BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix)
{ {
m_base_is_updating = true; m_base_is_updating = true;
EasyPhysic::BaseTransformChanged(PreviousMatrix, NewMatrix); EasyPhysic::BaseTransformChanged(PreviousMatrix, NewMatrix);
m_base_is_updating = false; m_base_is_updating = false;
} }


//--- //---
char const *EasyCharacterController::GetName() char const *EasyCharacterController::GetName()
{ {
return "<EasyCharacterController>"; return "<EasyCharacterController>";
} }


//Physic Tick //Physic Tick
void EasyCharacterController::TickGame(float seconds) void EasyCharacterController::TickGame(float seconds)
{ {
Entity::TickGame(seconds); Entity::TickGame(seconds);
//Send final velocity in Bullet
//Send final velocity in Bullet {
{ int IterationsNb = (int)(seconds / m_owner_simulation->m_timestep);
int IterationsNb = (int)(seconds / m_owner_simulation->m_timestep); float NewSeconds = IterationsNb * m_owner_simulation->m_timestep;
float NewSeconds = IterationsNb * m_owner_simulation->m_timestep; m_character->SetVelocityForTimeInterval((m_base_cached_movement + m_frame_cached_movement) / NewSeconds, NewSeconds);
m_character->SetVelocityForTimeInterval((m_base_cached_movement + m_frame_cached_movement) / NewSeconds, NewSeconds); m_base_cached_movement = vec3(.0f);
m_base_cached_movement = vec3(.0f); }
}
} }


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET


+ 12
- 12
test/Physics/Src/EasyConstraint.cpp Näytä tiedosto

@@ -21,22 +21,22 @@ namespace phys


void EasyConstraint::AddToSimulation(class Simulation* current_simulation) void EasyConstraint::AddToSimulation(class Simulation* current_simulation)
{ {
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
if (dynamics_world && m_typed_constraint) if (dynamics_world && m_typed_constraint)
{ {
dynamics_world->addConstraint(m_typed_constraint, m_disable_a2b_collision); dynamics_world->addConstraint(m_typed_constraint, m_disable_a2b_collision);
current_simulation->ObjectRegistration(true, this); current_simulation->ObjectRegistration(true, this);
} }
} }


void EasyConstraint::RemoveFromSimulation(class Simulation* current_simulation) void EasyConstraint::RemoveFromSimulation(class Simulation* current_simulation)
{ {
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
if (dynamics_world && m_typed_constraint) if (dynamics_world && m_typed_constraint)
{ {
dynamics_world->removeConstraint(m_typed_constraint); dynamics_world->removeConstraint(m_typed_constraint);
current_simulation->ObjectRegistration(false, this); current_simulation->ObjectRegistration(false, this);
} }
} }


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET


+ 176
- 176
test/Physics/Src/EasyPhysics.cpp Näytä tiedosto

@@ -29,29 +29,29 @@ namespace phys
//EASY_PHYSIC //EASY_PHYSIC
//-- //--


EasyPhysic::EasyPhysic(WorldEntity* NewOwnerEntity) : EasyPhysic::EasyPhysic(WorldEntity* NewOwnerEntity) :
m_collision_object(NULL), m_collision_object(NULL),
m_ghost_object(NULL), m_ghost_object(NULL),
m_rigid_body(NULL), m_rigid_body(NULL),
m_local_inertia(btVector3(.0f, .0f, .0f)), m_local_inertia(btVector3(.0f, .0f, .0f)),
m_collision_shape(NULL), m_collision_shape(NULL),
m_convex_shape(NULL), m_convex_shape(NULL),
m_motion_state(NULL), m_motion_state(NULL),
m_mass(.0f), m_mass(.0f),
m_collision_group(1), m_collision_group(1),
m_collision_mask(1), m_collision_mask(1),
m_owner_entity(NewOwnerEntity), m_owner_entity(NewOwnerEntity),
m_owner_simulation(NULL), m_owner_simulation(NULL),
m_base_physic(NULL) m_base_physic(NULL)
{ {
} }


EasyPhysic::~EasyPhysic() EasyPhysic::~EasyPhysic()
{ {
m_rigid_body = NULL; m_rigid_body = NULL;
delete m_collision_object; delete m_collision_object;
delete m_collision_shape; delete m_collision_shape;
delete m_motion_state; delete m_motion_state;
} }


//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@@ -60,55 +60,55 @@ EasyPhysic::~EasyPhysic()


void EasyPhysic::SetShapeTo(btCollisionShape* collision_shape) void EasyPhysic::SetShapeTo(btCollisionShape* collision_shape)
{ {
bool bReinitToRigidBody = false; bool bReinitToRigidBody = false;
if (m_rigid_body) if (m_rigid_body)
{ {
bReinitToRigidBody = true; bReinitToRigidBody = true;
delete m_rigid_body; delete m_rigid_body;
} }
if (m_collision_shape) if (m_collision_shape)
delete m_collision_shape; delete m_collision_shape;
m_collision_shape = collision_shape;
m_collision_shape = collision_shape; if (bReinitToRigidBody)
InitBodyToRigid();
if (bReinitToRigidBody)
InitBodyToRigid();
} }


//Box Shape support //Box Shape support
void EasyPhysic::SetShapeToBox(lol::vec3& box_size) void EasyPhysic::SetShapeToBox(lol::vec3& box_size)
{ {
vec3 new_box_size = box_size * LOL2BT_UNIT * LOL2BT_SIZE; vec3 new_box_size = box_size * LOL2BT_UNIT * LOL2BT_SIZE;
m_convex_shape = new btBoxShape(LOL2BT_VEC3(new_box_size)); m_convex_shape = new btBoxShape(LOL2BT_VEC3(new_box_size));
SetShapeTo(m_convex_shape); SetShapeTo(m_convex_shape);
} }


void EasyPhysic::SetShapeToSphere(float radius) void EasyPhysic::SetShapeToSphere(float radius)
{ {
m_convex_shape = new btSphereShape(radius * LOL2BT_UNIT * LOL2BT_SIZE); m_convex_shape = new btSphereShape(radius * LOL2BT_UNIT * LOL2BT_SIZE);
SetShapeTo(m_convex_shape); SetShapeTo(m_convex_shape);
} }


void EasyPhysic::SetShapeToCone(float radius, float height) void EasyPhysic::SetShapeToCone(float radius, float height)
{ {
m_convex_shape = new btConeShape( radius * LOL2BT_UNIT, m_convex_shape = new btConeShape( radius * LOL2BT_UNIT,
height * LOL2BT_UNIT); height * LOL2BT_UNIT);
SetShapeTo(m_convex_shape); SetShapeTo(m_convex_shape);
} }


void EasyPhysic::SetShapeToCylinder(lol::vec3& cyl_size) void EasyPhysic::SetShapeToCylinder(lol::vec3& cyl_size)
{ {
vec3 new_cyl_size = cyl_size * LOL2BT_UNIT; vec3 new_cyl_size = cyl_size * LOL2BT_UNIT;
new_cyl_size.y *= LOL2BT_SIZE; new_cyl_size.y *= LOL2BT_SIZE;
m_convex_shape = new btCylinderShape(LOL2BT_VEC3(new_cyl_size)); m_convex_shape = new btCylinderShape(LOL2BT_VEC3(new_cyl_size));
SetShapeTo(m_convex_shape); SetShapeTo(m_convex_shape);
} }


void EasyPhysic::SetShapeToCapsule(float radius, float height) void EasyPhysic::SetShapeToCapsule(float radius, float height)
{ {
m_convex_shape = new btCapsuleShape(radius * LOL2BT_UNIT * LOL2BT_SIZE, m_convex_shape = new btCapsuleShape(radius * LOL2BT_UNIT * LOL2BT_SIZE,
height * LOL2BT_UNIT * LOL2BT_SIZE); height * LOL2BT_UNIT * LOL2BT_SIZE);
SetShapeTo(m_convex_shape); SetShapeTo(m_convex_shape);
} }


//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@@ -118,57 +118,57 @@ void EasyPhysic::SetShapeToCapsule(float radius, float height)
//Getter //Getter
mat4 EasyPhysic::GetTransform() mat4 EasyPhysic::GetTransform()
{ {
m_local_to_world = lol::mat4(1.0f); m_local_to_world = lol::mat4(1.0f);
if (m_rigid_body && m_motion_state) if (m_rigid_body && m_motion_state)
{ {
btTransform CurTransform; btTransform CurTransform;
m_motion_state->getWorldTransform(CurTransform); m_motion_state->getWorldTransform(CurTransform);
CurTransform.getOpenGLMatrix(&m_local_to_world[0][0]); CurTransform.getOpenGLMatrix(&m_local_to_world[0][0]);
} }
else if (m_collision_object) else if (m_collision_object)
m_collision_object->getWorldTransform().getOpenGLMatrix(&m_local_to_world[0][0]); m_collision_object->getWorldTransform().getOpenGLMatrix(&m_local_to_world[0][0]);
return m_local_to_world; return m_local_to_world;
} }


//Setter //Setter
void EasyPhysic::SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation) void EasyPhysic::SetTransform(const lol::vec3& base_location, const lol::quat& base_rotation)
{ {
lol::mat4 PreviousMatrix = m_local_to_world; lol::mat4 PreviousMatrix = m_local_to_world;
m_local_to_world = lol::mat4::translate(base_location) * lol::mat4(base_rotation); m_local_to_world = lol::mat4::translate(base_location) * lol::mat4(base_rotation);
if (m_ghost_object)
if (m_ghost_object) m_ghost_object->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location)));
m_ghost_object->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location))); else
else {
{ if (m_motion_state)
if (m_motion_state) m_motion_state->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location)));
m_motion_state->setWorldTransform(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location))); else
else m_motion_state = new btDefaultMotionState(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location)));
m_motion_state = new btDefaultMotionState(btTransform(LOL2BT_QUAT(base_rotation), LOL2BT_VEC3(LOL2BT_UNIT * base_location))); }
} for (int i = 0; i < m_based_physic_list.Count(); i++)
{
for (int i = 0; i < m_based_physic_list.Count(); i++) if (m_based_physic_list[i])
{ m_based_physic_list[i]->BaseTransformChanged(PreviousMatrix, m_local_to_world);
if (m_based_physic_list[i]) else
m_based_physic_list[i]->BaseTransformChanged(PreviousMatrix, m_local_to_world); m_based_physic_list.Remove(i--);
else }
m_based_physic_list.Remove(i--);
}
} }


//Internal callback when Base transform has changed. //Internal callback when Base transform has changed.
void EasyPhysic::BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix) void EasyPhysic::BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol::mat4& NewMatrix)
{ {
mat4 PreviousMatrixLoc = ((m_base_lock_location)?(PreviousMatrix):(lol::mat4::translate(PreviousMatrix.v3.xyz))); mat4 PreviousMatrixLoc = ((m_base_lock_location)?(PreviousMatrix):(lol::mat4::translate(PreviousMatrix.v3.xyz)));
mat4 PreviousMatrixRot = ((m_base_lock_rotation)?(lol::mat4(lol::quat(PreviousMatrix))):(lol::mat4(1.f))); mat4 PreviousMatrixRot = ((m_base_lock_rotation)?(lol::mat4(lol::quat(PreviousMatrix))):(lol::mat4(1.f)));
mat4 NewMatrixLoc = ((m_base_lock_location)?(NewMatrix):(lol::mat4::translate(NewMatrix.v3.xyz))); mat4 NewMatrixLoc = ((m_base_lock_location)?(NewMatrix):(lol::mat4::translate(NewMatrix.v3.xyz)));
mat4 NewMatrixRot = ((m_base_lock_rotation)?(lol::mat4(lol::quat(NewMatrix))):(lol::mat4(1.f))); mat4 NewMatrixRot = ((m_base_lock_rotation)?(lol::mat4(lol::quat(NewMatrix))):(lol::mat4(1.f)));
if (m_ghost_object || (m_rigid_body->getCollisionFlags() & btCollisionObject::CF_KINEMATIC_OBJECT))
if (m_ghost_object || (m_rigid_body->getCollisionFlags() & btCollisionObject::CF_KINEMATIC_OBJECT)) {
{ mat4 ThisMatrixLoc = NewMatrixLoc * inverse(PreviousMatrixLoc) * lol::mat4::translate(m_local_to_world.v3.xyz);
mat4 ThisMatrixLoc = NewMatrixLoc * inverse(PreviousMatrixLoc) * lol::mat4::translate(m_local_to_world.v3.xyz); mat4 ThisMatrixRot = NewMatrixRot * inverse(PreviousMatrixRot) * lol::mat4(lol::quat(m_local_to_world));
mat4 ThisMatrixRot = NewMatrixRot * inverse(PreviousMatrixRot) * lol::mat4(lol::quat(m_local_to_world)); SetTransform(ThisMatrixLoc.v3.xyz, lol::mat4(lol::quat(ThisMatrixRot)));
SetTransform(ThisMatrixLoc.v3.xyz, lol::mat4(lol::quat(ThisMatrixRot))); }
}
} }


//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@@ -177,13 +177,13 @@ void EasyPhysic::BaseTransformChanged(const lol::mat4& PreviousMatrix, const lol
//Set Shape functions //Set Shape functions
void EasyPhysic::SetMass(float mass) void EasyPhysic::SetMass(float mass)
{ {
m_mass = mass; m_mass = mass;


if (m_rigid_body) if (m_rigid_body)
{ {
SetLocalInertia(m_mass); SetLocalInertia(m_mass);
m_rigid_body->setMassProps(mass, m_local_inertia); m_rigid_body->setMassProps(mass, m_local_inertia);
} }
} }


//------------------------------------------------------------------------- //-------------------------------------------------------------------------
@@ -193,49 +193,49 @@ void EasyPhysic::SetMass(float mass)
//Init to rigid body //Init to rigid body
void EasyPhysic::InitBodyToRigid(bool SetToKinematic) void EasyPhysic::InitBodyToRigid(bool SetToKinematic)
{ {
if (m_collision_object) if (m_collision_object)
delete m_collision_object; delete m_collision_object;
if (!m_motion_state)
if (!m_motion_state) SetTransform(vec3(.0f));
SetTransform(vec3(.0f)); btRigidBody::btRigidBodyConstructionInfo NewInfos(m_mass, m_motion_state, m_collision_shape, m_local_inertia);
m_rigid_body = new btRigidBody(NewInfos);
btRigidBody::btRigidBodyConstructionInfo NewInfos(m_mass, m_motion_state, m_collision_shape, m_local_inertia); m_collision_object = m_rigid_body;
m_rigid_body = new btRigidBody(NewInfos); m_collision_object->setUserPointer(this);
m_collision_object = m_rigid_body; if (m_mass == .0f)
m_collision_object->setUserPointer(this); {
if (SetToKinematic)
if (m_mass == .0f) {
{ m_rigid_body->setActivationState(DISABLE_DEACTIVATION);
if (SetToKinematic) m_rigid_body->setCollisionFlags(m_rigid_body->getCollisionFlags() | btCollisionObject::CF_KINEMATIC_OBJECT);
{ }
m_rigid_body->setActivationState(DISABLE_DEACTIVATION); }
m_rigid_body->setCollisionFlags(m_rigid_body->getCollisionFlags() | btCollisionObject::CF_KINEMATIC_OBJECT); else
} SetMass(m_mass);
}
else
SetMass(m_mass);
} }


//Return correct Ghost Object //Return correct Ghost Object
btGhostObject* EasyPhysic::GetGhostObjectInstance() btGhostObject* EasyPhysic::GetGhostObjectInstance()
{ {
return new btGhostObject(); return new btGhostObject();
} }


//Init to Ghost object, for Overlap/Sweep Test/Touching logic //Init to Ghost object, for Overlap/Sweep Test/Touching logic
void EasyPhysic::InitBodyToGhost() void EasyPhysic::InitBodyToGhost()
{ {
if (m_collision_object) if (m_collision_object)
delete m_collision_object; delete m_collision_object;


m_ghost_object = GetGhostObjectInstance(); m_ghost_object = GetGhostObjectInstance();
m_ghost_object->setCollisionShape(m_collision_shape); m_ghost_object->setCollisionShape(m_collision_shape);
m_collision_object = m_ghost_object; m_collision_object = m_ghost_object;
m_collision_object->setUserPointer(this); m_collision_object->setUserPointer(this);


SetTransform(m_local_to_world.v3.xyz, lol::quat(m_local_to_world)); SetTransform(m_local_to_world.v3.xyz, lol::quat(m_local_to_world));


m_ghost_object->setCollisionFlags(m_ghost_object->getCollisionFlags()); m_ghost_object->setCollisionFlags(m_ghost_object->getCollisionFlags());
} }


//------------- //-------------
@@ -250,7 +250,7 @@ void EasyPhysic::InitBodyToGhost()
// manifoldArray.clear(); // manifoldArray.clear();


// const btBroadphasePair& pair = pairArray[i]; // const btBroadphasePair& pair = pairArray[i];
// //
// //unless we manually perform collision detection on this pair, the contacts are in the dynamics world paircache: // //unless we manually perform collision detection on this pair, the contacts are in the dynamics world paircache:
// btBroadphasePair* collisionPair = dynamicsWorld->getPairCache()->findPair(pair.m_pProxy0,pair.m_pProxy1); // btBroadphasePair* collisionPair = dynamicsWorld->getPairCache()->findPair(pair.m_pProxy0,pair.m_pProxy1);
// if (!collisionPair) // if (!collisionPair)
@@ -265,14 +265,14 @@ void EasyPhysic::InitBodyToGhost()
// btScalar directionSign = manifold->getBody0() == m_ghostObject ? btScalar(-1.0) : btScalar(1.0); // btScalar directionSign = manifold->getBody0() == m_ghostObject ? btScalar(-1.0) : btScalar(1.0);
// for (int p=0;p<manifold->getNumContacts();p++) // for (int p=0;p<manifold->getNumContacts();p++)
// { // {
// const btManifoldPoint&pt = manifold->getContactPoint(p); // const btManifoldPoint&pt = manifold->getContactPoint(p);
// if (pt.getDistance()<0.f) // if (pt.getDistance()<0.f)
//{ //{
// const btVector3& ptA = pt.getPositionWorldOnA(); // const btVector3& ptA = pt.getPositionWorldOnA();
// const btVector3& ptB = pt.getPositionWorldOnB(); // const btVector3& ptB = pt.getPositionWorldOnB();
// const btVector3& normalOnB = pt.m_normalWorldOnB; // const btVector3& normalOnB = pt.m_normalWorldOnB;
// /// work here // /// work here
//} //}
// } // }
// } // }
// } // }
@@ -281,61 +281,61 @@ void EasyPhysic::InitBodyToGhost()
//Add Physic object to the simulation //Add Physic object to the simulation
void EasyPhysic::AddToSimulation(class Simulation* current_simulation) void EasyPhysic::AddToSimulation(class Simulation* current_simulation)
{ {
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
if (dynamics_world) if (dynamics_world)
{ {
if (m_ghost_object) if (m_ghost_object)
{ {
dynamics_world->addCollisionObject(m_ghost_object, m_collision_group, m_collision_mask); dynamics_world->addCollisionObject(m_ghost_object, m_collision_group, m_collision_mask);
current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Ghost); current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Ghost);
} }
else if (m_rigid_body) else if (m_rigid_body)
{ {
dynamics_world->addRigidBody(m_rigid_body, m_collision_group, m_collision_mask); dynamics_world->addRigidBody(m_rigid_body, m_collision_group, m_collision_mask);
if (m_mass != .0f) if (m_mass != .0f)
current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Dynamic); current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Dynamic);
else else
current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Static); current_simulation->ObjectRegistration(true, this, Simulation::EEPT_Static);
} }
else else
{ {
dynamics_world->addCollisionObject(m_collision_object, m_collision_group, m_collision_mask); dynamics_world->addCollisionObject(m_collision_object, m_collision_group, m_collision_mask);
current_simulation->ObjectRegistration(true, this, Simulation::EEPT_CollisionObject); current_simulation->ObjectRegistration(true, this, Simulation::EEPT_CollisionObject);
} }
} }
} }


//Remove Physic object to the simulation //Remove Physic object to the simulation
void EasyPhysic::RemoveFromSimulation(class Simulation* current_simulation) void EasyPhysic::RemoveFromSimulation(class Simulation* current_simulation)
{ {
btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld(); btDiscreteDynamicsWorld* dynamics_world = current_simulation->GetWorld();
if (dynamics_world) if (dynamics_world)
{ {
if (m_rigid_body) if (m_rigid_body)
{ {
dynamics_world->removeRigidBody(m_rigid_body); dynamics_world->removeRigidBody(m_rigid_body);
if (m_mass != .0f) if (m_mass != .0f)
current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Dynamic); current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Dynamic);
else else
current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Static); current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Static);
} }
else else
{ {
dynamics_world->removeCollisionObject(m_collision_object); dynamics_world->removeCollisionObject(m_collision_object);
if (m_ghost_object) if (m_ghost_object)
current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Ghost); current_simulation->ObjectRegistration(false, this, Simulation::EEPT_Ghost);
current_simulation->ObjectRegistration(false, this, Simulation::EEPT_CollisionObject); current_simulation->ObjectRegistration(false, this, Simulation::EEPT_CollisionObject);
} }
} }
} }


//Set Local Inertia //Set Local Inertia
void EasyPhysic::SetLocalInertia(float mass) void EasyPhysic::SetLocalInertia(float mass)
{ {
if (mass != .0f) if (mass != .0f)
m_collision_shape->calculateLocalInertia(mass, m_local_inertia); m_collision_shape->calculateLocalInertia(mass, m_local_inertia);
else else
m_local_inertia = btVector3(.0f, .0f, .0f); m_local_inertia = btVector3(.0f, .0f, .0f);
} }


#endif // HAVE_PHYS_USE_BULLET #endif // HAVE_PHYS_USE_BULLET


||||||
x
 
000:0
Ladataan…
Peruuta
Tallenna