2010-11-20 10:37:14 +00:00
|
|
|
/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */
|
|
|
|
/* If you are missing that file, acquire a complete release at teeworlds.com. */
|
2010-05-29 07:25:38 +00:00
|
|
|
#include "gamecore.h"
|
2007-09-22 18:55:00 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
const char *CTuningParams::m_apNames[] =
|
2008-02-02 12:38:36 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
#define MACRO_TUNING_PARAM(Name,ScriptName,Value) #ScriptName,
|
|
|
|
#include "tuning.h"
|
2008-02-02 12:38:36 +00:00
|
|
|
#undef MACRO_TUNING_PARAM
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
bool CTuningParams::Set(int Index, float Value)
|
2008-02-02 12:38:36 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Index < 0 || Index >= Num())
|
2008-02-02 12:38:36 +00:00
|
|
|
return false;
|
2010-05-29 07:25:38 +00:00
|
|
|
((CTuneParam *)this)[Index] = Value;
|
2008-02-02 12:38:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
bool CTuningParams::Get(int Index, float *pValue)
|
2008-02-02 12:38:36 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Index < 0 || Index >= Num())
|
2008-02-02 12:38:36 +00:00
|
|
|
return false;
|
2010-05-29 07:25:38 +00:00
|
|
|
*pValue = (float)((CTuneParam *)this)[Index];
|
2008-02-02 12:38:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
bool CTuningParams::Set(const char *pName, float Value)
|
2008-02-02 12:38:36 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
for(int i = 0; i < Num(); i++)
|
|
|
|
if(str_comp_nocase(pName, m_apNames[i]) == 0)
|
|
|
|
return Set(i, Value);
|
2008-02-02 12:38:36 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
bool CTuningParams::Get(const char *pName, float *pValue)
|
2008-02-02 12:38:36 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
for(int i = 0; i < Num(); i++)
|
|
|
|
if(str_comp_nocase(pName, m_apNames[i]) == 0)
|
|
|
|
return Get(i, pValue);
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-02-02 12:38:36 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-02-13 06:47:51 +00:00
|
|
|
float HermiteBasis1(float v)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
return 2*v*v*v - 3*v*v+1;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
float VelocityRamp(float Value, float Start, float Range, float Curvature)
|
2008-03-16 22:32:17 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Value < Start)
|
|
|
|
return 1.0f;
|
|
|
|
return 1.0f/powf(Curvature, (Value-Start)/Range);
|
2008-03-16 22:32:17 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Init(CWorldCore *pWorld, CCollision *pCollision)
|
2008-03-16 22:32:17 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_pWorld = pWorld;
|
|
|
|
m_pCollision = pCollision;
|
2008-03-16 22:32:17 +00:00
|
|
|
}
|
2008-02-24 18:41:02 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Reset()
|
2008-02-24 18:41:02 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Pos = vec2(0,0);
|
|
|
|
m_Vel = vec2(0,0);
|
|
|
|
m_HookPos = vec2(0,0);
|
|
|
|
m_HookDir = vec2(0,0);
|
|
|
|
m_HookTick = 0;
|
|
|
|
m_HookState = HOOK_IDLE;
|
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_Jumped = 0;
|
|
|
|
m_TriggeredEvents = 0;
|
2008-02-24 18:41:02 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Tick(bool UseInput)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
float PhysSize = 28.0f;
|
|
|
|
m_TriggeredEvents = 0;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// get ground state
|
2010-05-29 07:25:38 +00:00
|
|
|
bool Grounded = false;
|
|
|
|
if(m_pCollision->CheckPoint(m_Pos.x+PhysSize/2, m_Pos.y+PhysSize/2+5))
|
|
|
|
Grounded = true;
|
|
|
|
if(m_pCollision->CheckPoint(m_Pos.x-PhysSize/2, m_Pos.y+PhysSize/2+5))
|
|
|
|
Grounded = true;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
vec2 TargetDirection = normalize(vec2(m_Input.m_TargetX, m_Input.m_TargetY));
|
2007-09-22 18:55:00 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Vel.y += m_pWorld->m_Tuning.m_Gravity;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
float MaxSpeed = Grounded ? m_pWorld->m_Tuning.m_GroundControlSpeed : m_pWorld->m_Tuning.m_AirControlSpeed;
|
|
|
|
float Accel = Grounded ? m_pWorld->m_Tuning.m_GroundControlAccel : m_pWorld->m_Tuning.m_AirControlAccel;
|
|
|
|
float Friction = Grounded ? m_pWorld->m_Tuning.m_GroundFriction : m_pWorld->m_Tuning.m_AirFriction;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// handle input
|
2010-05-29 07:25:38 +00:00
|
|
|
if(UseInput)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Direction = m_Input.m_Direction;
|
2008-09-23 07:43:41 +00:00
|
|
|
|
|
|
|
// setup angle
|
2011-02-13 06:47:51 +00:00
|
|
|
float a = 0;
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_Input.m_TargetX == 0)
|
2011-02-13 06:47:51 +00:00
|
|
|
a = atanf((float)m_Input.m_TargetY);
|
2008-09-23 07:43:41 +00:00
|
|
|
else
|
2011-02-13 06:47:51 +00:00
|
|
|
a = atanf((float)m_Input.m_TargetY/(float)m_Input.m_TargetX);
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_Input.m_TargetX < 0)
|
2011-02-13 06:47:51 +00:00
|
|
|
a = a+pi;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2011-02-13 06:47:51 +00:00
|
|
|
m_Angle = (int)(a*256.0f);
|
2008-09-23 07:43:41 +00:00
|
|
|
|
|
|
|
// handle jump
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_Input.m_Jump)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(!(m_Jumped&1))
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Grounded)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_TriggeredEvents |= COREEVENT_GROUND_JUMP;
|
|
|
|
m_Vel.y = -m_pWorld->m_Tuning.m_GroundJumpImpulse;
|
|
|
|
m_Jumped |= 1;
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
else if(!(m_Jumped&2))
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_TriggeredEvents |= COREEVENT_AIR_JUMP;
|
|
|
|
m_Vel.y = -m_pWorld->m_Tuning.m_AirJumpImpulse;
|
|
|
|
m_Jumped |= 3;
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Jumped &= ~1;
|
2008-09-23 07:43:41 +00:00
|
|
|
|
|
|
|
// handle hook
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_Input.m_Hook)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookState == HOOK_IDLE)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookState = HOOK_FLYING;
|
|
|
|
m_HookPos = m_Pos+TargetDirection*PhysSize*1.5f;
|
|
|
|
m_HookDir = TargetDirection;
|
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_HookTick = 0;
|
|
|
|
m_TriggeredEvents |= COREEVENT_HOOK_LAUNCH;
|
2011-04-13 18:37:12 +00:00
|
|
|
}
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_HookState = HOOK_IDLE;
|
2011-04-13 18:37:12 +00:00
|
|
|
m_HookPos = m_Pos;
|
|
|
|
}
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// add the speed modification according to players wanted direction
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_Direction < 0)
|
|
|
|
m_Vel.x = SaturatedAdd(-MaxSpeed, MaxSpeed, m_Vel.x, -Accel);
|
|
|
|
if(m_Direction > 0)
|
|
|
|
m_Vel.x = SaturatedAdd(-MaxSpeed, MaxSpeed, m_Vel.x, Accel);
|
|
|
|
if(m_Direction == 0)
|
|
|
|
m_Vel.x *= Friction;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// handle jumping
|
2007-12-09 09:48:53 +00:00
|
|
|
// 1 bit = to keep track if a jump has been made on this input
|
|
|
|
// 2 bit = to keep track if a air-jump has been made
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Grounded)
|
|
|
|
m_Jumped &= ~2;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// do hook
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookState == HOOK_IDLE)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_HookState = HOOK_IDLE;
|
|
|
|
m_HookPos = m_Pos;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
else if(m_HookState >= HOOK_RETRACT_START && m_HookState < HOOK_RETRACT_END)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookState++;
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
else if(m_HookState == HOOK_RETRACT_END)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookState = HOOK_RETRACTED;
|
|
|
|
m_TriggeredEvents |= COREEVENT_HOOK_RETRACT;
|
|
|
|
m_HookState = HOOK_RETRACTED;
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
else if(m_HookState == HOOK_FLYING)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
vec2 NewPos = m_HookPos+m_HookDir*m_pWorld->m_Tuning.m_HookFireSpeed;
|
|
|
|
if(distance(m_Pos, NewPos) > m_pWorld->m_Tuning.m_HookLength)
|
2008-03-22 13:03:52 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookState = HOOK_RETRACT_START;
|
|
|
|
NewPos = m_Pos + normalize(NewPos-m_Pos) * m_pWorld->m_Tuning.m_HookLength;
|
2008-03-19 23:08:26 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// make sure that the hook doesn't go though the ground
|
2010-05-29 07:25:38 +00:00
|
|
|
bool GoingToHitGround = false;
|
|
|
|
bool GoingToRetract = false;
|
|
|
|
int Hit = m_pCollision->IntersectLine(m_HookPos, NewPos, &NewPos, 0);
|
|
|
|
if(Hit)
|
2008-09-23 14:38:13 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(Hit&CCollision::COLFLAG_NOHOOK)
|
|
|
|
GoingToRetract = true;
|
2008-09-23 14:38:13 +00:00
|
|
|
else
|
2010-05-29 07:25:38 +00:00
|
|
|
GoingToHitGround = true;
|
2008-09-23 14:38:13 +00:00
|
|
|
}
|
2007-09-22 18:55:00 +00:00
|
|
|
|
2008-09-23 07:43:41 +00:00
|
|
|
// Check against other players first
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_pWorld && m_pWorld->m_Tuning.m_PlayerHooking)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
float Distance = 0.0f;
|
2007-09-22 18:55:00 +00:00
|
|
|
for(int i = 0; i < MAX_CLIENTS; i++)
|
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
CCharacterCore *pCharCore = m_pWorld->m_apCharacters[i];
|
|
|
|
if(!pCharCore || pCharCore == this)
|
2007-09-22 18:55:00 +00:00
|
|
|
continue;
|
|
|
|
|
2011-02-12 10:40:36 +00:00
|
|
|
vec2 ClosestPoint = closest_point_on_line(m_HookPos, NewPos, pCharCore->m_Pos);
|
|
|
|
if(distance(pCharCore->m_Pos, ClosestPoint) < PhysSize+2.0f)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
if (m_HookedPlayer == -1 || distance(m_HookPos, pCharCore->m_Pos) < Distance)
|
2009-01-09 22:41:26 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_TriggeredEvents |= COREEVENT_HOOK_ATTACH_PLAYER;
|
|
|
|
m_HookState = HOOK_GRABBED;
|
|
|
|
m_HookedPlayer = i;
|
2011-02-13 06:47:51 +00:00
|
|
|
Distance = distance(m_HookPos, pCharCore->m_Pos);
|
2009-01-09 22:41:26 +00:00
|
|
|
}
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
}
|
2008-09-23 07:43:41 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookState == HOOK_FLYING)
|
2008-09-23 07:43:41 +00:00
|
|
|
{
|
|
|
|
// check against ground
|
2010-05-29 07:25:38 +00:00
|
|
|
if(GoingToHitGround)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_TriggeredEvents |= COREEVENT_HOOK_ATTACH_GROUND;
|
|
|
|
m_HookState = HOOK_GRABBED;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
2010-05-29 07:25:38 +00:00
|
|
|
else if(GoingToRetract)
|
2008-09-23 14:38:13 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_TriggeredEvents |= COREEVENT_HOOK_HIT_NOHOOK;
|
|
|
|
m_HookState = HOOK_RETRACT_START;
|
2008-09-23 14:38:13 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookPos = NewPos;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookState == HOOK_GRABBED)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookedPlayer != -1)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
CCharacterCore *pCharCore = m_pWorld->m_apCharacters[m_HookedPlayer];
|
|
|
|
if(pCharCore)
|
|
|
|
m_HookPos = pCharCore->m_Pos;
|
2007-12-09 15:58:24 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// release hook
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_HookState = HOOK_RETRACTED;
|
2011-04-13 18:37:12 +00:00
|
|
|
m_HookPos = m_Pos;
|
2007-12-09 15:58:24 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// keep players hooked for a max of 1.5sec
|
2010-05-29 07:25:38 +00:00
|
|
|
//if(Server()->Tick() > hook_tick+(Server()->TickSpeed()*3)/2)
|
2007-09-22 18:55:00 +00:00
|
|
|
//release_hooked();
|
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-12-15 10:49:47 +00:00
|
|
|
// don't do this hook rutine when we are hook to a player
|
2010-05-29 07:25:38 +00:00
|
|
|
if(m_HookedPlayer == -1 && distance(m_HookPos, m_Pos) > 46.0f)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
vec2 HookVel = normalize(m_HookPos-m_Pos)*m_pWorld->m_Tuning.m_HookDragAccel;
|
2007-09-22 18:55:00 +00:00
|
|
|
// the hook as more power to drag you up then down.
|
|
|
|
// this makes it easier to get on top of an platform
|
2010-05-29 07:25:38 +00:00
|
|
|
if(HookVel.y > 0)
|
|
|
|
HookVel.y *= 0.3f;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// the hook will boost it's power if the player wants to move
|
|
|
|
// in that direction. otherwise it will dampen everything abit
|
2011-04-13 18:37:12 +00:00
|
|
|
if((HookVel.x < 0 && m_Direction < 0) || (HookVel.x > 0 && m_Direction > 0))
|
2010-05-29 07:25:38 +00:00
|
|
|
HookVel.x *= 0.95f;
|
2007-09-22 18:55:00 +00:00
|
|
|
else
|
2010-05-29 07:25:38 +00:00
|
|
|
HookVel.x *= 0.75f;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
vec2 NewVel = m_Vel+HookVel;
|
2008-01-29 21:39:41 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// check if we are under the legal limit for the hook
|
2010-05-29 07:25:38 +00:00
|
|
|
if(length(NewVel) < m_pWorld->m_Tuning.m_HookDragSpeed || length(NewVel) < length(m_Vel))
|
|
|
|
m_Vel = NewVel; // no problem. apply
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
2007-12-09 15:56:56 +00:00
|
|
|
|
2009-01-11 11:54:41 +00:00
|
|
|
// release hook (max hook time is 1.25
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookTick++;
|
|
|
|
if(m_HookedPlayer != -1 && (m_HookTick > SERVER_TICK_SPEED+SERVER_TICK_SPEED/5 || !m_pWorld->m_apCharacters[m_HookedPlayer]))
|
2007-12-09 15:56:56 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_HookedPlayer = -1;
|
|
|
|
m_HookState = HOOK_RETRACTED;
|
2011-04-13 18:37:12 +00:00
|
|
|
m_HookPos = m_Pos;
|
2007-12-09 15:56:56 +00:00
|
|
|
}
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2011-08-06 11:57:59 +00:00
|
|
|
if(m_pWorld)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
|
|
|
for(int i = 0; i < MAX_CLIENTS; i++)
|
|
|
|
{
|
2011-02-12 10:40:36 +00:00
|
|
|
CCharacterCore *pCharCore = m_pWorld->m_apCharacters[i];
|
|
|
|
if(!pCharCore)
|
2007-09-22 18:55:00 +00:00
|
|
|
continue;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
//player *p = (player*)ent;
|
2011-02-12 10:40:36 +00:00
|
|
|
if(pCharCore == this) // || !(p->flags&FLAG_ALIVE)
|
2007-09-22 18:55:00 +00:00
|
|
|
continue; // make sure that we don't nudge our self
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// handle player <-> player collision
|
2011-02-13 06:47:51 +00:00
|
|
|
float Distance = distance(m_Pos, pCharCore->m_Pos);
|
2011-02-12 10:40:36 +00:00
|
|
|
vec2 Dir = normalize(m_Pos - pCharCore->m_Pos);
|
2011-08-06 11:57:59 +00:00
|
|
|
if(m_pWorld->m_Tuning.m_PlayerCollision && Distance < PhysSize*1.25f && Distance > 0.0f)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
float a = (PhysSize*1.45f - Distance);
|
|
|
|
float Velocity = 0.5f;
|
2011-01-18 05:50:24 +00:00
|
|
|
|
2007-12-15 10:49:47 +00:00
|
|
|
// make sure that we don't add excess force by checking the
|
2011-01-18 05:50:24 +00:00
|
|
|
// direction against the current velocity. if not zero.
|
|
|
|
if (length(m_Vel) > 0.0001)
|
2011-02-13 06:47:51 +00:00
|
|
|
Velocity = 1-(dot(normalize(m_Vel), Dir)+1)/2;
|
2011-01-18 05:50:24 +00:00
|
|
|
|
2011-02-13 06:47:51 +00:00
|
|
|
m_Vel += Dir*a*(Velocity*0.75f);
|
2011-01-18 05:50:24 +00:00
|
|
|
m_Vel *= 0.85f;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-09-22 18:55:00 +00:00
|
|
|
// handle hook influence
|
2011-08-06 11:57:59 +00:00
|
|
|
if(m_HookedPlayer == i && m_pWorld->m_Tuning.m_PlayerHooking)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
if(Distance > PhysSize*1.50f) // TODO: fix tweakable variable
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2011-02-13 06:47:51 +00:00
|
|
|
float Accel = m_pWorld->m_Tuning.m_HookDragAccel * (Distance/m_pWorld->m_Tuning.m_HookLength);
|
2010-05-29 07:25:38 +00:00
|
|
|
float DragSpeed = m_pWorld->m_Tuning.m_HookDragSpeed;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2007-12-15 10:49:47 +00:00
|
|
|
// add force to the hooked player
|
2011-02-12 10:40:36 +00:00
|
|
|
pCharCore->m_Vel.x = SaturatedAdd(-DragSpeed, DragSpeed, pCharCore->m_Vel.x, Accel*Dir.x*1.5f);
|
|
|
|
pCharCore->m_Vel.y = SaturatedAdd(-DragSpeed, DragSpeed, pCharCore->m_Vel.y, Accel*Dir.y*1.5f);
|
2007-12-15 10:49:47 +00:00
|
|
|
|
|
|
|
// add a little bit force to the guy who has the grip
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Vel.x = SaturatedAdd(-DragSpeed, DragSpeed, m_Vel.x, -Accel*Dir.x*0.25f);
|
|
|
|
m_Vel.y = SaturatedAdd(-DragSpeed, DragSpeed, m_Vel.y, -Accel*Dir.y*0.25f);
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
}
|
2008-01-20 15:19:30 +00:00
|
|
|
|
|
|
|
// clamp the velocity to something sane
|
2010-05-29 07:25:38 +00:00
|
|
|
if(length(m_Vel) > 6000)
|
|
|
|
m_Vel = normalize(m_Vel) * 6000;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Move()
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
float RampValue = VelocityRamp(length(m_Vel)*50, m_pWorld->m_Tuning.m_VelrampStart, m_pWorld->m_Tuning.m_VelrampRange, m_pWorld->m_Tuning.m_VelrampCurvature);
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Vel.x = m_Vel.x*RampValue;
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2011-07-30 16:08:24 +00:00
|
|
|
vec2 NewPos = m_Pos;
|
2011-03-27 14:15:10 +00:00
|
|
|
m_pCollision->MoveBox(&NewPos, &m_Vel, vec2(28.0f, 28.0f), 0);
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Vel.x = m_Vel.x*(1.0f/RampValue);
|
2011-03-27 14:15:10 +00:00
|
|
|
|
|
|
|
if(m_pWorld && m_pWorld->m_Tuning.m_PlayerCollision)
|
|
|
|
{
|
2011-07-30 16:08:24 +00:00
|
|
|
// check player collision
|
|
|
|
float Distance = distance(m_Pos, NewPos);
|
|
|
|
int End = Distance+1;
|
|
|
|
vec2 LastPos = m_Pos;
|
|
|
|
for(int i = 0; i < End; i++)
|
|
|
|
{
|
|
|
|
float a = i/Distance;
|
|
|
|
vec2 Pos = mix(m_Pos, NewPos, a);
|
|
|
|
for(int p = 0; p < MAX_CLIENTS; p++)
|
|
|
|
{
|
|
|
|
CCharacterCore *pCharCore = m_pWorld->m_apCharacters[p];
|
|
|
|
if(!pCharCore || pCharCore == this)
|
|
|
|
continue;
|
|
|
|
float D = distance(Pos, pCharCore->m_Pos);
|
|
|
|
if(D < 28.0f && D > 0.0f)
|
|
|
|
{
|
|
|
|
if(a > 0.0f)
|
|
|
|
m_Pos = LastPos;
|
|
|
|
else if(distance(NewPos, pCharCore->m_Pos) > D)
|
|
|
|
m_Pos = NewPos;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LastPos = Pos;
|
2011-04-01 19:32:16 +00:00
|
|
|
}
|
2011-03-27 14:15:10 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2011-04-01 19:32:16 +00:00
|
|
|
m_Pos = NewPos;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Write(CNetObj_CharacterCore *pObjCore)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
pObjCore->m_X = round(m_Pos.x);
|
|
|
|
pObjCore->m_Y = round(m_Pos.y);
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
pObjCore->m_VelX = round(m_Vel.x*256.0f);
|
|
|
|
pObjCore->m_VelY = round(m_Vel.y*256.0f);
|
|
|
|
pObjCore->m_HookState = m_HookState;
|
|
|
|
pObjCore->m_HookTick = m_HookTick;
|
|
|
|
pObjCore->m_HookX = round(m_HookPos.x);
|
|
|
|
pObjCore->m_HookY = round(m_HookPos.y);
|
|
|
|
pObjCore->m_HookDx = round(m_HookDir.x*256.0f);
|
|
|
|
pObjCore->m_HookDy = round(m_HookDir.y*256.0f);
|
|
|
|
pObjCore->m_HookedPlayer = m_HookedPlayer;
|
|
|
|
pObjCore->m_Jumped = m_Jumped;
|
|
|
|
pObjCore->m_Direction = m_Direction;
|
|
|
|
pObjCore->m_Angle = m_Angle;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Read(const CNetObj_CharacterCore *pObjCore)
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
m_Pos.x = pObjCore->m_X;
|
|
|
|
m_Pos.y = pObjCore->m_Y;
|
|
|
|
m_Vel.x = pObjCore->m_VelX/256.0f;
|
|
|
|
m_Vel.y = pObjCore->m_VelY/256.0f;
|
|
|
|
m_HookState = pObjCore->m_HookState;
|
|
|
|
m_HookTick = pObjCore->m_HookTick;
|
|
|
|
m_HookPos.x = pObjCore->m_HookX;
|
|
|
|
m_HookPos.y = pObjCore->m_HookY;
|
|
|
|
m_HookDir.x = pObjCore->m_HookDx/256.0f;
|
|
|
|
m_HookDir.y = pObjCore->m_HookDy/256.0f;
|
|
|
|
m_HookedPlayer = pObjCore->m_HookedPlayer;
|
|
|
|
m_Jumped = pObjCore->m_Jumped;
|
|
|
|
m_Direction = pObjCore->m_Direction;
|
|
|
|
m_Angle = pObjCore->m_Angle;
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
void CCharacterCore::Quantize()
|
2007-09-22 18:55:00 +00:00
|
|
|
{
|
2010-05-29 07:25:38 +00:00
|
|
|
CNetObj_CharacterCore Core;
|
|
|
|
Write(&Core);
|
|
|
|
Read(&Core);
|
2007-09-22 18:55:00 +00:00
|
|
|
}
|
|
|
|
|