789 lines
16 KiB
C++
Raw Normal View History

2015-06-30 15:46:07 +06:00
#include "precompiled.h"
short s_iBeamSprite = 0;
2015-06-30 15:46:07 +06:00
constexpr int COS_TABLE_SIZE = 256;
float cosTable[COS_TABLE_SIZE];
2015-06-30 15:46:07 +06:00
bool UTIL_IsNameTaken(const char *name, bool ignoreHumans)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
for (int i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->IsPlayer() && pPlayer->IsBot())
2015-06-30 15:46:07 +06:00
{
// bots can have prefixes so we need to check the name
// against the profile name instead.
2017-10-12 21:50:56 +07:00
CBot *bot = static_cast<CBot *>(pPlayer);
2015-06-30 15:46:07 +06:00
if (FStrEq(name, bot->GetProfile()->GetName()))
{
return true;
}
}
else
{
if (!ignoreHumans)
{
2017-10-12 21:50:56 +07:00
if (FStrEq(name, STRING(pPlayer->pev->netname)))
2015-06-30 15:46:07 +06:00
return true;
}
}
}
2015-06-30 15:46:07 +06:00
return false;
}
int UTIL_ClientsInGame()
2015-06-30 15:46:07 +06:00
{
int iCount = 0;
2017-10-12 21:50:56 +07:00
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
2015-06-30 15:46:07 +06:00
{
CBaseEntity *pPlayer = UTIL_PlayerByIndex(iIndex);
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
if (FStrEq(STRING(pPlayer->pev->netname), ""))
continue;
2017-10-12 21:50:56 +07:00
iCount++;
2015-06-30 15:46:07 +06:00
}
2015-06-30 15:46:07 +06:00
return iCount;
}
int UTIL_ActivePlayersInGame()
2015-06-30 15:46:07 +06:00
{
int iCount = 0;
2017-10-12 21:50:56 +07:00
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
// ignore spectators
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam != TERRORIST && pPlayer->m_iTeam != CT)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iJoiningState != JOINED)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
iCount++;
2015-06-30 15:46:07 +06:00
}
2015-08-20 16:35:01 +06:00
2015-06-30 15:46:07 +06:00
return iCount;
}
2015-08-20 16:35:01 +06:00
int UTIL_HumansInGame(bool ignoreSpectators)
2015-06-30 15:46:07 +06:00
{
int iCount = 0;
2017-10-12 21:50:56 +07:00
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
2019-06-13 00:07:17 +07:00
#ifdef REGAMEDLL_FIXES
if (ignoreSpectators && pPlayer->IsProxy())
2019-06-13 00:07:17 +07:00
continue;
#endif
2017-10-12 21:50:56 +07:00
if (pPlayer->IsBot())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (ignoreSpectators && pPlayer->m_iTeam != TERRORIST && pPlayer->m_iTeam != CT)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (ignoreSpectators && pPlayer->m_iJoiningState != JOINED)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
iCount++;
2015-06-30 15:46:07 +06:00
}
2015-08-20 16:35:01 +06:00
2015-06-30 15:46:07 +06:00
return iCount;
}
// Returns the number of human spectators in the game
int UTIL_SpectatorsInGame()
{
int iCount = 0;
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
if (!UTIL_IsValidPlayer(pPlayer))
continue;
if (FStrEq(STRING(pPlayer->pev->netname), ""))
continue;
if (pPlayer->IsProxy())
continue;
if (pPlayer->IsBot())
continue;
if (pPlayer->m_iTeam != SPECTATOR)
continue;
iCount++;
}
return iCount;
}
int UTIL_HumansOnTeam(int teamID, bool isAlive)
2015-06-30 15:46:07 +06:00
{
int iCount = 0;
2017-10-12 21:50:56 +07:00
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->IsBot())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam != teamID)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (isAlive && !pPlayer->IsAlive())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
iCount++;
2015-06-30 15:46:07 +06:00
}
2015-06-30 15:46:07 +06:00
return iCount;
}
int UTIL_BotsInGame()
2015-06-30 15:46:07 +06:00
{
int iCount = 0;
2017-10-12 21:50:56 +07:00
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
2015-06-30 15:46:07 +06:00
{
2016-05-31 20:04:51 +06:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
if (FStrEq(STRING(pPlayer->pev->netname), ""))
continue;
if (!pPlayer->IsBot())
continue;
2017-10-12 21:50:56 +07:00
iCount++;
2015-06-30 15:46:07 +06:00
}
2015-06-30 15:46:07 +06:00
return iCount;
}
2015-08-20 16:35:01 +06:00
bool UTIL_KickBotFromTeam(TeamName kickTeam)
2015-06-30 15:46:07 +06:00
{
int i;
2015-08-20 16:35:01 +06:00
// try to kick a dead bot first
2017-10-12 21:50:56 +07:00
for (i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-08-20 16:35:01 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
const char *name = STRING(pPlayer->pev->netname);
2015-06-30 15:46:07 +06:00
if (FStrEq(name, ""))
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsBot())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsAlive() && pPlayer->m_iTeam == kickTeam)
2015-06-30 15:46:07 +06:00
{
2015-08-20 16:35:01 +06:00
// its a bot on the right team - kick it
2017-10-12 21:50:56 +07:00
SERVER_COMMAND(UTIL_VarArgs("kick \"%s\"\n", STRING(pPlayer->pev->netname)));
2015-06-30 15:46:07 +06:00
return true;
}
}
2015-08-20 16:35:01 +06:00
// no dead bots, kick any bot on the given team
2017-10-12 21:50:56 +07:00
for (i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
const char *name = STRING(pPlayer->pev->netname);
2015-06-30 15:46:07 +06:00
if (FStrEq(name, ""))
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsBot())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam == kickTeam)
2015-06-30 15:46:07 +06:00
{
2015-08-20 16:35:01 +06:00
// its a bot on the right team - kick it
2017-10-12 21:50:56 +07:00
SERVER_COMMAND(UTIL_VarArgs("kick \"%s\"\n", STRING(pPlayer->pev->netname)));
2015-06-30 15:46:07 +06:00
return true;
}
}
2015-06-30 15:46:07 +06:00
return false;
}
bool UTIL_IsTeamAllBots(int team)
2015-06-30 15:46:07 +06:00
{
int botCount = 0;
2017-10-12 21:50:56 +07:00
for (int i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam != team)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (!(pPlayer->pev->flags & FL_FAKECLIENT))
2015-06-30 15:46:07 +06:00
return false;
2017-10-12 21:50:56 +07:00
botCount++;
2015-06-30 15:46:07 +06:00
}
return botCount ? true : false;
2015-06-30 15:46:07 +06:00
}
// Return the closest active player to the given position.
// If 'distance' is non-NULL, the distance to the closest player is returned in it.
extern CBasePlayer *UTIL_GetClosestPlayer(const Vector *pos, float *distance)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *closePlayer = nullptr;
float closeDistSq = 1.0e12f; // 999999999999.9f
2015-06-30 15:46:07 +06:00
2017-10-12 21:50:56 +07:00
for (int i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
2017-10-12 21:50:56 +07:00
if (!IsEntityValid(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsAlive())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
float distSq = (pPlayer->pev->origin - *pos).LengthSquared();
2015-06-30 15:46:07 +06:00
if (distSq < closeDistSq)
{
closeDistSq = distSq;
2017-10-12 21:50:56 +07:00
closePlayer = pPlayer;
2015-06-30 15:46:07 +06:00
}
}
if (distance)
*distance = Q_sqrt(closeDistSq);
2015-06-30 15:46:07 +06:00
return closePlayer;
}
// Return the closest active player on the given team to the given position.
// If 'distance' is non-NULL, the distance to the closest player is returned in it.
extern CBasePlayer *UTIL_GetClosestPlayer(const Vector *pos, int team, float *distance)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *closePlayer = nullptr;
2015-12-09 04:39:54 +06:00
float closeDistSq = 1.0e12f; // 999999999999.9f
2015-06-30 15:46:07 +06:00
2017-10-12 21:50:56 +07:00
for (int i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
2017-10-12 21:50:56 +07:00
if (!IsEntityValid(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsAlive())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam != team)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
float distSq = (pPlayer->pev->origin - *pos).LengthSquared();
2015-06-30 15:46:07 +06:00
if (distSq < closeDistSq)
{
closeDistSq = distSq;
2017-10-12 21:50:56 +07:00
closePlayer = pPlayer;
2015-06-30 15:46:07 +06:00
}
}
if (distance)
*distance = Q_sqrt(closeDistSq);
2015-06-30 15:46:07 +06:00
return closePlayer;
}
const char *UTIL_GetBotPrefix()
2015-06-30 15:46:07 +06:00
{
return cv_bot_prefix.string;
}
void UTIL_ConstructBotNetName(char *name, int nameLength, const BotProfile *profile)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
if (!profile)
2015-06-30 15:46:07 +06:00
{
name[0] = '\0';
2015-06-30 15:46:07 +06:00
return;
}
// if there is no bot prefix just use the profile name.
2017-10-12 21:50:56 +07:00
if (!UTIL_GetBotPrefix() || Q_strlen(UTIL_GetBotPrefix()) == 0)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
Q_strncpy(name, profile->GetName(), nameLength - 1);
name[nameLength - 1] = '\0';
2015-06-30 15:46:07 +06:00
return;
}
2015-06-30 15:46:07 +06:00
Q_snprintf(name, nameLength, "%s %s", UTIL_GetBotPrefix(), profile->GetName());
}
bool UTIL_IsVisibleToTeam(const Vector &spot, int team, float maxRange)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
for (int i = 1; i <= gpGlobals->maxClients; i++)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
CBasePlayer *pPlayer = UTIL_PlayerByIndex(i);
2015-06-30 15:46:07 +06:00
if (!UTIL_IsValidPlayer(pPlayer))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (FStrEq(STRING(pPlayer->pev->netname), ""))
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (!pPlayer->IsAlive())
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (pPlayer->m_iTeam != team)
2015-06-30 15:46:07 +06:00
continue;
2017-10-12 21:50:56 +07:00
if (maxRange > 0.0f && (spot - pPlayer->Center()).IsLengthGreaterThan(maxRange))
2015-06-30 15:46:07 +06:00
continue;
TraceResult result;
2017-10-12 21:50:56 +07:00
UTIL_TraceLine(pPlayer->EyePosition(), spot, ignore_monsters, ignore_glass, ENT(pPlayer->pev), &result);
2015-06-30 15:46:07 +06:00
if (result.flFraction == 1.0f)
return true;
}
2015-06-30 15:46:07 +06:00
return false;
}
2017-11-23 00:27:55 +07:00
// Return the local player
CBasePlayer *UTIL_GetLocalPlayer()
2015-06-30 15:46:07 +06:00
{
2017-11-23 00:27:55 +07:00
// no "local player" if this is a dedicated server or a single player game
if (IS_DEDICATED_SERVER())
{
#ifdef _DEBUG
// just try to find any player
for (int iIndex = 1; iIndex <= gpGlobals->maxClients; iIndex++)
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex(iIndex);
if (!UTIL_IsValidPlayer(pPlayer))
2017-11-23 00:27:55 +07:00
continue;
if (FStrEq(STRING(pPlayer->pev->netname), ""))
continue;
if (pPlayer->IsBot())
continue;
if (pPlayer->m_iTeam != TERRORIST && pPlayer->m_iTeam != CT)
continue;
if (pPlayer->m_iJoiningState != JOINED)
continue;
return pPlayer;
}
#endif
return nullptr;
}
2015-06-30 15:46:07 +06:00
2017-11-23 00:27:55 +07:00
return UTIL_PlayerByIndex(1);
2015-06-30 15:46:07 +06:00
}
NOXREF Vector UTIL_ComputeOrigin(entvars_t *pevVars)
2015-06-30 15:46:07 +06:00
{
if (pevVars->origin.x == 0.0f && pevVars->origin.y == 0.0f && pevVars->origin.z == 0.0f)
return (pevVars->absmax + pevVars->absmin) * 0.5f;
2015-06-30 15:46:07 +06:00
else
return pevVars->origin;
}
NOXREF Vector UTIL_ComputeOrigin(CBaseEntity *pEntity)
2015-06-30 15:46:07 +06:00
{
return UTIL_ComputeOrigin(pEntity->pev);
}
NOXREF Vector UTIL_ComputeOrigin(edict_t *pentEdict)
2015-06-30 15:46:07 +06:00
{
return UTIL_ComputeOrigin(VARS(pentEdict));
}
NOXREF void UTIL_DrawBeamFromEnt(int iIndex, Vector vecEnd, int iLifetime, byte bRed, byte bGreen, byte bBlue)
2015-06-30 15:46:07 +06:00
{
MESSAGE_BEGIN(MSG_PVS, SVC_TEMPENTITY, vecEnd);
WRITE_BYTE(TE_BEAMENTPOINT);
WRITE_SHORT(iIndex);
WRITE_COORD(vecEnd.x);
WRITE_COORD(vecEnd.y);
WRITE_COORD(vecEnd.z);
WRITE_SHORT(s_iBeamSprite);
WRITE_BYTE(0);
WRITE_BYTE(0);
WRITE_BYTE(iLifetime);
WRITE_BYTE(10);
WRITE_BYTE(0);
WRITE_BYTE(bRed);
WRITE_BYTE(bGreen);
WRITE_BYTE(bBlue);
WRITE_BYTE(255);
WRITE_BYTE(0);
MESSAGE_END();
}
void UTIL_DrawBeamPoints(Vector vecStart, Vector vecEnd, int iLifetime, byte bRed, byte bGreen, byte bBlue)
2015-06-30 15:46:07 +06:00
{
MESSAGE_BEGIN(MSG_PVS, SVC_TEMPENTITY, vecStart);
WRITE_BYTE(TE_BEAMPOINTS);
WRITE_COORD(vecStart.x);
WRITE_COORD(vecStart.y);
WRITE_COORD(vecStart.z);
WRITE_COORD(vecEnd.x);
WRITE_COORD(vecEnd.y);
WRITE_COORD(vecEnd.z);
WRITE_SHORT(s_iBeamSprite);
WRITE_BYTE(0);
WRITE_BYTE(0);
WRITE_BYTE(iLifetime);
WRITE_BYTE(10);
WRITE_BYTE(0);
WRITE_BYTE(bRed);
WRITE_BYTE(bGreen);
WRITE_BYTE(bBlue);
WRITE_BYTE(255);
WRITE_BYTE(0);
MESSAGE_END();
}
void UTIL_DrawBox(Extent *extent, int lifetime, int red, int green, int blue)
{
Vector v[8];
v[0].x = extent->lo.x; v[0].y = extent->lo.y; v[0].z = extent->lo.z;
v[1].x = extent->hi.x; v[1].y = extent->lo.y; v[1].z = extent->lo.z;
v[2].x = extent->hi.x; v[2].y = extent->hi.y; v[2].z = extent->lo.z;
v[3].x = extent->lo.x; v[3].y = extent->hi.y; v[3].z = extent->lo.z;
v[4].x = extent->lo.x; v[4].y = extent->lo.y; v[4].z = extent->hi.z;
v[5].x = extent->hi.x; v[5].y = extent->lo.y; v[5].z = extent->hi.z;
v[6].x = extent->hi.x; v[6].y = extent->hi.y; v[6].z = extent->hi.z;
v[7].x = extent->lo.x; v[7].y = extent->hi.y; v[7].z = extent->hi.z;
static int edge[] =
{
1, 2, 3, 4, -1,
5, 6, 7, 8, -5,
1, -5,
2, -6,
3, -7,
4, -8,
0 // end iterator
};
Vector from, to;
bool restart = true;
for (int i = 0; edge[i] != 0; i++)
{
if (restart)
{
to = v[edge[i] - 1];
restart = false;
continue;
}
from = to;
int index = edge[i];
if (index < 0)
{
restart = true;
index = -index;
}
to = v[index - 1];
UTIL_DrawBeamPoints(from, to, lifetime, red, green, blue);
UTIL_DrawBeamPoints(to, from, lifetime, red, green, blue);
}
}
void CONSOLE_ECHO(const char *pszMsg, ...)
2015-06-30 15:46:07 +06:00
{
va_list argptr;
static char szStr[1024];
va_start(argptr, pszMsg);
Q_vsnprintf(szStr, sizeof(szStr), pszMsg, argptr);
2015-06-30 15:46:07 +06:00
va_end(argptr);
2015-08-20 16:35:01 +06:00
SERVER_PRINT(szStr);
2015-06-30 15:46:07 +06:00
}
void CONSOLE_ECHO_LOGGED(const char *pszMsg, ...)
2015-06-30 15:46:07 +06:00
{
va_list argptr;
static char szStr[1024];
va_start(argptr, pszMsg);
Q_vsnprintf(szStr, sizeof(szStr), pszMsg, argptr);
2015-06-30 15:46:07 +06:00
va_end(argptr);
2015-08-20 16:35:01 +06:00
SERVER_PRINT(szStr);
2015-06-30 15:46:07 +06:00
UTIL_LogPrintf(szStr);
}
void BotPrecache()
2015-06-30 15:46:07 +06:00
{
#ifdef REGAMEDLL_FIXES
// all resources above are used between navarea, improved bots and tutor
// you can run cs1.6 with bots so it's not only limited to czero
if (!AreRunningCZero() && !AreBotsAllowed())
return;
#endif
2015-06-30 15:46:07 +06:00
s_iBeamSprite = PRECACHE_MODEL("sprites/smoke.spr");
PRECACHE_SOUND("buttons/bell1.wav");
PRECACHE_SOUND("buttons/blip1.wav");
PRECACHE_SOUND("buttons/blip2.wav");
PRECACHE_SOUND("buttons/button11.wav");
PRECACHE_SOUND("buttons/latchunlocked2.wav");
PRECACHE_SOUND("buttons/lightswitch2.wav");
#ifdef REGAMEDLL_FIXES
PRECACHE_GENERIC("sound/ambience/quail1.wav");
#else
PRECACHE_SOUND("ambience/quail1.wav"); // not used internally
#endif
2015-06-30 15:46:07 +06:00
PRECACHE_SOUND("events/tutor_msg.wav");
PRECACHE_SOUND("events/enemy_died.wav");
PRECACHE_SOUND("events/friend_died.wav");
PRECACHE_SOUND("events/task_complete.wav");
}
void InitBotTrig()
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
for (int i = 0; i < COS_TABLE_SIZE; i++)
2015-06-30 15:46:07 +06:00
{
2017-11-23 00:27:55 +07:00
real_t angle = 2.0f * M_PI * float(i) / float(COS_TABLE_SIZE - 1);
cosTable[i] = Q_cos(angle);
2015-06-30 15:46:07 +06:00
}
}
float BotCOS(float angle)
{
angle = NormalizeAnglePositive(angle);
int i = angle * ((COS_TABLE_SIZE - 1) / 360.0f);
2017-10-12 21:50:56 +07:00
return cosTable[i];
2015-06-30 15:46:07 +06:00
}
float BotSIN(float angle)
{
angle = NormalizeAnglePositive(angle - 90);
int i = angle * ((COS_TABLE_SIZE - 1) / 360.0f);
2017-10-12 21:50:56 +07:00
return cosTable[i];
2015-06-30 15:46:07 +06:00
}
// Determine if this event is audible, and if so, return its audible range and priority
2017-11-23 00:27:55 +07:00
bool IsGameEventAudible(GameEventType event, CBaseEntity *pEntity, CBaseEntity *pOther, float *range, PriorityType *priority, bool *isHostile)
2015-06-30 15:46:07 +06:00
{
2017-11-23 00:27:55 +07:00
CBasePlayer *pPlayer = static_cast<CBasePlayer *>(pEntity);
2017-11-23 00:27:55 +07:00
if (!pEntity || !pPlayer->IsPlayer())
2017-10-12 21:50:56 +07:00
pPlayer = nullptr;
2015-06-30 15:46:07 +06:00
const float ShortRange = 1000.0f;
const float NormalRange = 2000.0f;
2015-09-17 02:19:21 +06:00
switch (event)
2015-06-30 15:46:07 +06:00
{
// TODO: Check weapon type (knives are pretty quiet)
// TODO: Use actual volume, account for silencers, etc.
case EVENT_WEAPON_FIRED:
{
2017-10-12 21:50:56 +07:00
if (!pPlayer->m_pActiveItem)
return false;
2015-06-30 15:46:07 +06:00
2017-10-12 21:50:56 +07:00
switch (pPlayer->m_pActiveItem->m_iId)
{
// silent "firing"
case WEAPON_HEGRENADE:
case WEAPON_SMOKEGRENADE:
case WEAPON_FLASHBANG:
case WEAPON_SHIELDGUN:
case WEAPON_C4:
return false;
// quiet
case WEAPON_KNIFE:
case WEAPON_TMP:
*range = ShortRange;
break;
// M4A1 - check for silencer
case WEAPON_M4A1:
{
2017-10-12 21:50:56 +07:00
CBasePlayerWeapon *pWeapon = static_cast<CBasePlayerWeapon *>(pPlayer->m_pActiveItem);
if (pWeapon->m_iWeaponState & WPNSTATE_M4A1_SILENCED)
*range = ShortRange;
else
*range = NormalRange;
break;
}
// USP - check for silencer
case WEAPON_USP:
{
2017-10-12 21:50:56 +07:00
CBasePlayerWeapon *pWeapon = static_cast<CBasePlayerWeapon *>(pPlayer->m_pActiveItem);
if (pWeapon->m_iWeaponState & WPNSTATE_USP_SILENCED)
*range = ShortRange;
else
*range = NormalRange;
break;
2015-06-30 15:46:07 +06:00
}
// loud
case WEAPON_AWP:
2015-06-30 15:46:07 +06:00
*range = 99999.0f;
break;
// normal
default:
*range = NormalRange;
break;
}
2015-06-30 15:46:07 +06:00
*priority = PRIORITY_HIGH;
*isHostile = true;
return true;
2015-06-30 15:46:07 +06:00
}
case EVENT_HE_GRENADE_EXPLODED:
*range = 99999.0f;
*priority = PRIORITY_HIGH;
*isHostile = true;
return true;
case EVENT_FLASHBANG_GRENADE_EXPLODED:
*range = 1000.0f;
*priority = PRIORITY_LOW;
*isHostile = true;
return true;
case EVENT_SMOKE_GRENADE_EXPLODED:
*range = 1000.0f;
*priority = PRIORITY_LOW;
*isHostile = true;
return true;
case EVENT_GRENADE_BOUNCED:
*range = 500.0f;
*priority = PRIORITY_LOW;
*isHostile = true;
return true;
case EVENT_BREAK_GLASS:
case EVENT_BREAK_WOOD:
case EVENT_BREAK_METAL:
case EVENT_BREAK_FLESH:
case EVENT_BREAK_CONCRETE:
*range = 1100.0f;
*priority = PRIORITY_MEDIUM;
*isHostile = true;
return true;
case EVENT_DOOR:
*range = 1100.0f;
*priority = PRIORITY_MEDIUM;
*isHostile = false;
return true;
case EVENT_WEAPON_FIRED_ON_EMPTY:
case EVENT_PLAYER_FOOTSTEP:
case EVENT_WEAPON_RELOADED:
case EVENT_WEAPON_ZOOMED:
case EVENT_PLAYER_LANDED_FROM_HEIGHT:
*range = 1100.0f;
*priority = PRIORITY_LOW;
*isHostile = false;
return true;
case EVENT_HOSTAGE_USED:
case EVENT_HOSTAGE_CALLED_FOR_HELP:
*range = 1200.0f;
*priority = PRIORITY_MEDIUM;
*isHostile = false;
return true;
}
2015-06-30 15:46:07 +06:00
return false;
}
void HintMessageToAllPlayers(const char *message)
2015-06-30 15:46:07 +06:00
{
hudtextparms_t textParms;
2015-08-20 16:35:01 +06:00
textParms.x = -1.0f;
textParms.y = -1.0f;
textParms.effect = 0;
2015-06-30 15:46:07 +06:00
textParms.r1 = 100;
2015-08-20 16:35:01 +06:00
textParms.g1 = 255;
2015-06-30 15:46:07 +06:00
textParms.b1 = 100;
2015-08-20 16:35:01 +06:00
textParms.r2 = 255;
textParms.g2 = 255;
textParms.b2 = 255;
2015-08-20 16:35:01 +06:00
textParms.fadeinTime = 1.0f;
textParms.fadeoutTime = 5.0f;
textParms.holdTime = 5.0f;
textParms.fxTime = 0.0f;
2015-06-30 15:46:07 +06:00
textParms.channel = 0;
UTIL_HudMessageAll(textParms, message);
}