2011-12-25 13:33:05 +00:00
|
|
|
/* (c) Shereef Marzouk. See "licence DDRace.txt" and the readme.txt in the root of the distribution for more information. */
|
2020-09-26 19:41:58 +00:00
|
|
|
#include "dragger.h"
|
2010-07-29 05:21:18 +00:00
|
|
|
#include <engine/config.h>
|
|
|
|
#include <engine/server.h>
|
2015-09-01 21:15:48 +00:00
|
|
|
#include <engine/shared/config.h>
|
2010-07-29 05:21:18 +00:00
|
|
|
#include <game/generated/protocol.h>
|
|
|
|
#include <game/server/gamecontext.h>
|
2011-12-25 13:00:47 +00:00
|
|
|
#include <game/server/gamemodes/DDRace.h>
|
2021-01-09 14:37:02 +00:00
|
|
|
#include <game/server/player.h>
|
2020-09-26 19:41:58 +00:00
|
|
|
#include <game/server/teams.h>
|
2021-10-14 20:59:39 +00:00
|
|
|
#include <game/version.h>
|
2010-07-29 05:21:18 +00:00
|
|
|
|
2021-01-09 14:18:52 +00:00
|
|
|
#include "character.h"
|
|
|
|
|
2011-12-25 13:51:04 +00:00
|
|
|
CDragger::CDragger(CGameWorld *pGameWorld, vec2 Pos, float Strength, bool NW,
|
2020-09-26 19:41:58 +00:00
|
|
|
int CaughtTeam, int Layer, int Number) :
|
|
|
|
CEntity(pGameWorld, CGameWorld::ENTTYPE_LASER)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-11-04 22:06:44 +00:00
|
|
|
m_Target = 0;
|
2010-11-13 13:22:19 +00:00
|
|
|
m_Layer = Layer;
|
|
|
|
m_Number = Number;
|
2010-08-30 23:45:42 +00:00
|
|
|
m_Pos = Pos;
|
|
|
|
m_Strength = Strength;
|
|
|
|
m_EvalTick = Server()->Tick();
|
|
|
|
m_NW = NW;
|
2019-04-08 19:54:33 +00:00
|
|
|
m_CaughtTeam = CaughtTeam;
|
2010-07-29 05:21:18 +00:00
|
|
|
GameWorld()->InsertEntity(this);
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-10-26 14:14:07 +00:00
|
|
|
for(int &SoloID : m_SoloIDs)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2020-10-26 14:14:07 +00:00
|
|
|
SoloID = -1;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
|
|
|
|
2010-08-30 23:45:42 +00:00
|
|
|
void CDragger::Move()
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(m_Target && (!m_Target->IsAlive() || (m_Target->IsAlive() && (m_Target->m_Super || m_Target->IsPaused() || (m_Layer == LAYER_SWITCH && m_Number && !GameServer()->Collision()->m_pSwitchers[m_Number].m_Status[m_Target->Team()])))))
|
2010-11-13 13:22:19 +00:00
|
|
|
m_Target = 0;
|
2014-05-08 13:42:20 +00:00
|
|
|
|
2014-05-07 22:53:21 +00:00
|
|
|
mem_zero(m_SoloEnts, sizeof(m_SoloEnts));
|
2014-05-08 13:42:20 +00:00
|
|
|
CCharacter *TempEnts[MAX_CLIENTS];
|
|
|
|
|
2015-09-01 21:15:48 +00:00
|
|
|
int Num = GameServer()->m_World.FindEntities(m_Pos, g_Config.m_SvDraggerRange,
|
2020-09-26 19:41:58 +00:00
|
|
|
(CEntity **)m_SoloEnts, MAX_CLIENTS, CGameWorld::ENTTYPE_CHARACTER);
|
2014-05-08 13:42:20 +00:00
|
|
|
mem_copy(TempEnts, m_SoloEnts, sizeof(TempEnts));
|
|
|
|
|
2011-12-25 13:51:04 +00:00
|
|
|
int Id = -1;
|
|
|
|
int MinLen = 0;
|
2014-05-07 22:53:21 +00:00
|
|
|
CCharacter *Temp;
|
2020-09-26 19:41:58 +00:00
|
|
|
for(int i = 0; i < Num; i++)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2014-05-07 22:53:21 +00:00
|
|
|
Temp = m_SoloEnts[i];
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Temp->Team() != m_CaughtTeam)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
|
|
|
m_SoloEnts[i] = 0;
|
2011-12-25 13:51:04 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2020-09-26 19:41:58 +00:00
|
|
|
if(m_Layer == LAYER_SWITCH && m_Number && !GameServer()->Collision()->m_pSwitchers[m_Number].m_Status[Temp->Team()])
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
|
|
|
m_SoloEnts[i] = 0;
|
2011-12-25 13:51:04 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2011-12-25 13:51:04 +00:00
|
|
|
int Res =
|
2020-11-04 22:09:22 +00:00
|
|
|
m_NW ?
|
|
|
|
GameServer()->Collision()->IntersectNoLaserNW(m_Pos, Temp->m_Pos, 0, 0) :
|
|
|
|
GameServer()->Collision()->IntersectNoLaser(m_Pos, Temp->m_Pos, 0, 0);
|
2010-07-29 05:21:18 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Res == 0)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2014-05-07 22:53:21 +00:00
|
|
|
int Len = length(Temp->m_Pos - m_Pos);
|
2020-09-26 19:41:58 +00:00
|
|
|
if(MinLen == 0 || MinLen > Len)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2011-12-25 13:51:04 +00:00
|
|
|
MinLen = Len;
|
|
|
|
Id = i;
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(!Temp->Teams()->m_Core.GetSolo(Temp->GetPlayer()->GetCID()))
|
2014-05-07 22:53:21 +00:00
|
|
|
m_SoloEnts[i] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_SoloEnts[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(!m_Target)
|
2014-05-08 13:42:20 +00:00
|
|
|
m_Target = Id != -1 ? TempEnts[Id] : 0;
|
2020-11-11 04:41:42 +00:00
|
|
|
|
|
|
|
if(m_Target)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2020-10-26 14:14:07 +00:00
|
|
|
for(auto &SoloEnt : m_SoloEnts)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2020-10-26 14:14:07 +00:00
|
|
|
if(SoloEnt == m_Target)
|
|
|
|
SoloEnt = 0;
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-30 23:45:42 +00:00
|
|
|
void CDragger::Drag()
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-11-04 22:09:22 +00:00
|
|
|
if(!m_Target)
|
|
|
|
return;
|
2010-09-22 00:03:14 +00:00
|
|
|
|
2020-11-04 22:09:22 +00:00
|
|
|
CCharacter *Target = m_Target;
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-11-04 22:09:22 +00:00
|
|
|
for(int i = -1; i < MAX_CLIENTS; i++)
|
|
|
|
{
|
|
|
|
if(i >= 0)
|
|
|
|
Target = m_SoloEnts[i];
|
|
|
|
|
|
|
|
if(!Target)
|
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-11-04 22:09:22 +00:00
|
|
|
int Res = 0;
|
|
|
|
if(!m_NW)
|
|
|
|
Res = GameServer()->Collision()->IntersectNoLaser(m_Pos,
|
|
|
|
Target->m_Pos, 0, 0);
|
|
|
|
else
|
|
|
|
Res = GameServer()->Collision()->IntersectNoLaserNW(m_Pos,
|
|
|
|
Target->m_Pos, 0, 0);
|
|
|
|
if(Res || length(m_Pos - Target->m_Pos) > g_Config.m_SvDraggerRange)
|
|
|
|
{
|
|
|
|
Target = 0;
|
|
|
|
if(i == -1)
|
|
|
|
m_Target = 0;
|
2014-05-07 22:53:21 +00:00
|
|
|
else
|
2020-11-04 22:09:22 +00:00
|
|
|
m_SoloEnts[i] = 0;
|
|
|
|
}
|
|
|
|
else if(length(m_Pos - Target->m_Pos) > 28)
|
|
|
|
{
|
|
|
|
vec2 Temp = Target->Core()->m_Vel + (normalize(m_Pos - Target->m_Pos) * m_Strength);
|
|
|
|
Target->Core()->m_Vel = ClampVel(Target->m_MoveRestrictions, Temp);
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-21 17:32:42 +00:00
|
|
|
void CDragger::Reset()
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2021-05-22 18:02:00 +00:00
|
|
|
m_MarkedForDestroy = true;
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
|
|
|
|
2010-08-21 17:32:42 +00:00
|
|
|
void CDragger::Tick()
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(((CGameControllerDDRace *)GameServer()->m_pController)->m_Teams.GetTeamState(m_CaughtTeam) == CGameTeams::TEAMSTATE_EMPTY)
|
2011-12-25 13:00:47 +00:00
|
|
|
return;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Server()->Tick() % int(Server()->TickSpeed() * 0.15f) == 0)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2010-11-01 01:51:17 +00:00
|
|
|
int Flags;
|
2011-12-25 13:51:04 +00:00
|
|
|
m_EvalTick = Server()->Tick();
|
|
|
|
int index = GameServer()->Collision()->IsMover(m_Pos.x, m_Pos.y,
|
2020-09-26 19:41:58 +00:00
|
|
|
&Flags);
|
|
|
|
if(index)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2011-12-25 13:51:04 +00:00
|
|
|
m_Core = GameServer()->Collision()->CpSpeed(index, Flags);
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
2011-12-25 13:51:04 +00:00
|
|
|
m_Pos += m_Core;
|
2010-08-30 23:45:42 +00:00
|
|
|
Move();
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
2010-08-30 23:45:42 +00:00
|
|
|
Drag();
|
2010-07-29 05:21:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-30 23:45:42 +00:00
|
|
|
void CDragger::Snap(int SnappingClient)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(((CGameControllerDDRace *)GameServer()->m_pController)->m_Teams.GetTeamState(m_CaughtTeam) == CGameTeams::TEAMSTATE_EMPTY)
|
2011-12-25 13:00:47 +00:00
|
|
|
return;
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2021-10-14 20:59:39 +00:00
|
|
|
int SnappingClientVersion = SnappingClient >= 0 ? GameServer()->GetClientVersion(SnappingClient) : CLIENT_VERSIONNR;
|
|
|
|
|
2014-05-08 13:42:20 +00:00
|
|
|
CCharacter *Target = m_Target;
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-10-26 14:14:07 +00:00
|
|
|
for(int &SoloID : m_SoloIDs)
|
2010-07-29 05:21:18 +00:00
|
|
|
{
|
2020-10-26 14:14:07 +00:00
|
|
|
if(SoloID == -1)
|
2014-05-07 22:53:21 +00:00
|
|
|
break;
|
2010-07-29 05:21:18 +00:00
|
|
|
|
2020-10-26 14:14:07 +00:00
|
|
|
Server()->SnapFreeID(SoloID);
|
|
|
|
SoloID = -1;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2014-01-13 16:00:49 +00:00
|
|
|
|
2014-05-07 22:53:21 +00:00
|
|
|
int pos = 0;
|
2014-01-13 16:00:49 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
for(int i = -1; i < MAX_CLIENTS; i++)
|
2011-08-27 10:58:09 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(i >= 0)
|
2014-05-08 13:42:20 +00:00
|
|
|
{
|
2014-05-07 22:53:21 +00:00
|
|
|
Target = m_SoloEnts[i];
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(!Target)
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Target)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(NetworkClipped(SnappingClient, m_Pos) && NetworkClipped(SnappingClient, Target->m_Pos))
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2020-09-26 19:41:58 +00:00
|
|
|
else if(NetworkClipped(SnappingClient, m_Pos))
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
CCharacter *Char = GameServer()->GetPlayerChar(SnappingClient);
|
2014-05-07 22:53:21 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(SnappingClient > -1 && (GameServer()->m_apPlayers[SnappingClient]->GetTeam() == -1 || GameServer()->m_apPlayers[SnappingClient]->IsPaused()) && GameServer()->m_apPlayers[SnappingClient]->m_SpectatorID != SPEC_FREEVIEW)
|
2014-05-07 22:53:21 +00:00
|
|
|
Char = GameServer()->GetPlayerChar(GameServer()->m_apPlayers[SnappingClient]->m_SpectatorID);
|
|
|
|
|
2021-10-14 20:59:39 +00:00
|
|
|
if(i != -1 || SnappingClientVersion < VERSION_DDNET_SWITCH)
|
|
|
|
{
|
|
|
|
int Tick = (Server()->Tick() % Server()->TickSpeed()) % 11;
|
|
|
|
if(Char && Char->IsAlive() && (m_Layer == LAYER_SWITCH && m_Number && !GameServer()->Collision()->m_pSwitchers[m_Number].m_Status[Char->Team()] && (!Tick)))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Char && Char->IsAlive())
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Char->Team() != m_CaughtTeam)
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// send to spectators only active draggers and some inactive from team 0
|
2020-09-26 19:41:58 +00:00
|
|
|
if(!((Target && Target->IsAlive()) || m_CaughtTeam == 0))
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2010-09-08 16:22:11 +00:00
|
|
|
|
2020-12-19 08:27:10 +00:00
|
|
|
if(Char && Char->IsAlive() && Target && Target->IsAlive() && Target->GetPlayer()->GetCID() != Char->GetPlayer()->GetCID() && ((Char->GetPlayer()->m_ShowOthers == 0 && (Char->Teams()->m_Core.GetSolo(SnappingClient) || Char->Teams()->m_Core.GetSolo(Target->GetPlayer()->GetCID()))) || (Char->GetPlayer()->m_ShowOthers == 2 && !Target->SameTeam(SnappingClient))))
|
2014-06-15 18:51:30 +00:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-05-07 22:53:21 +00:00
|
|
|
CNetObj_Laser *obj;
|
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(i == -1)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
|
|
|
obj = static_cast<CNetObj_Laser *>(Server()->SnapNewItem(
|
2021-01-09 20:40:17 +00:00
|
|
|
NETOBJTYPE_LASER, GetID(), sizeof(CNetObj_Laser)));
|
2021-10-20 23:02:44 +00:00
|
|
|
|
|
|
|
CNetObj_EntityEx *pEntData = static_cast<CNetObj_EntityEx *>(Server()->SnapNewItem(NETOBJTYPE_ENTITYEX, GetID(), sizeof(CNetObj_EntityEx)));
|
|
|
|
if(!pEntData)
|
|
|
|
return;
|
|
|
|
|
|
|
|
pEntData->m_SwitchNumber = m_Number;
|
|
|
|
pEntData->m_Layer = m_Layer;
|
|
|
|
pEntData->m_EntityClass = clamp(ENTITYCLASS_DRAGGER_WEAK + round_to_int(m_Strength) - 1, (int)ENTITYCLASS_DRAGGER_WEAK, (int)ENTITYCLASS_DRAGGER_STRONG);
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_SoloIDs[pos] = Server()->SnapNewID();
|
|
|
|
obj = static_cast<CNetObj_Laser *>(Server()->SnapNewItem( // TODO: Have to free IDs again?
|
2020-09-26 19:41:58 +00:00
|
|
|
NETOBJTYPE_LASER, m_SoloIDs[pos], sizeof(CNetObj_Laser)));
|
2014-05-07 22:53:21 +00:00
|
|
|
pos++;
|
|
|
|
}
|
2010-07-29 05:21:18 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
if(!obj)
|
2014-05-08 13:42:20 +00:00
|
|
|
continue;
|
2017-03-21 10:24:44 +00:00
|
|
|
obj->m_X = (int)m_Pos.x;
|
|
|
|
obj->m_Y = (int)m_Pos.y;
|
2020-09-26 19:41:58 +00:00
|
|
|
if(Target)
|
2014-05-07 22:53:21 +00:00
|
|
|
{
|
2017-03-21 10:24:44 +00:00
|
|
|
obj->m_FromX = (int)Target->m_Pos.x;
|
|
|
|
obj->m_FromY = (int)Target->m_Pos.y;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-21 10:24:44 +00:00
|
|
|
obj->m_FromX = (int)m_Pos.x;
|
|
|
|
obj->m_FromY = (int)m_Pos.y;
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
|
|
|
|
2021-10-14 20:59:39 +00:00
|
|
|
if(i != -1 || SnappingClientVersion < VERSION_DDNET_SWITCH)
|
|
|
|
{
|
|
|
|
int StartTick = m_EvalTick;
|
|
|
|
if(StartTick < Server()->Tick() - 4)
|
|
|
|
StartTick = Server()->Tick() - 4;
|
|
|
|
else if(StartTick > Server()->Tick())
|
|
|
|
StartTick = Server()->Tick();
|
|
|
|
obj->m_StartTick = StartTick;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
obj->m_StartTick = 0;
|
|
|
|
}
|
2014-05-07 22:53:21 +00:00
|
|
|
}
|
2010-07-29 05:21:18 +00:00
|
|
|
}
|
2010-09-22 10:43:59 +00:00
|
|
|
|
2011-12-25 13:51:04 +00:00
|
|
|
CDraggerTeam::CDraggerTeam(CGameWorld *pGameWorld, vec2 Pos, float Strength,
|
2020-09-26 19:41:58 +00:00
|
|
|
bool NW, int Layer, int Number)
|
2011-12-25 13:51:04 +00:00
|
|
|
{
|
2020-09-26 19:41:58 +00:00
|
|
|
for(int i = 0; i < MAX_CLIENTS; ++i)
|
2011-12-25 13:51:04 +00:00
|
|
|
{
|
2019-04-08 19:54:12 +00:00
|
|
|
m_Draggers[i] = new CDragger(pGameWorld, Pos, Strength, NW, i, Layer, Number);
|
2010-09-22 10:43:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-17 10:58:53 +00:00
|
|
|
//CDraggerTeam::~CDraggerTeam()
|
|
|
|
//{
|
|
|
|
// for (int i = 0; i < MAX_CLIENTS; ++i)
|
|
|
|
// {
|
|
|
|
// delete m_Draggers[i];
|
|
|
|
// }
|
|
|
|
//}
|