#include "precompiled.h" // Return the number of bots following the given player /* <2e85c8> ../cstrike/dlls/bot/cs_bot.cpp:24 */ 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(entity); if (!player->IsBot()) continue; if (!player->IsAlive()) continue; CCSBot *bot = dynamic_cast(player); if (bot && bot->GetFollowLeader() == leader) ++count; } return count; } // Change movement speed to walking /* <2e86df> ../cstrike/dlls/bot/cs_bot.cpp:62 */ void CCSBot::__MAKE_VHOOK(Walk)(void) { 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. /* <2e8732> ../cstrike/dlls/bot/cs_bot.cpp:80 */ bool CCSBot::__MAKE_VHOOK(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 /* <2e8e39> ../cstrike/dlls/bot/cs_bot.cpp:97 */ int CCSBot::__MAKE_VHOOK(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(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 = reinterpret_cast(attacker); m_attackedTimestamp = gpGlobals->time; // no longer safe AdjustSafeTime(); if (!IsSurprised() && (m_attacker != lastAttacker || m_attackedTimestamp != lastAttackedTimestamp)) { CBasePlayer *enemy = static_cast(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 /* <2e8769> ../cstrike/dlls/bot/cs_bot.cpp:171 */ void CCSBot::__MAKE_VHOOK(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 /* <2e7c11> ../cstrike/dlls/bot/cs_bot.cpp:200 */ 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. /* <2e87d9> ../cstrike/dlls/bot/cs_bot.cpp:253 */ 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 we are defusing a bomb, don't move if (IsDefusingBomb()) return; CBasePlayer *player = static_cast(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(static_cast(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(); } } } /* <2e89e3> ../cstrike/dlls/bot/cs_bot.cpp:335 */ bool CCSBot::IsBusy(void) const { if (IsAttacking() || IsBuying() || IsDefusingBomb() || GetTask() == PLANT_BOMB || GetTask() == RESCUE_HOSTAGES || IsSniping()) { return true; } return false; } /* <2e8a0c> ../cstrike/dlls/bot/cs_bot.cpp:351 */ void CCSBot::BotDeathThink(void) { ; } /* <2e8a34> ../cstrike/dlls/bot/cs_bot.cpp:358 */ CBasePlayer *CCSBot::FindNearbyPlayer(void) { 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(pEntity); } return NULL; } // Assign given player as our current enemy to attack /* <2e8abd> ../cstrike/dlls/bot/cs_bot.cpp:385 */ 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. /* <2e8af6> ../cstrike/dlls/bot/cs_bot.cpp:400 */ bool CCSBot::StayOnNavMesh(void) { 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; } /* <2e8c56> ../cstrike/dlls/bot/cs_bot.cpp:450 */ 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_precision 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_precision 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"); } /* <2e9047> ../cstrike/dlls/bot/cs_bot.cpp:527 */ bool CCSBot::IsDoingScenario(void) 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) /* <2e9070> ../cstrike/dlls/bot/cs_bot.cpp:544 */ bool CCSBot::NoticeLooseBomb(void) 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 /* <2e90d4> ../cstrike/dlls/bot/cs_bot.cpp:566 */ bool CCSBot::CanSeeLooseBomb(void) 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 /* <2e9140> ../cstrike/dlls/bot/cs_bot.cpp:588 */ bool CCSBot::CanSeePlantedBomb(void) 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(bombPos, CHECK_FOV)) return true; return false; } // Return last enemy that hurt us /* <2e918e> ../cstrike/dlls/bot/cs_bot.cpp:610 */ CBasePlayer *CCSBot::GetAttacker(void) const { if (m_attacker != NULL && m_attacker->IsAlive()) return m_attacker; return NULL; } // Immediately jump off of our ladder, if we're on one /* <2e91b7> ../cstrike/dlls/bot/cs_bot.cpp:622 */ void CCSBot::GetOffLadder(void) { if (IsUsingLadder()) { Jump(MUST_JUMP); DestroyPath(); } } // Return time when given spot was last checked /* <2e91f1> ../cstrike/dlls/bot/cs_bot.cpp:637 */ 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. /* <2e9240> ../cstrike/dlls/bot/cs_bot.cpp:651 */ 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; } } // Periodic check of hostage count in case we lost some /* <2e92b8> ../cstrike/dlls/bot/cs_bot.cpp:693 */ void CCSBot::UpdateHostageEscortCount(void) { const float updateInterval = 1.0f; if (m_hostageEscortCount == 0 || gpGlobals->time - m_hostageEscortCountTimestamp < updateInterval) return; m_hostageEscortCountTimestamp = gpGlobals->time; // recount the hostages in case we lost some m_hostageEscortCount = 0; CHostage *hostage = NULL; while ((hostage = static_cast(UTIL_FindEntityByClassname(hostage, "hostage_entity"))) != NULL) { if (FNullEnt(hostage->edict())) break; // skip dead or rescued hostages if (!hostage->IsValid()) continue; // check if hostage has targeted us, and is following if (hostage->IsFollowing(this)) ++m_hostageEscortCount; } } // Return true if we are outnumbered by enemies /* <2e940d> ../cstrike/dlls/bot/cs_bot.cpp:722 */ bool CCSBot::IsOutnumbered(void) const { return (GetNearbyFriendCount() < GetNearbyEnemyCount() - 1) ? true : false; } // Return number of enemies we are outnumbered by /* <2e94a0> ../cstrike/dlls/bot/cs_bot.cpp:731 */ int CCSBot::OutnumberedCount(void) const { if (IsOutnumbered()) { return (GetNearbyEnemyCount() - 1) - GetNearbyFriendCount(); } return 0; } // Return the closest "important" enemy for the given scenario (bomb carrier, VIP, hostage escorter) /* <2e95cb> ../cstrike/dlls/bot/cs_bot.cpp:744 */ 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(entity); // is it alive? if (!player->IsAlive()) continue; // skip friends if (player->m_iTeam == m_iTeam) continue; // is it "important" if (!ctrl->IsImportantPlayer(player)) continue; // 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 /* <2e9719> ../cstrike/dlls/bot/cs_bot.cpp:801 */ void CCSBot::SetDisposition(DispositionType disposition) { m_disposition = disposition; if (m_disposition != IGNORE_ENEMIES) { m_ignoreEnemiesTimer.Invalidate(); } } // Return our current disposition /* <2e9762> ../cstrike/dlls/bot/cs_bot.cpp:814 */ CCSBot::DispositionType CCSBot::GetDisposition(void) const { if (!m_ignoreEnemiesTimer.IsElapsed()) return IGNORE_ENEMIES; return m_disposition; } // Ignore enemies for a short durationy /* <2e979b> ../cstrike/dlls/bot/cs_bot.cpp:826 */ void CCSBot::IgnoreEnemies(float duration) { m_ignoreEnemiesTimer.Start(duration); } // Increase morale one step /* <2e97fc> ../cstrike/dlls/bot/cs_bot.cpp:835 */ void CCSBot::IncreaseMorale(void) { if (m_morale < EXCELLENT) { m_morale = static_cast(m_morale + 1); } } // Decrease morale one step /* <2e9824> ../cstrike/dlls/bot/cs_bot.cpp:845 */ void CCSBot::DecreaseMorale(void) { if (m_morale > TERRIBLE) { m_morale = static_cast(m_morale - 1); } } // Return true if we are acting like a rogue (not listening to teammates, not doing scenario goals) // TODO: Account for morale /* <2e984c> ../cstrike/dlls/bot/cs_bot.cpp:857 */ bool CCSBot::IsRogue(void) 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 /* <2e98f1> ../cstrike/dlls/bot/cs_bot.cpp:882 */ bool CCSBot::IsHurrying(void) const { if (!m_hurryTimer.IsElapsed()) return true; CCSBotManager *ctrl = TheCSBots(); // if the bomb has been planted, we are in a hurry, CT or T (they could be defusing it!) if (ctrl->GetScenario() == CCSBotManager::SCENARIO_DEFUSE_BOMB && ctrl->IsBombPlanted()) return true; // if we are a T and hostages are being rescued, we are in a hurry if (ctrl->GetScenario() == CCSBotManager::SCENARIO_RESCUE_HOSTAGES && m_iTeam == TERRORIST && GetGameState()->AreAllHostagesBeingRescued()) return true; return false; } // Return true if it is the early, "safe", part of the round /* <2e9942> ../cstrike/dlls/bot/cs_bot.cpp:906 */ bool CCSBot::IsSafe(void) 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 /* <2e9987> ../cstrike/dlls/bot/cs_bot.cpp:920 */ bool CCSBot::IsWellPastSafe(void) 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 /* <2e99d8> ../cstrike/dlls/bot/cs_bot.cpp:934 */ bool CCSBot::IsEndOfSafeTime(void) const { return m_wasSafe && !IsSafe(); } // Return the amount of "safe time" we have left /* <2e9a3e> ../cstrike/dlls/bot/cs_bot.cpp:943 */ float CCSBot::GetSafeTimeRemaining(void) const { CCSBotManager *ctrl = TheCSBots(); return m_safeTime - ctrl->GetElapsedRoundTime(); } // Called when enemy seen to adjust safe time for this round /* <2e9a8f> ../cstrike/dlls/bot/cs_bot.cpp:954 */ void CCSBot::AdjustSafeTime(void) { 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" /* <2e9ad0> ../cstrike/dlls/bot/cs_bot.cpp:970 */ bool CCSBot::HasNotSeenEnemyForLongTime(void) const { const float longTime = 30.0f; return (GetTimeSinceLastSawEnemy() > longTime); } // Pick a random zone and hide near it /* <2e9b26> ../cstrike/dlls/bot/cs_bot.cpp:980 */ 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. /* <2e9c1f> ../cstrike/dlls/bot/cs_bot.cpp:1066 */ 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. /* <2eaa1d> ../cstrike/dlls/bot/cs_bot.cpp:1116 */ float CCSBot::GetRangeToFarthestEscortedHostage(void) const { FarthestHostage away(this); g_pHostages->ForEachHostage(away); return away.m_farRange; } #ifdef HOOK_GAMEDLL void CCSBot::Walk(void) { Walk_(); } bool CCSBot::Jump(bool mustJump) { return Jump_(mustJump); } int CCSBot::TakeDamage(entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType) { return TakeDamage_(pevInflictor, pevAttacker, flDamage, bitsDamageType); } void CCSBot::Killed(entvars_t *pevAttacker, int iGib) { Killed_(pevAttacker, iGib); } #endif // HOOK_GAMEDLL