|
|
|
#include "bot_common.h"
|
|
|
|
|
|
|
|
// Return the number of bots following the given player
|
|
|
|
|
|
|
|
int GetBotFollowCount(CBasePlayer *leader)
|
|
|
|
{
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (int i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
CBaseEntity *entity = UTIL_PlayerByIndex(i);
|
|
|
|
|
|
|
|
if (entity == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (FNullEnt(entity->pev))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (FStrEq(STRING(entity->pev->netname), ""))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CBasePlayer *player = static_cast<CBasePlayer *>(entity);
|
|
|
|
|
|
|
|
if (!player->IsBot())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!player->IsAlive())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CCSBot *bot = dynamic_cast<CCSBot *>(player);
|
|
|
|
if (bot != NULL && bot->GetFollowLeader() == leader)
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change movement speed to walking
|
|
|
|
|
|
|
|
void CCSBot::Walk()
|
|
|
|
{
|
|
|
|
if (m_mustRunTimer.IsElapsed())
|
|
|
|
{
|
|
|
|
CBot::Walk();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// must run
|
|
|
|
Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if jump was started.
|
|
|
|
// This is extended from the base jump to disallow jumping when in a crouch area.
|
|
|
|
|
|
|
|
bool CCSBot::Jump(bool mustJump)
|
|
|
|
{
|
|
|
|
// prevent jumping if we're crouched, unless we're in a crouchjump area - jump wins
|
|
|
|
bool inCrouchJumpArea = (m_lastKnownArea &&
|
|
|
|
(m_lastKnownArea->GetAttributes() & NAV_CROUCH) &&
|
|
|
|
!(m_lastKnownArea->GetAttributes() & NAV_JUMP));
|
|
|
|
|
|
|
|
if (inCrouchJumpArea)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CBot::Jump(mustJump);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Invoked when injured by something
|
|
|
|
// NOTE: We dont want to directly call Attack() here, or the bots will have super-human reaction times when injured
|
|
|
|
|
|
|
|
int CCSBot::TakeDamage(entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType)
|
|
|
|
{
|
|
|
|
CBaseEntity *attacker = GetClassPtr((CBaseEntity *)pevInflictor);
|
|
|
|
|
|
|
|
// if we were attacked by a teammate, rebuke
|
|
|
|
if (attacker->IsPlayer())
|
|
|
|
{
|
|
|
|
CBasePlayer *player = static_cast<CBasePlayer *>(attacker);
|
|
|
|
|
|
|
|
/*if (player->m_iTeam == m_iTeam && !player->IsBot())
|
|
|
|
{
|
|
|
|
GetChatter()->FriendlyFire();
|
|
|
|
}*/
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attacker->IsPlayer() && IsEnemy(attacker))
|
|
|
|
{
|
|
|
|
// Track previous attacker so we don't try to panic multiple times for a shotgun blast
|
|
|
|
CBasePlayer *lastAttacker = m_attacker;
|
|
|
|
float lastAttackedTimestamp = m_attackedTimestamp;
|
|
|
|
|
|
|
|
// keep track of our last attacker
|
|
|
|
m_attacker = static_cast<CBasePlayer *>(attacker);
|
|
|
|
m_attackedTimestamp = gpGlobals->time;
|
|
|
|
|
|
|
|
// no longer safe
|
|
|
|
AdjustSafeTime();
|
|
|
|
|
|
|
|
if (!IsSurprised() && (m_attacker != lastAttacker || m_attackedTimestamp != lastAttackedTimestamp))
|
|
|
|
{
|
|
|
|
CBasePlayer *enemy = static_cast<CBasePlayer *>(attacker);
|
|
|
|
|
|
|
|
// being hurt by an enemy we can't see causes panic
|
|
|
|
if (!IsVisible(enemy, CHECK_FOV))
|
|
|
|
{
|
|
|
|
bool panic = false;
|
|
|
|
|
|
|
|
// if not attacking anything, look around to try to find attacker
|
|
|
|
if (!IsAttacking())
|
|
|
|
{
|
|
|
|
panic = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// we are attacking
|
|
|
|
if (!IsEnemyVisible())
|
|
|
|
{
|
|
|
|
// can't see our current enemy, panic to acquire new attacker
|
|
|
|
panic = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!panic)
|
|
|
|
{
|
|
|
|
float invSkill = 1.0f - GetProfile()->GetSkill();
|
|
|
|
float panicChance = invSkill * invSkill * 50.0f;
|
|
|
|
|
|
|
|
if (panicChance > RANDOM_FLOAT(0, 100))
|
|
|
|
{
|
|
|
|
panic = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (panic != false)
|
|
|
|
{
|
|
|
|
// can't see our current enemy, panic to acquire new attacker
|
|
|
|
Panic(m_attacker);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// extend
|
|
|
|
return CBasePlayer::TakeDamage(pevInflictor, pevAttacker, flDamage, bitsDamageType);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Invoked when killed
|
|
|
|
|
|
|
|
void CCSBot::Killed(entvars_t *pevAttacker, int iGib)
|
|
|
|
{
|
|
|
|
PrintIfWatched("Killed( attacker = %s )\n", STRING(pevAttacker->netname));
|
|
|
|
|
|
|
|
GetChatter()->OnDeath();
|
|
|
|
|
|
|
|
// increase the danger where we died
|
|
|
|
const float deathDanger = 1.0f;
|
|
|
|
const float deathDangerRadius = 500.0f;
|
|
|
|
IncreaseDangerNearby(m_iTeam - 1, deathDanger, m_lastKnownArea, &pev->origin, deathDangerRadius);
|
|
|
|
|
|
|
|
// end voice feedback
|
|
|
|
// EndVoiceFeedback();
|
|
|
|
|
|
|
|
// extend
|
|
|
|
CBasePlayer::Killed(pevAttacker, iGib);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if line segment intersects rectagular volume
|
|
|
|
|
|
|
|
bool IsIntersectingBox(const Vector *start, const Vector *end, const Vector *boxMin, const Vector *boxMax)
|
|
|
|
{
|
|
|
|
unsigned char startFlags = 0;
|
|
|
|
unsigned char endFlags = 0;
|
|
|
|
|
|
|
|
// classify start point
|
|
|
|
if (start->x < boxMin->x)
|
|
|
|
startFlags |= LO_X;
|
|
|
|
if (start->x > boxMax->x)
|
|
|
|
startFlags |= HI_X;
|
|
|
|
|
|
|
|
if (start->y < boxMin->y)
|
|
|
|
startFlags |= LO_Y;
|
|
|
|
if (start->y > boxMax->y)
|
|
|
|
startFlags |= HI_Y;
|
|
|
|
|
|
|
|
if (start->z < boxMin->z)
|
|
|
|
startFlags |= LO_Z;
|
|
|
|
if (start->z > boxMax->z)
|
|
|
|
startFlags |= HI_Z;
|
|
|
|
|
|
|
|
// classify end point
|
|
|
|
if (end->x < boxMin->x)
|
|
|
|
endFlags |= LO_X;
|
|
|
|
if (end->x > boxMax->x)
|
|
|
|
endFlags |= HI_X;
|
|
|
|
|
|
|
|
if (end->y < boxMin->y)
|
|
|
|
endFlags |= LO_Y;
|
|
|
|
if (end->y > boxMax->y)
|
|
|
|
endFlags |= HI_Y;
|
|
|
|
|
|
|
|
if (end->z < boxMin->z)
|
|
|
|
endFlags |= LO_Z;
|
|
|
|
if (end->z > boxMax->z)
|
|
|
|
endFlags |= HI_Z;
|
|
|
|
|
|
|
|
// trivial reject
|
|
|
|
if (startFlags & endFlags)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// TODO: Do exact line/box intersection check
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// When bot is touched by another entity.
|
|
|
|
|
|
|
|
void CCSBot::BotTouch(CBaseEntity *other)
|
|
|
|
{
|
|
|
|
// if we have touched a higher-priority player, make way
|
|
|
|
// TODO: Need to account for reaction time, etc.
|
|
|
|
if (other->IsPlayer())
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
// if we are defusing a bomb, don't move
|
|
|
|
if (IsDefusingBomb())
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
CBasePlayer *player = static_cast<CBasePlayer *>(other);
|
|
|
|
|
|
|
|
// get priority of other player
|
|
|
|
unsigned int otherPri = TheCSBots()->GetPlayerPriority(player);
|
|
|
|
|
|
|
|
// get our priority
|
|
|
|
unsigned int myPri = TheCSBots()->GetPlayerPriority(this);
|
|
|
|
|
|
|
|
// if our priority is better, don't budge
|
|
|
|
if (myPri < otherPri)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// they are higher priority - make way, unless we're already making way for someone more important
|
|
|
|
if (m_avoid != NULL)
|
|
|
|
{
|
|
|
|
unsigned int avoidPri = TheCSBots()->GetPlayerPriority(static_cast<CBasePlayer *>(static_cast<CBaseEntity *>(m_avoid)));
|
|
|
|
if (avoidPri < otherPri)
|
|
|
|
{
|
|
|
|
// ignore 'other' because we're already avoiding someone better
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_avoid = other;
|
|
|
|
m_avoidTimestamp = gpGlobals->time;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we won't be able to break it, don't try
|
|
|
|
if (other->pev->takedamage != DAMAGE_YES)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (IsAttacking())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// See if it's breakable
|
|
|
|
if (FClassnameIs(other->pev, "func_breakable"))
|
|
|
|
{
|
|
|
|
Vector center = (other->pev->absmax + other->pev->absmin) / 2.0f;
|
|
|
|
bool breakIt = true;
|
|
|
|
|
|
|
|
if (m_pathLength)
|
|
|
|
{
|
|
|
|
Vector goal = m_goalPosition + Vector(0, 0, HalfHumanHeight);
|
|
|
|
breakIt = IsIntersectingBox(&pev->origin, &goal, &other->pev->absmin, &other->pev->absmax);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (breakIt)
|
|
|
|
{
|
|
|
|
// it's breakable - try to shoot it.
|
|
|
|
SetLookAt("Breakable", ¢er, PRIORITY_HIGH, 0.2, 0, 5.0);
|
|
|
|
|
|
|
|
if (IsUsingGrenade())
|
|
|
|
{
|
|
|
|
EquipBestWeapon(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PrimaryAttack();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCSBot::IsBusy() const
|
|
|
|
{
|
|
|
|
if (IsAttacking() ||
|
|
|
|
#if 0
|
|
|
|
IsBuying() ||
|
|
|
|
IsDefusingBomb() ||
|
|
|
|
GetTask() == PLANT_BOMB ||
|
|
|
|
GetTask() == RESCUE_HOSTAGES ||
|
|
|
|
#endif
|
|
|
|
IsSniping())
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCSBot::BotDeathThink()
|
|
|
|
{
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
CBasePlayer *CCSBot::FindNearbyPlayer()
|
|
|
|
{
|
|
|
|
CBaseEntity *pEntity = NULL;
|
|
|
|
Vector vecSrc = pev->origin;
|
|
|
|
const float flRadius = 800.0f;
|
|
|
|
|
|
|
|
while ((pEntity = UTIL_FindEntityInSphere(pEntity, vecSrc, flRadius)) != NULL)
|
|
|
|
{
|
|
|
|
if (!pEntity->IsPlayer())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!(pEntity->pev->flags & FL_FAKECLIENT))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return static_cast<CBasePlayer *>(pEntity);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assign given player as our current enemy to attack
|
|
|
|
|
|
|
|
void CCSBot::SetEnemy(CBasePlayer *enemy)
|
|
|
|
{
|
|
|
|
if (m_enemy != enemy)
|
|
|
|
{
|
|
|
|
m_enemy = enemy;
|
|
|
|
m_currentEnemyAcquireTimestamp = gpGlobals->time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are not on the navigation mesh (m_currentArea == NULL),
|
|
|
|
// move towards last known area.
|
|
|
|
// Return false if off mesh.
|
|
|
|
|
|
|
|
bool CCSBot::StayOnNavMesh()
|
|
|
|
{
|
|
|
|
if (m_currentArea != NULL)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// move back onto the area map
|
|
|
|
|
|
|
|
// if we have no lastKnownArea, we probably started off
|
|
|
|
// of the nav mesh - find the closest nav area and use it
|
|
|
|
CNavArea *goalArea;
|
|
|
|
if (!m_currentArea && !m_lastKnownArea)
|
|
|
|
{
|
|
|
|
goalArea = TheNavAreaGrid.GetNearestNavArea(&pev->origin);
|
|
|
|
PrintIfWatched("Started off the nav mesh - moving to closest nav area...\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
goalArea = m_lastKnownArea;
|
|
|
|
PrintIfWatched("Getting out of NULL area...\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (goalArea != NULL)
|
|
|
|
{
|
|
|
|
Vector pos;
|
|
|
|
goalArea->GetClosestPointOnArea(&pev->origin, &pos);
|
|
|
|
|
|
|
|
// move point into area
|
|
|
|
Vector to = pos - pev->origin;
|
|
|
|
to.NormalizeInPlace();
|
|
|
|
|
|
|
|
// how far to "step into" an area - must be less than min area size
|
|
|
|
const float stepInDist = 5.0f;
|
|
|
|
pos = pos + (stepInDist * to);
|
|
|
|
|
|
|
|
MoveTowardsPosition(&pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we're stuck, try to get un-stuck
|
|
|
|
// do stuck movements last, so they override normal movement
|
|
|
|
if (m_isStuck)
|
|
|
|
{
|
|
|
|
Wiggle();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCSBot::Panic(CBasePlayer *enemy)
|
|
|
|
{
|
|
|
|
if (IsSurprised())
|
|
|
|
return;
|
|
|
|
|
|
|
|
Vector2D dir(BotCOS(pev->v_angle.y), BotSIN(pev->v_angle.y));
|
|
|
|
Vector2D perp(-dir.y, dir.x);
|
|
|
|
Vector spot;
|
|
|
|
|
|
|
|
if (GetProfile()->GetSkill() >= 0.5f)
|
|
|
|
{
|
|
|
|
Vector2D toEnemy = (enemy->pev->origin - pev->origin).Make2D();
|
|
|
|
toEnemy.NormalizeInPlace();
|
|
|
|
|
|
|
|
float along = DotProduct(toEnemy, dir);
|
|
|
|
|
|
|
|
float c45 = 0.7071f;
|
|
|
|
float size = 100.0f;
|
|
|
|
|
|
|
|
float shift = RANDOM_FLOAT(-75.0, 75.0);
|
|
|
|
|
|
|
|
if (along > c45)
|
|
|
|
{
|
|
|
|
spot.x = pev->origin.x + dir.x * size + perp.x * shift;
|
|
|
|
spot.y = pev->origin.y + dir.y * size + perp.y * shift;
|
|
|
|
}
|
|
|
|
else if (along < -c45)
|
|
|
|
{
|
|
|
|
spot.x = pev->origin.x - dir.x * size + perp.x * shift;
|
|
|
|
spot.y = pev->origin.y - dir.y * size + perp.y * shift;
|
|
|
|
}
|
|
|
|
else if (DotProduct(toEnemy, perp) > 0.0)
|
|
|
|
{
|
|
|
|
spot.x = pev->origin.x + perp.x * size + dir.x * shift;
|
|
|
|
spot.y = pev->origin.y + perp.y * size + dir.y * shift;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
spot.x = pev->origin.x - perp.x * size + dir.x * shift;
|
|
|
|
spot.y = pev->origin.y - perp.y * size + dir.y * shift;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const float offset = 200.0f;
|
|
|
|
float side = RANDOM_FLOAT(-offset, offset) * 2.0f;
|
|
|
|
|
|
|
|
spot.x = pev->origin.x - dir.x * offset + perp.x * side;
|
|
|
|
spot.y = pev->origin.y - dir.y * offset + perp.y * side;
|
|
|
|
}
|
|
|
|
|
|
|
|
spot.z = pev->origin.z + RANDOM_FLOAT(-50.0, 50.0);
|
|
|
|
|
|
|
|
// we are stunned for a moment
|
|
|
|
m_surpriseDelay = RANDOM_FLOAT(0.1, 0.2);
|
|
|
|
m_surpriseTimestamp = gpGlobals->time;
|
|
|
|
|
|
|
|
SetLookAt("Panic", &spot, PRIORITY_HIGH, 0, 0, 5.0);
|
|
|
|
PrintIfWatched("Aaaah!\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCSBot::IsDoingScenario() const
|
|
|
|
{
|
|
|
|
if (cv_bot_defer_to_human.value <= 0.0f)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return !UTIL_HumansOnTeam(m_iTeam, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if we noticed the bomb on the ground or on the radar (for T's only)
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
bool CCSBot::NoticeLooseBomb() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (ctrl->GetScenario() != CCSBotManager::SCENARIO_DEFUSE_BOMB)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
CBaseEntity *bomb = ctrl->GetLooseBomb();
|
|
|
|
|
|
|
|
if (bomb != NULL)
|
|
|
|
{
|
|
|
|
// T's can always see bomb on their radar
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if can see the bomb lying on the ground
|
|
|
|
|
|
|
|
bool CCSBot::CanSeeLooseBomb() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (ctrl->GetScenario() != CCSBotManager::SCENARIO_DEFUSE_BOMB)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
CBaseEntity *bomb = ctrl->GetLooseBomb();
|
|
|
|
|
|
|
|
if (bomb != NULL)
|
|
|
|
{
|
|
|
|
if (IsVisible(&bomb->pev->origin, CHECK_FOV))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if can see the planted bomb
|
|
|
|
|
|
|
|
bool CCSBot::CanSeePlantedBomb() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (ctrl->GetScenario() != CCSBotManager::SCENARIO_DEFUSE_BOMB)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!GetGameState()->IsBombPlanted())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const Vector *bombPos = GetGameState()->GetBombPosition();
|
|
|
|
|
|
|
|
if (bombPos != NULL && IsVisible((Vector*)bombPos, CHECK_FOV))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
// Return last enemy that hurt us
|
|
|
|
|
|
|
|
CBasePlayer *CCSBot::GetAttacker() const
|
|
|
|
{
|
|
|
|
if (m_attacker != NULL && m_attacker->IsAlive())
|
|
|
|
return m_attacker;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Immediately jump off of our ladder, if we're on one
|
|
|
|
|
|
|
|
void CCSBot::GetOffLadder()
|
|
|
|
{
|
|
|
|
if (IsUsingLadder())
|
|
|
|
{
|
|
|
|
Jump(MUST_JUMP);
|
|
|
|
DestroyPath();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return time when given spot was last checked
|
|
|
|
|
|
|
|
float CCSBot::GetHidingSpotCheckTimestamp(HidingSpot *spot) const
|
|
|
|
{
|
|
|
|
for (int i = 0; i < m_checkedHidingSpotCount; ++i)
|
|
|
|
{
|
|
|
|
if (m_checkedHidingSpot[i].spot->GetID() == spot->GetID())
|
|
|
|
return m_checkedHidingSpot[i].timestamp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -999999.9f;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the timestamp of the given spot to now.
|
|
|
|
// If the spot is not in the set, overwrite the least recently checked spot.
|
|
|
|
|
|
|
|
void CCSBot::SetHidingSpotCheckTimestamp(HidingSpot *spot)
|
|
|
|
{
|
|
|
|
int leastRecent = 0;
|
|
|
|
float leastRecentTime = gpGlobals->time + 1.0f;
|
|
|
|
|
|
|
|
for (int i = 0; i < m_checkedHidingSpotCount; ++i)
|
|
|
|
{
|
|
|
|
// if spot is in the set, just update its timestamp
|
|
|
|
if (m_checkedHidingSpot[i].spot->GetID() == spot->GetID())
|
|
|
|
{
|
|
|
|
m_checkedHidingSpot[i].timestamp = gpGlobals->time;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// keep track of least recent spot
|
|
|
|
if (m_checkedHidingSpot[i].timestamp < leastRecentTime)
|
|
|
|
{
|
|
|
|
leastRecentTime = m_checkedHidingSpot[i].timestamp;
|
|
|
|
leastRecent = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is room for more spots, append this one
|
|
|
|
if (m_checkedHidingSpotCount < MAX_CHECKED_SPOTS)
|
|
|
|
{
|
|
|
|
m_checkedHidingSpot[ m_checkedHidingSpotCount ].spot = spot;
|
|
|
|
m_checkedHidingSpot[ m_checkedHidingSpotCount ].timestamp = gpGlobals->time;
|
|
|
|
++m_checkedHidingSpotCount;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// replace the least recent spot
|
|
|
|
m_checkedHidingSpot[ leastRecent ].spot = spot;
|
|
|
|
m_checkedHidingSpot[ leastRecent ].timestamp = gpGlobals->time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
// Periodic check of hostage count in case we lost some
|
|
|
|
|
|
|
|
void CCSBot::UpdateHostageEscortCount()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
// Return true if we are outnumbered by enemies
|
|
|
|
|
|
|
|
bool CCSBot::IsOutnumbered() const
|
|
|
|
{
|
|
|
|
return (GetNearbyFriendCount() < GetNearbyEnemyCount() - 1) ? true : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return number of enemies we are outnumbered by
|
|
|
|
|
|
|
|
int CCSBot::OutnumberedCount() const
|
|
|
|
{
|
|
|
|
if (IsOutnumbered())
|
|
|
|
{
|
|
|
|
return (GetNearbyEnemyCount() - 1) - GetNearbyFriendCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the closest "important" enemy for the given scenario (bomb carrier, VIP, hostage escorter)
|
|
|
|
|
|
|
|
CBasePlayer *CCSBot::GetImportantEnemy(bool checkVisibility) const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
CBasePlayer *nearEnemy = NULL;
|
|
|
|
float nearDist = 999999999.9f;
|
|
|
|
|
|
|
|
for (int i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
CBaseEntity *entity = UTIL_PlayerByIndex(i);
|
|
|
|
|
|
|
|
if (entity == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (FNullEnt(entity->pev))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (FStrEq(STRING(entity->pev->netname), ""))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// is it a player?
|
|
|
|
if (!entity->IsPlayer())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CBasePlayer *player = static_cast<CBasePlayer *>(entity);
|
|
|
|
|
|
|
|
// is it alive?
|
|
|
|
if (!player->IsAlive())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// skip friends
|
|
|
|
|
|
|
|
// is it closest?
|
|
|
|
Vector d = pev->origin - player->pev->origin;
|
|
|
|
|
|
|
|
float distSq = d.x * d.x + d.y * d.y + d.z * d.z;
|
|
|
|
if (distSq < nearDist)
|
|
|
|
{
|
|
|
|
if (checkVisibility && !IsVisible(player, CHECK_FOV))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
nearEnemy = player;
|
|
|
|
nearDist = distSq;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nearEnemy;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sets our current disposition
|
|
|
|
|
|
|
|
void CCSBot::SetDisposition(DispositionType disposition)
|
|
|
|
{
|
|
|
|
m_disposition = disposition;
|
|
|
|
|
|
|
|
if (m_disposition != IGNORE_ENEMIES)
|
|
|
|
{
|
|
|
|
m_ignoreEnemiesTimer.Invalidate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return our current disposition
|
|
|
|
|
|
|
|
CCSBot::DispositionType CCSBot::GetDisposition() const
|
|
|
|
{
|
|
|
|
if (!m_ignoreEnemiesTimer.IsElapsed())
|
|
|
|
return IGNORE_ENEMIES;
|
|
|
|
|
|
|
|
return m_disposition;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore enemies for a short durationy
|
|
|
|
|
|
|
|
void CCSBot::IgnoreEnemies(float duration)
|
|
|
|
{
|
|
|
|
m_ignoreEnemiesTimer.Start(duration);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Increase morale one step
|
|
|
|
|
|
|
|
void CCSBot::IncreaseMorale()
|
|
|
|
{
|
|
|
|
if (m_morale < EXCELLENT)
|
|
|
|
{
|
|
|
|
m_morale = static_cast<MoraleType>(m_morale + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decrease morale one step
|
|
|
|
|
|
|
|
void CCSBot::DecreaseMorale()
|
|
|
|
{
|
|
|
|
if (m_morale > TERRIBLE)
|
|
|
|
{
|
|
|
|
m_morale = static_cast<MoraleType>(m_morale - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if we are acting like a rogue (not listening to teammates, not doing scenario goals)
|
|
|
|
// TODO: Account for morale
|
|
|
|
|
|
|
|
bool CCSBot::IsRogue() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (!ctrl->AllowRogues())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// periodically re-evaluate our rogue status
|
|
|
|
if (m_rogueTimer.IsElapsed())
|
|
|
|
{
|
|
|
|
m_rogueTimer.Start(RANDOM_FLOAT(10, 30));
|
|
|
|
|
|
|
|
// our chance of going rogue is inversely proportional to our teamwork attribute
|
|
|
|
const float rogueChance = 100.0f * (1.0f - GetProfile()->GetTeamwork());
|
|
|
|
|
|
|
|
m_isRogue = (RANDOM_FLOAT(0, 100) < rogueChance);
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_isRogue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if we are in a hurry
|
|
|
|
|
|
|
|
bool CCSBot::IsHurrying() const
|
|
|
|
{
|
|
|
|
if (!m_hurryTimer.IsElapsed())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if it is the early, "safe", part of the round
|
|
|
|
|
|
|
|
bool CCSBot::IsSafe() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (ctrl->GetElapsedRoundTime() < m_safeTime)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if it is well past the early, "safe", part of the round
|
|
|
|
|
|
|
|
bool CCSBot::IsWellPastSafe() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
if (ctrl->GetElapsedRoundTime() > 1.25f * m_safeTime)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if we were in the safe time last update, but not now
|
|
|
|
|
|
|
|
bool CCSBot::IsEndOfSafeTime() const
|
|
|
|
{
|
|
|
|
return m_wasSafe && !IsSafe();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the amount of "safe time" we have left
|
|
|
|
|
|
|
|
float CCSBot::GetSafeTimeRemaining() const
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
return m_safeTime - ctrl->GetElapsedRoundTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Called when enemy seen to adjust safe time for this round
|
|
|
|
|
|
|
|
void CCSBot::AdjustSafeTime()
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
|
|
|
|
// if we spotted an enemy sooner than we thought possible, adjust our notion of "safe" time
|
|
|
|
if (m_safeTime > ctrl->GetElapsedRoundTime())
|
|
|
|
{
|
|
|
|
// since right now is not safe, adjust safe time to be a few seconds ago
|
|
|
|
m_safeTime = ctrl->GetElapsedRoundTime() - 2.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if we haven't seen an enemy for "a long time"
|
|
|
|
|
|
|
|
bool CCSBot::HasNotSeenEnemyForLongTime() const
|
|
|
|
{
|
|
|
|
const float longTime = 30.0f;
|
|
|
|
return (GetTimeSinceLastSawEnemy() > longTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pick a random zone and hide near it
|
|
|
|
|
|
|
|
bool CCSBot::GuardRandomZone(float range)
|
|
|
|
{
|
|
|
|
CCSBotManager *ctrl = TheCSBots();
|
|
|
|
const CCSBotManager::Zone *zone = ctrl->GetRandomZone();
|
|
|
|
|
|
|
|
if (zone != NULL)
|
|
|
|
{
|
|
|
|
CNavArea *rescueArea = ctrl->GetRandomAreaInZone(zone);
|
|
|
|
if (rescueArea != NULL)
|
|
|
|
{
|
|
|
|
Hide(rescueArea, -1.0f, range);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do a breadth-first search to find a good retreat spot.
|
|
|
|
// Don't pick a spot that a Player is currently occupying.
|
|
|
|
|
|
|
|
const Vector *FindNearbyRetreatSpot(CCSBot *me, float maxRange)
|
|
|
|
{
|
|
|
|
CNavArea *area = me->GetLastKnownArea();
|
|
|
|
if (area == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
// collect spots that enemies cannot see
|
|
|
|
CollectRetreatSpotsFunctor collector(me, maxRange);
|
|
|
|
SearchSurroundingAreas(area, &me->pev->origin, collector, maxRange);
|
|
|
|
|
|
|
|
if (collector.m_count == 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
// select a hiding spot at random
|
|
|
|
int which = RANDOM_LONG(0, collector.m_count - 1);
|
|
|
|
return collector.m_spot[ which ];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return euclidean distance to farthest escorted hostage.
|
|
|
|
// Return -1 if no hostage is following us.
|
|
|
|
#if 0
|
|
|
|
float CCSBot::GetRangeToFarthestEscortedHostage() const
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|