4645 lines
121 KiB
C++
Raw Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
// tf_bot.cpp
// Team Fortress NextBot
// Michael Booth, February 2009
#include "cbase.h"
#include "tf_player.h"
#include "tf_gamerules.h"
#include "tf_obj_sentrygun.h"
#include "team_control_point_master.h"
#include "tf_weapon_pipebomblauncher.h"
#include "team_train_watcher.h"
#include "tf_bot.h"
#include "tf_bot_manager.h"
#include "tf_bot_vision.h"
#include "tf_team.h"
#include "bot/map_entities/tf_bot_generator.h"
#include "trigger_area_capture.h"
#include "GameEventListener.h"
#include "NextBotUtil.h"
#include "tier3/tier3.h"
#include "vgui/ILocalize.h"
#include "econ_item_system.h"
#include "bot/behavior/tf_bot_use_item.h"
#include "tf_wearable_item_demoshield.h"
#include "tf_weapon_buff_item.h"
#include "tf_weapon_lunchbox.h"
#include "func_respawnroom.h"
#include "soundenvelope.h"
#include "econ_entity_creation.h"
#include "player_vs_environment/tf_population_manager.h"
#include "bot/behavior/tf_bot_behavior.h"
#include "bot/map_entities/tf_bot_generator.h"
#include "bot/map_entities/tf_bot_hint_entity.h"
ConVar tf_bot_force_class( "tf_bot_force_class", "", FCVAR_GAMEDLL, "If set to a class name, all TFBots will respawn as that class" );
ConVar tf_bot_notice_gunfire_range( "tf_bot_notice_gunfire_range", "3000", FCVAR_GAMEDLL );
ConVar tf_bot_notice_quiet_gunfire_range( "tf_bot_notice_quiet_gunfire_range", "500", FCVAR_GAMEDLL );
ConVar tf_bot_sniper_personal_space_range( "tf_bot_sniper_personal_space_range", "1000", FCVAR_CHEAT, "Enemies beyond this range don't worry the Sniper" );
ConVar tf_bot_pyro_deflect_tolerance( "tf_bot_pyro_deflect_tolerance", "0.5", FCVAR_CHEAT );
ConVar tf_bot_keep_class_after_death( "tf_bot_keep_class_after_death", "0", FCVAR_GAMEDLL );
ConVar tf_bot_prefix_name_with_difficulty( "tf_bot_prefix_name_with_difficulty", "0", FCVAR_GAMEDLL, "Append the skill level of the bot to the bot's name" );
ConVar tf_bot_near_point_travel_distance( "tf_bot_near_point_travel_distance", "750", FCVAR_CHEAT, "If within this travel distance to the current point, bot is 'near' it" );
ConVar tf_bot_pyro_shove_away_range( "tf_bot_pyro_shove_away_range", "250", FCVAR_CHEAT, "If a Pyro bot's target is closer than this, compression blast them away" );
ConVar tf_bot_pyro_always_reflect( "tf_bot_pyro_always_reflect", "0", FCVAR_CHEAT, "Pyro bots will always reflect projectiles fired at them. For tesing/debugging purposes." );
ConVar tf_bot_sniper_spot_min_range( "tf_bot_sniper_spot_min_range", "1000", FCVAR_CHEAT );
ConVar tf_bot_sniper_spot_max_count( "tf_bot_sniper_spot_max_count", "10", FCVAR_CHEAT, "Stop searching for sniper spots when each side has found this many" );
ConVar tf_bot_sniper_spot_search_count( "tf_bot_sniper_spot_search_count", "10", FCVAR_CHEAT, "Search this many times per behavior update frame" );
ConVar tf_bot_sniper_spot_point_tolerance( "tf_bot_sniper_spot_point_tolerance", "750", FCVAR_CHEAT );
ConVar tf_bot_sniper_spot_epsilon( "tf_bot_sniper_spot_epsilon", "100", FCVAR_CHEAT );
ConVar tf_bot_sniper_goal_entity_move_tolerance( "tf_bot_sniper_goal_entity_move_tolerance", "500", FCVAR_CHEAT );
ConVar tf_bot_suspect_spy_touch_interval( "tf_bot_suspect_spy_touch_interval", "5", FCVAR_CHEAT, "How many seconds back to look for touches against suspicious spies" );
ConVar tf_bot_suspect_spy_forget_cooldown( "tf_bot_suspect_spy_forget_cooldown", "5", FCVAR_CHEAT, "How long to consider a suspicious spy as suspicious" );
ConVar tf_bot_debug_tags( "tf_bot_debug_tags", "0", FCVAR_CHEAT, "ent_text will only show tags on bots" );
extern ConVar tf_bot_sniper_spot_max_count;
extern ConVar tf_bot_fire_weapon_min_time;
extern ConVar tf_bot_sniper_misfire_chance;
extern ConVar tf_bot_difficulty;
extern ConVar tf_bot_farthest_visible_theater_sample_count;
extern ConVar tf_bot_sniper_spot_min_range;
extern ConVar tf_bot_sniper_spot_epsilon;
extern ConVar tf_mvm_miniboss_min_health;
extern ConVar tf_bot_path_lookahead_range;
extern ConVar tf_mvm_miniboss_scale;
//-----------------------------------------------------------------------------------------------------
bool IsPlayerClassname( const char *string )
{
for ( int i = TF_CLASS_SCOUT; i < TF_CLASS_COUNT_ALL; ++i )
{
if ( !stricmp( string, GetPlayerClassData( i )->m_szClassName ) )
{
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------------------------------
bool IsTeamName( const char *string )
{
if ( !stricmp( string, "red" ) )
return true;
if ( !stricmp( string, "blue" ) )
return true;
return false;
}
//-----------------------------------------------------------------------------------------------------
CTFBot::DifficultyType StringToDifficultyLevel( const char *string )
{
if ( !stricmp( string, "easy" ) )
return CTFBot::EASY;
if ( !stricmp( string, "normal" ) )
return CTFBot::NORMAL;
if ( !stricmp( string, "hard" ) )
return CTFBot::HARD;
if ( !stricmp( string, "expert" ) )
return CTFBot::EXPERT;
return CTFBot::UNDEFINED;
}
//-----------------------------------------------------------------------------------------------------
const char *DifficultyLevelToString( CTFBot::DifficultyType skill )
{
switch( skill )
{
case CTFBot::EASY: return "Easy ";
case CTFBot::NORMAL: return "Normal ";
case CTFBot::HARD: return "Hard ";
case CTFBot::EXPERT: return "Expert ";
}
return "Undefined ";
}
//-----------------------------------------------------------------------------------------------------
const char *GetRandomBotName( void )
{
static const char *nameList[] =
{
"Chucklenuts",
"CryBaby",
"WITCH",
"ThatGuy",
"Still Alive",
"Hat-Wearing MAN",
"Me",
"Numnutz",
"H@XX0RZ",
"The G-Man",
"Chell",
"The Combine",
"Totally Not A Bot",
"Pow!",
"Zepheniah Mann",
"THEM",
"LOS LOS LOS",
"10001011101",
"DeadHead",
"ZAWMBEEZ",
"MindlessElectrons",
"TAAAAANK!",
"The Freeman",
"Black Mesa",
"Soulless",
"CEDA",
"BeepBeepBoop",
"NotMe",
"CreditToTeam",
"BoomerBile",
"Someone Else",
"Mann Co.",
"Dog",
"Kaboom!",
"AmNot",
"0xDEADBEEF",
"HI THERE",
"SomeDude",
"GLaDOS",
"Hostage",
"Headful of Eyeballs",
"CrySomeMore",
"Aperture Science Prototype XR7",
"Humans Are Weak",
"AimBot",
"C++",
"GutsAndGlory!",
"Nobody",
"Saxton Hale",
"RageQuit",
"Screamin' Eagles",
"Ze Ubermensch",
"Maggot",
"CRITRAWKETS",
"Herr Doktor",
"Gentlemanne of Leisure",
"Companion Cube",
"Target Practice",
"One-Man Cheeseburger Apocalypse",
"Crowbar",
"Delicious Cake",
"IvanTheSpaceBiker",
"I LIVE!",
"Cannon Fodder",
"trigger_hurt",
"Nom Nom Nom",
"Divide by Zero",
"GENTLE MANNE of LEISURE",
"MoreGun",
"Tiny Baby Man",
"Big Mean Muther Hubbard",
"Force of Nature",
"Crazed Gunman",
"Grim Bloody Fable",
"Poopy Joe",
"A Professional With Standards",
"Freakin' Unbelievable",
"SMELLY UNFORTUNATE",
"The Administrator",
"Mentlegen",
"Archimedes!",
"Ribs Grow Back",
"It's Filthy in There!",
"Mega Baboon",
"Kill Me",
"Glorified Toaster with Legs",
#ifdef STAGING_ONLY
"John Spartan",
"Leeloo Dallas Multipass",
"Sho'nuff",
"Bruce Leroy",
"CAN YOUUUUUUUUU DIG IT?!?!?!?!",
"Big Gulp, Huh?",
"Stupid Hot Dog",
"I'm your huckleberry",
"The Crocketeer",
#endif
NULL
};
static int nameCount = 0;
static int nameIndex = 0;
if ( nameCount == 0 )
{
for( ; nameList[ nameCount ]; ++nameCount );
// randomize the initial index
nameIndex = RandomInt( 0, nameCount-1 );
}
const char *name = nameList[ nameIndex++ ];
if ( nameIndex >= nameCount )
nameIndex = 0;
return name;
}
//-----------------------------------------------------------------------------------------------------
void CreateBotName( int iTeam, int iClassIndex, CTFBot::DifficultyType skill, char* pBuffer, int iBufferSize )
{
char szBotNameBuffer[256];
char szEnemyOrFriendlyString[256];
const char *pBotName = "";
const char *pFriendlyOrEnemyTitle = "";
// @note (Tom Bui): it is okay to get localized name in training, since we should be on a listen server
if ( TFGameRules()->IsInTraining() )
{
// get the friendly/enemy title
const char *pBotTitle = NULL;
if ( iTeam != TEAM_UNASSIGNED )
{
int iHumanTeam = TFGameRules()->GetAssignedHumanTeam();
if ( iHumanTeam != TEAM_ANY )
{
if ( iHumanTeam == iTeam )
{
pBotTitle = "#TF_Bot_Title_Friendly";
}
else
{
pBotTitle = "#TF_Bot_Title_Enemy";
}
}
}
wchar_t *pLocalizedTitle = pBotTitle ? g_pVGuiLocalize->Find( pBotTitle ) : NULL;
if ( pLocalizedTitle )
{
g_pVGuiLocalize->ConvertUnicodeToANSI( pLocalizedTitle, szEnemyOrFriendlyString, sizeof( szEnemyOrFriendlyString ) );
pFriendlyOrEnemyTitle = szEnemyOrFriendlyString;
}
// get the class name
wchar_t *pLocalizedName = NULL;
if ( iClassIndex >= TF_FIRST_NORMAL_CLASS && iClassIndex < TF_LAST_NORMAL_CLASS )
{
pLocalizedName = g_pVGuiLocalize->Find( g_aPlayerClassNames[ iClassIndex ] );
}
else
{
pLocalizedName = g_pVGuiLocalize->Find( "#TF_Bot_Generic_ClassName" );
}
g_pVGuiLocalize->ConvertUnicodeToANSI( pLocalizedName, szBotNameBuffer, sizeof( szBotNameBuffer ) );
pBotName = szBotNameBuffer;
}
else
{
pBotName = GetRandomBotName();
}
const char *pDifficultyString = tf_bot_prefix_name_with_difficulty.GetBool() ? DifficultyLevelToString( skill ) : "";
// we use this as our formatting, because we don't know the language of the downstream clients
CFmtStr name( "%s%s%s",
pDifficultyString, pFriendlyOrEnemyTitle, pBotName );
Q_strncpy( pBuffer, name.Access(), iBufferSize );
}
//-----------------------------------------------------------------------------------------------------
CON_COMMAND_F( tf_bot_add, "Add a bot.", FCVAR_GAMEDLL )
{
// Listenserver host or rcon access only!
if ( !UTIL_IsCommandIssuedByServerAdmin() )
return;
bool bQuotaManaged = true;
int botCount = 1;
const char *classname = NULL;
const char *teamname = "auto";
const char *pszBotNameViaArg = NULL;
CTFBot::DifficultyType skill = clamp( (CTFBot::DifficultyType)tf_bot_difficulty.GetInt(), CTFBot::EASY, CTFBot::EXPERT );
int i;
for( i=1; i<args.ArgC(); ++i )
{
CTFBot::DifficultyType trySkill = StringToDifficultyLevel( args.Arg(i) );
int nArgAsInteger = atoi( args.Arg(i) );
// each argument could be a classname, a team, a difficulty level, a count, or a name
if ( IsPlayerClassname( args.Arg(i) ) )
{
classname = args.Arg(i);
}
else if ( IsTeamName( args.Arg(i) ) )
{
teamname = args.Arg(i);
}
else if ( !stricmp( args.Arg( i ), "noquota" ) )
{
bQuotaManaged = false;
}
else if ( trySkill != CTFBot::UNDEFINED )
{
skill = trySkill;
}
else if ( nArgAsInteger > 0 )
{
botCount = nArgAsInteger;
pszBotNameViaArg = NULL; // can't have a custom name if spawning multiple bots
}
else if ( botCount == 1 )
{
pszBotNameViaArg = args.Arg( i );
}
else
{
Warning( "Invalid argument '%s'\n", args.Arg(i) );
}
}
// cvar can override classname
classname = FStrEq( tf_bot_force_class.GetString(), "" ) ? classname : tf_bot_force_class.GetString();
int iClassIndex = classname ? GetClassIndexFromString( classname ) : TF_CLASS_UNDEFINED;
int iTeam = TEAM_UNASSIGNED;
if ( FStrEq( teamname, "red" ) )
{
iTeam = TF_TEAM_RED;
}
else if ( FStrEq( teamname, "blue" ) )
{
iTeam = TF_TEAM_BLUE;
}
if ( TFGameRules()->IsInTraining() )
{
skill = CTFBot::EASY;
}
char name[256];
int iNumAdded = 0;
for( i=0; i<botCount; ++i )
{
CTFBot *pBot = NULL;
const char *pszBotName = NULL;
if ( !pszBotNameViaArg )
{
CreateBotName( iTeam, iClassIndex, skill, name, sizeof(name) );
pszBotName = name;
}
else
{
pszBotName = pszBotNameViaArg;
}
pBot = NextBotCreatePlayerBot< CTFBot >( pszBotName );
if ( pBot )
{
if ( bQuotaManaged )
{
pBot->SetAttribute( CTFBot::QUOTA_MANANGED );
}
pBot->HandleCommand_JoinTeam( teamname );
pBot->SetDifficulty( skill );
// if no class is set, auto-select one
const char *thisClassname = classname ? classname : pBot->GetNextSpawnClassname();
pBot->HandleCommand_JoinClass( thisClassname );
// set up a proper name now that we are in training
if ( TFGameRules()->IsInTraining() )
{
CreateBotName( pBot->GetTeamNumber(), pBot->GetPlayerClass()->GetClassIndex(), skill, name, sizeof(name) );
engine->SetFakeClientConVarValue( pBot->edict(), "name", name );
}
++iNumAdded;
}
}
if ( bQuotaManaged )
{
TheTFBots().OnForceAddedBots( iNumAdded );
}
}
//-----------------------------------------------------------------------------------------------------
CON_COMMAND_F( tf_bot_kick, "Remove a TFBot by name, or all bots (\"all\").", FCVAR_GAMEDLL )
{
// Listenserver host or rcon access only!
if ( !UTIL_IsCommandIssuedByServerAdmin() )
return;
if ( args.ArgC() < 2 )
{
DevMsg( "%s <bot name>, \"red\", \"blue\", or \"all\"> <optional: \"moveToSpectatorTeam\"> \n", args.Arg(0) );
return;
}
bool bMoveToSpectatorTeam = false;
int iTeam = TEAM_UNASSIGNED;
int i;
const char *pPlayerName = "";
for( i=1; i<args.ArgC(); ++i )
{
// each argument could be a classname, a team, or a count
if ( FStrEq( args.Arg(i), "red" ) )
{
iTeam = TF_TEAM_RED;
}
else if ( FStrEq( args.Arg(i), "blue" ) )
{
iTeam = TF_TEAM_BLUE;
}
else if ( FStrEq( args.Arg(i), "all" ) )
{
iTeam = TEAM_ANY;
}
else if ( FStrEq( args.Arg(i), "moveToSpectatorTeam" ) )
{
bMoveToSpectatorTeam = true;
}
else
{
pPlayerName = args.Arg(i);
}
}
int iNumKicked = 0;
for( int i=1; i<=gpGlobals->maxClients; ++i )
{
CBasePlayer *player = static_cast<CBasePlayer *>( UTIL_PlayerByIndex( i ) );
if ( !player )
continue;
if ( FNullEnt( player->edict() ) )
continue;
if ( player->MyNextBotPointer() )
{
if ( iTeam == TEAM_ANY ||
FStrEq( pPlayerName, player->GetPlayerName() ) ||
( player->GetTeamNumber() == iTeam ) ||
( player->GetTeamNumber() == iTeam ) )
{
if ( bMoveToSpectatorTeam )
{
player->ChangeTeam( TEAM_SPECTATOR, false, true );
}
else
{
engine->ServerCommand( UTIL_VarArgs( "kickid %d\n", player->GetUserID() ) );
}
CTFBot* pBot = dynamic_cast< CTFBot* >( player );
if ( pBot && pBot->HasAttribute( CTFBot::QUOTA_MANANGED ) )
{
++iNumKicked;
}
}
}
}
TheTFBots().OnForceKickedBots( iNumKicked );
}
//-----------------------------------------------------------------------------------------------------
CON_COMMAND_F( tf_bot_kill, "Kill a TFBot by name, or all bots (\"all\").", FCVAR_GAMEDLL )
{
// Listenserver host or rcon access only!
if ( !UTIL_IsCommandIssuedByServerAdmin() )
return;
if ( args.ArgC() < 2 )
{
DevMsg( "%s <bot name>, \"red\", \"blue\", or \"all\"> <optional: \"moveToSpectatorTeam\"> \n", args.Arg(0) );
return;
}
int iTeam = TEAM_UNASSIGNED;
int i;
const char *pPlayerName = "";
for( i=1; i<args.ArgC(); ++i )
{
// each argument could be a classname, a team, or a count
if ( FStrEq( args.Arg(i), "red" ) )
{
iTeam = TF_TEAM_RED;
}
else if ( FStrEq( args.Arg(i), "blue" ) )
{
iTeam = TF_TEAM_BLUE;
}
else if ( FStrEq( args.Arg(i), "all" ) )
{
iTeam = TEAM_ANY;
}
else if ( FStrEq( args.Arg(i), "moveToSpectatorTeam" ) )
{
// bMoveToSpectatorTeam = true;
}
else
{
pPlayerName = args.Arg(i);
}
}
for( int i=1; i<=gpGlobals->maxClients; ++i )
{
CBasePlayer *player = static_cast<CBasePlayer *>( UTIL_PlayerByIndex( i ) );
if ( !player )
continue;
if ( FNullEnt( player->edict() ) )
continue;
if ( player->MyNextBotPointer() )
{
if ( iTeam == TEAM_ANY ||
FStrEq( pPlayerName, player->GetPlayerName() ) ||
( player->GetTeamNumber() == iTeam ) ||
( player->GetTeamNumber() == iTeam ) )
{
CTakeDamageInfo info( player, player, 9999999.9f, DMG_ENERGYBEAM, TF_DMG_CUSTOM_NONE );
player->TakeDamage( info );
}
}
}
}
//-----------------------------------------------------------------------------------------------------
void CMD_BotWarpTeamToMe( void )
{
CBasePlayer *player = UTIL_GetListenServerHost();
if ( !player )
return;
CTeam *myTeam = player->GetTeam();
for( int i=0; i<myTeam->GetNumPlayers(); ++i )
{
if ( !myTeam->GetPlayer(i)->IsAlive() )
continue;
myTeam->GetPlayer(i)->SetAbsOrigin( player->GetAbsOrigin() );
}
}
static ConCommand tf_bot_warp_team_to_me( "tf_bot_warp_team_to_me", CMD_BotWarpTeamToMe, "", FCVAR_GAMEDLL | FCVAR_CHEAT );
//-----------------------------------------------------------------------------------------------------
IMPLEMENT_INTENTION_INTERFACE( CTFBot, CTFBotMainAction );
//-----------------------------------------------------------------------------------------------------
LINK_ENTITY_TO_CLASS( tf_bot, CTFBot );
//-----------------------------------------------------------------------------------------------------
/**
* Allocate a bot and bind it to the edict
*/
CBasePlayer *CTFBot::AllocatePlayerEntity( edict_t *edict, const char *playerName )
{
CBasePlayer::s_PlayerEdict = edict;
return static_cast< CBasePlayer * >( CreateEntityByName( "tf_bot" ) );
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::PressFireButton( float duration )
{
// can't fire if stunned
// @todo Tom Bui: Eventually, we'll probably want to check the actual weapon for supress fire
if ( m_Shared.IsControlStunned() || m_Shared.IsLoserStateStunned() || HasAttribute( CTFBot::SUPPRESS_FIRE ) )
{
ReleaseFireButton();
return;
}
BaseClass::PressFireButton( duration );
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::PressAltFireButton( float duration )
{
// can't fire if stunned
// @todo Tom Bui: Eventually, we'll probably want to check the actual weapon for supress fire
if ( m_Shared.IsControlStunned() || m_Shared.IsLoserStateStunned() || HasAttribute( CTFBot::SUPPRESS_FIRE ) )
{
ReleaseAltFireButton();
return;
}
BaseClass::PressAltFireButton( duration );
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::PressSpecialFireButton( float duration )
{
// can't fire if stunned
// @todo Tom Bui: Eventually, we'll probably want to check the actual weapon for supress fire
if ( m_Shared.IsControlStunned() || m_Shared.IsLoserStateStunned() || HasAttribute( CTFBot::SUPPRESS_FIRE ) )
{
ReleaseAltFireButton();
return;
}
BaseClass::PressSpecialFireButton( duration );
}
//-----------------------------------------------------------------------------------------------------
class CCountClassMembers
{
public:
CCountClassMembers( const CTFBot *me, int teamID )
{
m_me = me;
m_myTeam = teamID;
m_teamSize = 0;
for( int i=0; i<TF_LAST_NORMAL_CLASS; ++i )
m_count[i] = 0;
}
bool operator() ( CBasePlayer *basePlayer )
{
CTFPlayer *player = (CTFPlayer *)basePlayer;
if ( player->GetTeamNumber() != m_myTeam )
return true;
++m_teamSize;
if ( m_me->IsSelf( player ) )
return true;
++m_count[ player->GetDesiredPlayerClassIndex() ];
return true;
}
const CTFBot *m_me;
int m_myTeam;
int m_count[ TF_LAST_NORMAL_CLASS+1 ];
int m_teamSize;
};
//-----------------------------------------------------------------------------------------------------
/**
* NOTE: Assumes bot's difficulty has been set, and the bot is on a team.
*/
const char *CTFBot::GetNextSpawnClassname( void ) const
{
struct ClassSelectionInfo
{
int m_class;
int m_minTeamSizeToSelect; // team must have this many members to choose this class
int m_countPerTeamSize; // must have 1 Medic for each 4 team members, for example
int m_minLimit; // minimum that must be present (once other constraints are met)
int m_maxLimit[ NUM_DIFFICULTY_LEVELS ]; // maximum that can be present (-1 for infinite)
};
const int NoLimit = -1;
static ClassSelectionInfo defenseRoster[] =
{
{ TF_CLASS_ENGINEER, 0, 4, 1, { 1, 2, 3, 3 } },
{ TF_CLASS_SOLDIER, 0, 0, 0, { NoLimit, NoLimit, NoLimit, NoLimit } },
{ TF_CLASS_DEMOMAN, 0, 0, 0, { 2, 3, 3, 3 } },
{ TF_CLASS_PYRO, 3, 0, 0, { NoLimit, NoLimit, NoLimit, NoLimit } },
{ TF_CLASS_HEAVYWEAPONS, 3, 0, 0, { 1, 1, 2, 2 } },
{ TF_CLASS_MEDIC, 4, 4, 1, { 1, 1, 2, 2 } },
{ TF_CLASS_SNIPER, 5, 0, 0, { 0, 1, 1, 1 } },
{ TF_CLASS_SPY, 5, 0, 0, { 0, 1, 2, 2 } },
{ TF_CLASS_UNDEFINED, 0, -1 },
};
static ClassSelectionInfo offenseRoster[] =
{
{ TF_CLASS_SCOUT, 0, 0, 1, { 3, 3, 3, 3 } },
{ TF_CLASS_SOLDIER, 0, 0, 0, { NoLimit, NoLimit, NoLimit, NoLimit } },
{ TF_CLASS_DEMOMAN, 0, 0, 0, { 2, 3, 3, 3 } }, // must limit demomen, or the whole team will go demo to take out tough sentryguns
{ TF_CLASS_PYRO, 3, 0, 0, { NoLimit, NoLimit, NoLimit, NoLimit } },
{ TF_CLASS_HEAVYWEAPONS, 3, 0, 0, { 1, 1, 2, 2 } },
{ TF_CLASS_MEDIC, 4, 4, 1, { 1, 1, 2, 2 } },
{ TF_CLASS_SNIPER, 5, 0, 0, { 0, 1, 1, 1 } },
{ TF_CLASS_SPY, 5, 0, 0, { 0, 1, 2, 2 } },
{ TF_CLASS_ENGINEER, 5, 0, 0, { 1, 1, 1, 1 } },
{ TF_CLASS_UNDEFINED, 0, -1 },
};
static ClassSelectionInfo compRoster[] =
{
{ TF_CLASS_SCOUT, 0, 0, 0, { 0, 0, 2, 2 } },
{ TF_CLASS_SOLDIER, 0, 0, 0, { 0, 0, NoLimit, NoLimit } },
{ TF_CLASS_DEMOMAN, 0, 0, 0, { 0, 0, 2, 2 } }, // must limit demomen, or the whole team will go demo to take out tough sentryguns
{ TF_CLASS_PYRO, 0, -1 },
{ TF_CLASS_HEAVYWEAPONS, 3, 0, 0, { 0, 0, 2, 2 } },
{ TF_CLASS_MEDIC, 1, 0, 1, { 0, 0, 1, 1 } },
{ TF_CLASS_SNIPER, 0, -1 },
{ TF_CLASS_SPY, 0, -1 },
{ TF_CLASS_ENGINEER, 0, -1 },
{ TF_CLASS_UNDEFINED, 0, -1 },
};
// if we are an engineer with an active sentry or teleporters, don't switch
if ( IsPlayerClass( TF_CLASS_ENGINEER ) )
{
if ( const_cast< CTFBot * >( this )->GetObjectOfType( OBJ_SENTRYGUN ) ||
const_cast< CTFBot * >( this )->GetObjectOfType( OBJ_TELEPORTER, MODE_TELEPORTER_EXIT ) )
{
return "engineer";
}
}
// count classes in use by my team, not including me
CCountClassMembers currentRoster( this, GetTeamNumber() );
ForEachPlayer( currentRoster );
// assume offense
ClassSelectionInfo *desiredRoster = offenseRoster;
if ( TFGameRules()->IsMatchTypeCompetitive() )
{
desiredRoster = compRoster;
}
else if ( TFGameRules()->IsInKothMode() )
{
CTeamControlPoint *point = GetMyControlPoint();
if ( point )
{
if ( GetTeamNumber() == ObjectiveResource()->GetOwningTeam( point->GetPointIndex() ) )
{
// defend our point
desiredRoster = defenseRoster;
}
}
}
else if ( TFGameRules()->GetGameType() == TF_GAMETYPE_CP )
{
CUtlVector< CTeamControlPoint * > captureVector;
TFGameRules()->CollectCapturePoints( const_cast< CTFBot * >( this ), &captureVector );
CUtlVector< CTeamControlPoint * > defendVector;
TFGameRules()->CollectDefendPoints( const_cast< CTFBot * >( this ), &defendVector );
// if we have any points we can capture, try to do so
if ( captureVector.Count() > 0 || defendVector.Count() == 0 )
{
desiredRoster = offenseRoster;
}
else
{
desiredRoster = defenseRoster;
}
}
else if ( TFGameRules()->GetGameType() == TF_GAMETYPE_ESCORT )
{
if ( GetTeamNumber() == TF_TEAM_RED )
{
desiredRoster = defenseRoster;
}
}
// build vector of classes we can pick from
CUtlVector< int > desiredClassVector;
CUtlVector< int > allowedClassForBotRosterVector;
for( int i=0; desiredRoster[ i ].m_class != TF_CLASS_UNDEFINED; ++i )
{
ClassSelectionInfo *desiredClassInfo = &desiredRoster[ i ];
if ( TFGameRules()->CanBotChooseClass( const_cast< CTFBot * >( this ), desiredClassInfo->m_class ) == false )
{
// not allowed to use this class
continue;
}
// just in case we hit the class limits, we want to make sure we select a class that is allowed
allowedClassForBotRosterVector.AddToTail( desiredClassInfo->m_class );
if ( currentRoster.m_teamSize < desiredClassInfo->m_minTeamSizeToSelect )
{
// team is too small to choose this class
continue;
}
// check limits
if ( currentRoster.m_count[ desiredClassInfo->m_class ] < desiredClassInfo->m_minLimit )
{
// below required limit - choose only this class
desiredClassVector.RemoveAll();
desiredClassVector.AddToTail( desiredClassInfo->m_class );
break;
}
int maxLimit = desiredClassInfo->m_maxLimit[ (int)clamp( GetDifficulty(), CTFBot::EASY, CTFBot::EXPERT ) ];
if ( maxLimit > NoLimit && currentRoster.m_count[ desiredClassInfo->m_class ] >= maxLimit )
{
// at or above limit for this class
continue;
}
if ( desiredClassInfo->m_countPerTeamSize > 0 )
{
// how many of this class should there be at the given "per" count
int maxCountPer = currentRoster.m_teamSize / desiredClassInfo->m_countPerTeamSize;
if ( currentRoster.m_count[ desiredClassInfo->m_class ] - desiredClassInfo->m_minTeamSizeToSelect < maxCountPer )
{
// below required limit - choose only this class
desiredClassVector.RemoveAll();
desiredClassVector.AddToTail( desiredClassInfo->m_class );
break;
}
}
// valid class to choose
desiredClassVector.AddToTail( desiredClassInfo->m_class );
}
if ( desiredClassVector.Count() == 0 )
{
if ( allowedClassForBotRosterVector.Count() == 0 )
{
// nothing available
Warning( "TFBot unable to choose a class, defaulting to 'auto'\n" );
return "auto";
}
else
{
desiredClassVector = allowedClassForBotRosterVector;
}
}
int which = RandomInt( 0, desiredClassVector.Count()-1 );
// if we need to destroy a sentry, pick a class that can do so
if ( GetEnemySentry() )
{
// best sentry demolitions
int demoman = desiredClassVector.Find( TF_CLASS_DEMOMAN );
if ( demoman >= 0 )
{
which = demoman;
}
else
{
// next best sentry demolitions
int spy = desiredClassVector.Find( TF_CLASS_SPY );
if ( spy >= 0 )
{
which = spy;
}
else
{
// good sentry demolitions
int soldier = desiredClassVector.Find( TF_CLASS_SOLDIER );
if ( soldier >= 0 )
{
which = soldier;
}
}
}
}
TFPlayerClassData_t *classData = GetPlayerClassData( desiredClassVector[ which ] );
if ( classData )
{
return classData->m_szClassName;
}
Warning( "TFBot unable to get data for desired class, defaulting to 'auto'\n" );
return "auto";
}
//-----------------------------------------------------------------------------------------------------
CTFBot::CTFBot()
{
m_body = new CTFBotBody( this );
m_locomotor = new CTFBotLocomotion( this );
m_vision = new CTFBotVision( this );
ALLOCATE_INTENTION_INTERFACE( CTFBot );
m_spawnArea = NULL;
m_weaponRestrictionFlags = 0;
m_attributeFlags = 0;
m_homeArea = NULL;
m_squad = NULL;
m_didReselectClass = false;
m_enemySentry = NULL;
m_spotWhereEnemySentryLastInjuredMe = vec3_origin;
m_isLookingAroundForEnemies = true;
m_behaviorFlags = 0;
m_attentionFocusEntity = NULL;
m_noisyTimer.Invalidate();
if ( TFGameRules()->IsInTraining() )
{
m_difficulty = CTFBot::EASY;
}
else
{
m_difficulty = clamp( (CTFBot::DifficultyType)tf_bot_difficulty.GetInt(), CTFBot::EASY, CTFBot::EXPERT );
}
m_actionPoint = NULL;
m_proxy = NULL;
m_spawner = NULL;
m_myControlPoint = NULL;
SetMission( NO_MISSION, MISSION_DOESNT_RESET_BEHAVIOR_SYSTEM );
SetMissionTarget( NULL );
m_missionString.Clear();
m_fModelScaleOverride = -1.0f;
m_maxVisionRangeOverride = -1.0f;
m_squadFormationError = 0.0f;
m_hFollowingFlagTarget = NULL;
SetShouldQuickBuild( false );
SetAutoJump( 0.f, 0.f );
ClearSniperSpots();
ListenForGameEvent( "teamplay_point_startcapture" );
ListenForGameEvent( "teamplay_point_captured" );
ListenForGameEvent( "teamplay_round_win" );
ListenForGameEvent( "teamplay_flag_event" );
}
//-----------------------------------------------------------------------------------------------------
CTFBot::~CTFBot()
{
// delete Intention first, since destruction of Actions may access other components
DEALLOCATE_INTENTION_INTERFACE;
if ( m_body )
delete m_body;
if ( m_locomotor )
delete m_locomotor;
if ( m_vision )
delete m_vision;
m_suspectedSpyVector.PurgeAndDeleteElements();
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::Spawn()
{
BaseClass::Spawn();
m_spawnArea = NULL;
m_justLostPointTimer.Invalidate();
m_squad = NULL;
m_didReselectClass = false;
m_isLookingAroundForEnemies = true;
m_attentionFocusEntity = NULL;
m_suspectedSpyVector.PurgeAndDeleteElements();
m_knownSpyVector.RemoveAll();
m_delayedNoticeVector.RemoveAll();
m_myControlPoint = NULL;
ClearSniperSpots();
ClearTags();
m_hFollowingFlagTarget = NULL;
m_requiredWeaponStack.Clear();
SetShouldQuickBuild( false );
SetSquadFormationError( 0.0f );
SetBrokenFormation( false );
GetVisionInterface()->ForgetAllKnownEntities();
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::SetMission( MissionType mission, bool resetBehaviorSystem )
{
SetPrevMission( m_mission );
m_mission = mission;
if ( resetBehaviorSystem )
{
// reset the behavior system to start the given mission
GetIntentionInterface()->Reset();
}
// Temp hack - some missions play an idle loop
if ( m_mission > NO_MISSION )
{
StartIdleSound();
}
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::PhysicsSimulate( void )
{
BaseClass::PhysicsSimulate();
if ( m_spawnArea == NULL )
{
m_spawnArea = GetLastKnownArea();
}
if ( HasAttribute( CTFBot::ALWAYS_CRIT ) && !m_Shared.InCond( TF_COND_CRITBOOSTED_USER_BUFF ) )
{
m_Shared.AddCond( TF_COND_CRITBOOSTED_USER_BUFF );
}
// force my speed to be recalculated to keep squad together and restore speed afterwards
TeamFortress_SetSpeed();
if ( IsInASquad() )
{
if ( GetSquad()->GetMemberCount() <= 1 || GetSquad()->GetLeader() == NULL )
{
// squad has collapsed - disband it
LeaveSquad();
}
}
// If we're dead, choose a new class.
// We need to do this outside of the behavior system, since changing class can
// sometimes force an immediate respawn, which will destroy the bot's existing actions out from under it.
if ( !IsAlive() && !m_didReselectClass && tf_bot_keep_class_after_death.GetBool() == false && TFGameRules()->CanBotChangeClass( this ) )
{
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
return;
const char *classname = FStrEq( tf_bot_force_class.GetString(), "" ) ? GetNextSpawnClassname() : tf_bot_force_class.GetString();
HandleCommand_JoinClass( classname );
m_didReselectClass = true;
}
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::Touch( CBaseEntity *pOther )
{
BaseClass::Touch( pOther );
CTFPlayer *them = ToTFPlayer( pOther );
if ( them && IsEnemy( them ) )
{
if ( them->m_Shared.IsStealthed() || them->m_Shared.InCond( TF_COND_DISGUISED ) )
{
// bumped a spy - they are discovered!
if ( TFGameRules()->IsMannVsMachineMode() ) // we have to build up to knowing that they are a spy in MvM
{
SuspectSpy( them );
}
else
{
RealizeSpy( them );
}
}
// always notice if we bump an enemy
TheNextBots().OnWeaponFired( them, them->GetActiveTFWeapon() );
}
}
//-----------------------------------------------------------------------------------------------------
// Avoid penetrating teammates
void CTFBot::AvoidPlayers( CUserCmd *pCmd )
{
// Turn off the avoid player code.
if ( !tf_avoidteammates.GetBool() || !tf_avoidteammates_pushaway.GetBool() )
return;
Vector forward, right;
EyeVectors( &forward, &right );
CUtlVector< CTFPlayer * > playerVector;
CollectPlayers( &playerVector, GetTeamNumber(), COLLECT_ONLY_LIVING_PLAYERS );
Vector avoidVector = vec3_origin;
float tooClose = 50.0f;
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
// bots stay farther apart in MvM mode
tooClose = 150.0f;
}
for( int i=0; i<playerVector.Count(); ++i )
{
CTFPlayer *them = playerVector[i];
if ( IsSelf( them ) )
{
continue;
}
if ( HasTheFlag() )
{
// Don't push around the flag (bomb) carrier.
// We need this for MvM mode so friendly bots don't
// move the bomb jumper and cause him to restart.
continue;
}
if ( IsPlayerClass( TF_CLASS_MEDIC ) )
{
if ( !them->IsPlayerClass( TF_CLASS_MEDIC ) )
{
// medics only avoid other medics, so they stay with their patient
continue;
}
}
else if ( IsInASquad() )
{
// if I'm a non-Medic in a Squad, I'm part of a formation
continue;
}
Vector between = GetAbsOrigin() - them->GetAbsOrigin();
if ( between.IsLengthLessThan( tooClose ) )
{
float range = between.NormalizeInPlace();
avoidVector += ( 1.0f - ( range / tooClose ) ) * between;
}
}
if ( avoidVector.IsZero() )
{
m_Shared.SetSeparation( false );
m_Shared.SetSeparationVelocity( vec3_origin );
return;
}
avoidVector.NormalizeInPlace();
m_Shared.SetSeparation( true );
const float maxSpeed = 50.0f;
m_Shared.SetSeparationVelocity( avoidVector * maxSpeed );
float ahead = maxSpeed * DotProduct( forward, avoidVector );
float side = maxSpeed * DotProduct( right, avoidVector );
pCmd->forwardmove += ahead;
pCmd->sidemove += side;
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::UpdateOnRemove( void )
{
StopIdleSound();
BaseClass::UpdateOnRemove();
}
//-----------------------------------------------------------------------------------------------------
int CTFBot::ShouldTransmit( const CCheckTransmitInfo *pInfo )
{
if ( HasAttribute( USE_BOSS_HEALTH_BAR ) )
{
return FL_EDICT_ALWAYS;
}
return BaseClass::ShouldTransmit( pInfo );
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::ChangeTeam( int iTeamNum, bool bAutoTeam, bool bSilent, bool bAutoBalance /*= false*/ )
{
BaseClass::ChangeTeam( iTeamNum, bAutoTeam, bSilent, bAutoBalance );
if ( TFGameRules()->IsMannVsMachineMode() )
{
SetPrevMission( CTFBot::NO_MISSION );
ClearAllAttributes();
// Clear Sound
StopIdleSound();
}
}
//-----------------------------------------------------------------------------------------------------
bool CTFBot::ShouldGib( const CTakeDamageInfo &info )
{
// only gib giant/miniboss
if ( TFGameRules()->IsMannVsMachineMode() && ( IsMiniBoss() || GetModelScale() > 1.f ) )
{
return true;
}
return BaseClass::ShouldGib( info );
}
//-----------------------------------------------------------------------------------------------------
bool CTFBot::IsAllowedToPickUpFlag( void ) const
{
if ( !BaseClass::IsAllowedToPickUpFlag() )
{
return false;
}
// only the leader of a squad can pick up the flag
if ( IsInASquad() && !GetSquad()->IsLeader( const_cast< CTFBot * >( this ) ) )
return false;
// mission bots can't pick up the flag
return !IsOnAnyMission();
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::InitClass( void )
{
BaseClass::InitClass();
}
void CTFBot::ModifyMaxHealth( int nNewMaxHealth, bool bSetCurrentHealth /*= true*/, bool bAllowModelScaling /*= true*/ )
{
if ( GetMaxHealth() != nNewMaxHealth )
{
static CSchemaAttributeDefHandle pAttrDef_HiddenMaxHealthNonBuffed( "hidden maxhealth non buffed" );
if ( !pAttrDef_HiddenMaxHealthNonBuffed )
{
Warning( "TFBotSpawner: Invalid attribute 'hidden maxhealth non buffed'\n" );
}
else
{
CAttributeList *pAttrList = GetAttributeList();
if ( pAttrList )
{
pAttrList->SetRuntimeAttributeValue( pAttrDef_HiddenMaxHealthNonBuffed, nNewMaxHealth - GetMaxHealth() );
}
}
}
if ( bSetCurrentHealth )
{
SetHealth( nNewMaxHealth );
}
if ( bAllowModelScaling && IsMiniBoss() )
{
SetModelScale( m_fModelScaleOverride > 0.0f ? m_fModelScaleOverride : tf_mvm_miniboss_scale.GetFloat() );
}
}
//-----------------------------------------------------------------------------------------------------
/**
* Invoked when a game event occurs
*/
void CTFBot::FireGameEvent( IGameEvent *event )
{
const char *eventName = event->GetName();
if ( FStrEq( eventName, "teamplay_point_captured" ) )
{
ClearMyControlPoint();
int whoCapped = event->GetInt( "team" );
int pointID = event->GetInt( "cp" );
if ( whoCapped == GetTeamNumber() )
{
OnTerritoryCaptured( pointID );
}
else
{
OnTerritoryLost( pointID );
m_justLostPointTimer.Start( RandomFloat( 10.0f, 20.0f ) );
}
}
else if ( FStrEq( eventName, "teamplay_point_startcapture" ) )
{
int pointID = event->GetInt( "cp" );
OnTerritoryContested( pointID );
}
else if ( FStrEq( eventName, "teamplay_flag_event" ) )
{
if ( event->GetInt( "eventtype" ) == TF_FLAGEVENT_PICKUP )
{
int iPlayer = event->GetInt( "player" );
if ( iPlayer == entindex() )
{
// I just picked up the flag
OnPickUp( NULL, NULL );
}
}
}
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::Event_Killed( const CTakeDamageInfo &info )
{
BaseClass::Event_Killed( info );
if ( HasProxy() )
{
GetProxy()->OnKilled();
}
// announce Spies
if ( TFGameRules()->IsMannVsMachineMode() )
{
if ( IsPlayerClass( TF_CLASS_SPY ) )
{
CUtlVector< CTFPlayer * > playerVector;
CollectPlayers( &playerVector, TF_TEAM_PVE_INVADERS, COLLECT_ONLY_LIVING_PLAYERS );
int spyCount = 0;
for( int i=0; i<playerVector.Count(); ++i )
{
if ( playerVector[i]->IsPlayerClass( TF_CLASS_SPY ) )
{
++spyCount;
}
}
IGameEvent *event = gameeventmanager->CreateEvent( "mvm_mission_update" );
if ( event )
{
event->SetInt( "class", TF_CLASS_SPY );
event->SetInt( "count", spyCount );
gameeventmanager->FireEvent( event );
}
}
else if ( IsPlayerClass( TF_CLASS_ENGINEER ) )
{
// in MVM, when an engineer dies, we need to decouple his objects so they stay alive when his bot slot gets recycled
while ( GetObjectCount() > 0 )
{
// set to not have owner
CBaseObject *pObject = GetObject( 0 );
if ( pObject )
{
pObject->SetOwnerEntity( NULL );
pObject->SetBuilder( NULL );
}
RemoveObject( pObject );
}
// unown engineer nest if owned any
for ( int i=0; i<ITFBotHintEntityAutoList::AutoList().Count(); ++i )
{
CBaseTFBotHintEntity* pHint = static_cast< CBaseTFBotHintEntity* >( ITFBotHintEntityAutoList::AutoList()[i] );
if ( pHint->GetOwnerEntity() == this )
{
pHint->SetOwnerEntity( NULL );
}
}
CUtlVector< CTFPlayer* > playerVector;
CollectPlayers( &playerVector, TF_TEAM_PVE_INVADERS, COLLECT_ONLY_LIVING_PLAYERS );
bool bShouldAnnounceLastEngineerBotDeath = HasAttribute( CTFBot::TELEPORT_TO_HINT );
if ( bShouldAnnounceLastEngineerBotDeath )
{
for ( int i=0; i<playerVector.Count(); ++i )
{
if ( playerVector[i] != this && playerVector[i]->IsPlayerClass( TF_CLASS_ENGINEER ) )
{
bShouldAnnounceLastEngineerBotDeath = false;
break;
}
}
}
if ( bShouldAnnounceLastEngineerBotDeath )
{
bool bEngineerTeleporterInTheWorld = false;
for ( int i=0; i<IBaseObjectAutoList::AutoList().Count(); ++i )
{
CBaseObject* pObj = static_cast< CBaseObject* >( IBaseObjectAutoList::AutoList()[i] );
if ( pObj->GetType() == OBJ_TELEPORTER && pObj->GetTeamNumber() == TF_TEAM_PVE_INVADERS )
{
bEngineerTeleporterInTheWorld = true;
}
}
if ( bEngineerTeleporterInTheWorld )
{
TFGameRules()->BroadcastSound( 255, "Announcer.MVM_An_Engineer_Bot_Is_Dead_But_Not_Teleporter" );
}
else
{
TFGameRules()->BroadcastSound( 255, "Announcer.MVM_An_Engineer_Bot_Is_Dead" );
}
}
}
// remove this bot from following flag
for ( int i=0; i<ICaptureFlagAutoList::AutoList().Count(); ++i )
{
for ( int i=0; i<ICaptureFlagAutoList::AutoList().Count(); ++i )
{
CCaptureFlag *flag = static_cast< CCaptureFlag* >( ICaptureFlagAutoList::AutoList()[i] );
flag->RemoveFollower( this );
}
}
} // MvM
if ( HasSpawner() )
{
GetSpawner()->OnBotKilled( this );
}
if ( IsInASquad() )
{
LeaveSquad();
}
CTFNavArea *lastArea = (CTFNavArea *)GetLastKnownArea();
if ( lastArea )
{
// remove us from old visible set
NavAreaCollector wasVisible;
lastArea->ForAllPotentiallyVisibleAreas( wasVisible );
int i;
for( i=0; i<wasVisible.m_area.Count(); ++i )
{
CTFNavArea *area = (CTFNavArea *)wasVisible.m_area[i];
area->RemovePotentiallyVisibleActor( this );
}
}
if ( info.GetInflictor() && info.GetInflictor()->GetTeamNumber() != GetTeamNumber() )
{
CObjectSentrygun *sentrygun = dynamic_cast< CObjectSentrygun * >( info.GetInflictor() );
if ( sentrygun )
{
// we were killed by an enemy sentry - remember it
RememberEnemySentry( sentrygun, GetAbsOrigin() );
}
}
StopIdleSound();
}
//-----------------------------------------------------------------------------------------------------
CTeamControlPoint *CTFBot::SelectPointToCapture( CUtlVector< CTeamControlPoint * > *captureVector ) const
{
if ( !captureVector || captureVector->Count() == 0 )
{
return NULL;
}
if ( captureVector->Count() == 1 )
{
// only one choice
return captureVector->Element(0);
}
// if we're capturing a point, stay on it
if ( const_cast< CTFBot * >( this )->IsCapturingPoint() )
{
CTriggerAreaCapture *trigger = const_cast< CTFBot * >( this )->GetControlPointStandingOn();
if ( trigger )
{
return trigger->GetControlPoint();
}
}
// if we're near a point that is being captured, go help (in the event multiple points are being simultaneously captured)
CTeamControlPoint *closestPoint = SelectClosestControlPointByTravelDistance( captureVector );
if ( closestPoint )
{
bool alwaysUseClosest = false;
#ifdef STAGING_ONLY
alwaysUseClosest = TFGameRules() && TFGameRules()->IsBountyMode();
#endif // STAGING_ONLY
if ( IsPointBeingCaptured( closestPoint ) || alwaysUseClosest )
{
return closestPoint;
}
}
// if any point is being captured by our team, go help
for( int i=0; i<captureVector->Count(); ++i )
{
CTeamControlPoint *point = captureVector->Element(i);
if ( IsPointBeingCaptured( point ) )
{
return point;
}
}
// no points are currently being captured - pick the point with the least combat
CTeamControlPoint *safestPoint = NULL;
float safestPointCombat = FLT_MAX;
bool areAllPointsCombatFree = true;
for( int i=0; i<captureVector->Count(); ++i )
{
CTeamControlPoint *point = captureVector->Element(i);
CTFNavArea *pointArea = TheTFNavMesh()->GetControlPointCenterArea( point->GetPointIndex() );
if ( !pointArea )
{
continue;
}
float combat = pointArea->GetCombatIntensity();
const float minCombat = 0.1f;
if ( combat > minCombat )
{
areAllPointsCombatFree = false;
}
if ( combat < safestPointCombat )
{
safestPoint = point;
safestPointCombat = combat;
}
}
// if no points are in combat, pick a random point
if ( areAllPointsCombatFree )
{
const float decisionPeriod = 60.0f;
int which = captureVector->Count() * TransientlyConsistentRandomValue( decisionPeriod );
which = clamp( which, 0, captureVector->Count()-1 );
return captureVector->Element( which );
}
// choose the point with the least combat
return safestPoint;
}
//---------------------------------------------------------------------------------------------
CTeamControlPoint *CTFBot::SelectPointToDefend( CUtlVector< CTeamControlPoint * > *defendVector ) const
{
if ( defendVector && defendVector->Count() > 0 )
{
if ( HasAttribute( CTFBot::PRIORITIZE_DEFENSE ) )
{
return SelectClosestControlPointByTravelDistance( defendVector );
}
return defendVector->Element( RandomInt( 0, defendVector->Count()-1 ) );
}
return NULL;
}
//-----------------------------------------------------------------------------------------------------
/**
* Return the point we have decided to capture or defend
*/
CTeamControlPoint *CTFBot::GetMyControlPoint( void ) const
{
if ( m_myControlPoint != NULL && !m_evaluateControlPointTimer.IsElapsed() )
{
return m_myControlPoint;
}
m_evaluateControlPointTimer.Start( RandomFloat( 1.0f, 2.0f ) );
CUtlVector< CTeamControlPoint * > captureVector;
TFGameRules()->CollectCapturePoints( const_cast< CTFBot * >( this ), &captureVector );
CUtlVector< CTeamControlPoint * > defendVector;
TFGameRules()->CollectDefendPoints( const_cast< CTFBot * >( this ), &defendVector );
if ( IsPlayerClass( TF_CLASS_ENGINEER ) || IsPlayerClass( TF_CLASS_SNIPER ) || HasAttribute( CTFBot::PRIORITIZE_DEFENSE ) )
{
// engineers always try to defend first
if ( defendVector.Count() > 0 )
{
m_myControlPoint = SelectPointToDefend( &defendVector );
return m_myControlPoint;
}
}
// if we have a point we can capture - do it
m_myControlPoint = SelectPointToCapture( &captureVector );
if ( m_myControlPoint == NULL )
{
// otherwise, defend our point(s) from capture
m_myControlPoint = SelectPointToDefend( &defendVector );
}
return m_myControlPoint;
}
//-----------------------------------------------------------------------------------------------------
// Return flag we want to fetch
CCaptureFlag *CTFBot::GetFlagToFetch( void ) const
{
CUtlVector<CCaptureFlag *> flagsVector;
int nCarriedFlags = 0;
// MvM Engineer bot never pick up a flag
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
if ( GetTeamNumber() == TF_TEAM_PVE_INVADERS && IsPlayerClass( TF_CLASS_ENGINEER ) )
{
return NULL;
}
if( HasAttribute( CTFBot::IGNORE_FLAG ) )
{
return NULL;
}
if ( TFGameRules()->IsMannVsMachineMode() && HasFlagTaget() )
{
return GetFlagTarget();
}
}
// Collect flags
for ( int i=0; i<ICaptureFlagAutoList::AutoList().Count(); ++i )
{
CCaptureFlag *flag = static_cast< CCaptureFlag* >( ICaptureFlagAutoList::AutoList()[i] );
if ( flag->IsDisabled() )
continue;
// If I'm carrying a flag, look for mine and early-out
if ( HasTheFlag() )
{
if ( flag->GetOwnerEntity() == this )
{
return flag;
}
}
switch( flag->GetType() )
{
case TF_FLAGTYPE_CTF:
if ( flag->GetTeamNumber() == GetEnemyTeam( GetTeamNumber() ) )
{
// we want to steal the other team's flag
flagsVector.AddToTail( flag );
}
break;
case TF_FLAGTYPE_ATTACK_DEFEND:
case TF_FLAGTYPE_TERRITORY_CONTROL:
case TF_FLAGTYPE_INVADE:
if ( flag->GetTeamNumber() != GetEnemyTeam( GetTeamNumber() ) )
{
// we want to move our team's flag or a neutral flag
flagsVector.AddToTail( flag );
}
break;
}
if ( flag->IsStolen() )
{
nCarriedFlags++;
}
}
CCaptureFlag *pClosestFlag = NULL;
float flClosestFlagDist = FLT_MAX;
CCaptureFlag *pClosestUncarriedFlag = NULL;
float flClosestUncarriedFlagDist = FLT_MAX;
if ( TFGameRules() && TFGameRules()->IsMannVsMachineMode() )
{
int nMinFollower = INT_MAX;
FOR_EACH_VEC( flagsVector, i )
{
CCaptureFlag *pFlag = flagsVector[i];
if ( pFlag )
{
// find the one which needs the most love
if ( pFlag->GetNumFollowers() < nMinFollower )
{
nMinFollower = pFlag->GetNumFollowers();
pClosestFlag = NULL;
flClosestFlagDist = FLT_MAX;
pClosestUncarriedFlag = NULL;
flClosestUncarriedFlagDist = FLT_MAX;
}
if ( pFlag->GetNumFollowers() == nMinFollower )
{
// Find the closest
float flDist = ( pFlag->GetAbsOrigin() - GetAbsOrigin() ).LengthSqr();
if ( flDist < flClosestFlagDist )
{
pClosestFlag = pFlag;
flClosestFlagDist = flDist;
}
// Find the closest uncarried
if ( nCarriedFlags < flagsVector.Count() && !pFlag->IsStolen() )
{
if ( flDist < flClosestUncarriedFlagDist )
{
pClosestUncarriedFlag = flagsVector[i];
flClosestUncarriedFlagDist = flDist;
}
}
}
}
}
}
else
{
FOR_EACH_VEC( flagsVector, i )
{
if ( flagsVector[i] )
{
// Find the closest
float flDist = ( flagsVector[i]->GetAbsOrigin() - GetAbsOrigin() ).LengthSqr();
if ( flDist < flClosestFlagDist )
{
pClosestFlag = flagsVector[i];
flClosestFlagDist = flDist;
}
// Find the closest uncarried
if ( nCarriedFlags < flagsVector.Count() && !flagsVector[i]->IsStolen() )
{
if ( flDist < flClosestUncarriedFlagDist )
{
pClosestUncarriedFlag = flagsVector[i];
flClosestUncarriedFlagDist = flDist;
}
}
}
}
}
// If we have an uncarried flag, prioritize
if ( pClosestUncarriedFlag )
return pClosestUncarriedFlag;
return pClosestFlag;
}
//-----------------------------------------------------------------------------------------------------
// Return capture zone for our flag(s)
CCaptureZone *CTFBot::GetFlagCaptureZone( void ) const
{
for( int i=0; i<ICaptureZoneAutoList::AutoList().Count(); ++i )
{
CCaptureZone *zone = static_cast< CCaptureZone* >( ICaptureZoneAutoList::AutoList()[i] );
if ( zone->GetTeamNumber() == GetTeamNumber() )
{
return zone;
}
}
return NULL;
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::ClearMyControlPoint( void )
{
m_myControlPoint = NULL;
m_evaluateControlPointTimer.Invalidate();
}
//-----------------------------------------------------------------------------------------------------
/**
* Return true if no enemy has contested any point yet
*/
bool CTFBot::AreAllPointsUncontestedSoFar( void ) const
{
CTeamControlPointMaster *master = g_hControlPointMasters.Count() ? g_hControlPointMasters[0] : NULL;
if ( master )
{
for( int i=0; i<master->GetNumPoints(); ++i )
{
CTeamControlPoint *point = master->GetControlPoint( i );
if ( point && point->HasBeenContested() )
return false;
}
}
return true;
}
//-----------------------------------------------------------------------------------------------------
// Return true if the given point is being captured
bool CTFBot::IsPointBeingCaptured( CTeamControlPoint *point ) const
{
if ( point == NULL )
return false;
if ( point->LastContestedAt() > 0.0f && ( gpGlobals->curtime - point->LastContestedAt() ) < 5.0f )
{
// the point is, or was very recently, contested
return true;
}
return false;
}
//---------------------------------------------------------------------------------------------
// Return true if any point is being captured
bool CTFBot::IsAnyPointBeingCaptured( void ) const
{
CTeamControlPointMaster *master = g_hControlPointMasters.Count() ? g_hControlPointMasters[0] : NULL;
if ( master )
{
for( int i=0; i<master->GetNumPoints(); ++i )
{
CTeamControlPoint *point = master->GetControlPoint( i );
if ( IsPointBeingCaptured( point ) )
return true;
}
}
return false;
}
//---------------------------------------------------------------------------------------------
// Return true if we are within a short travel distance of the current point
bool CTFBot::IsNearPoint( CTeamControlPoint *point ) const
{
CTFNavArea *myArea = GetLastKnownArea();
if ( !myArea || !point )
{
return false;
}
CTFNavArea *pointArea = TheTFNavMesh()->GetControlPointCenterArea( point->GetPointIndex() );
if ( !pointArea )
{
return false;
}
float travelToPoint = fabs( myArea->GetIncursionDistance( GetTeamNumber() ) - pointArea->GetIncursionDistance( GetTeamNumber() ) );
return travelToPoint < tf_bot_near_point_travel_distance.GetFloat();
}
//---------------------------------------------------------------------------------------------
// Return time left to capture the point before we lose the game
float CTFBot::GetTimeLeftToCapture( void ) const
{
if ( TFGameRules()->IsInKothMode() )
{
if ( TFGameRules()->GetKothTeamTimer( GetEnemyTeam( GetTeamNumber() ) ) )
{
return TFGameRules()->GetKothTeamTimer( GetEnemyTeam( GetTeamNumber() ) )->GetTimeRemaining();
}
}
else if ( TFGameRules()->GetActiveRoundTimer() )
{
return TFGameRules()->GetActiveRoundTimer()->GetTimeRemaining();
}
return 0.0f;
}
//-----------------------------------------------------------------------------------------------------
// Do internal setup when control point changes
void CTFBot::SetupSniperSpotAccumulation( void )
{
VPROF_BUDGET( "CTFBot::SetupSniperSpotAccumulation", "NextBot" );
CBaseEntity *goalEntity = NULL;
if ( TFGameRules()->GetGameType() == TF_GAMETYPE_ESCORT )
{
// try to find a payload cart to guard
CTeamTrainWatcher *trainWatcher = TFGameRules()->GetPayloadToPush( GetTeamNumber() );
if ( !trainWatcher )
{
trainWatcher = TFGameRules()->GetPayloadToBlock( GetTeamNumber() );
}
if ( trainWatcher )
{
goalEntity = trainWatcher->GetTrainEntity();
}
}
else if ( TFGameRules()->GetGameType() == TF_GAMETYPE_CP )
{
goalEntity = GetMyControlPoint();
}
if ( !goalEntity )
{
ClearSniperSpots();
return;
}
if ( goalEntity == m_snipingGoalEntity )
{
// if goal has moved too much (ie: payload cart), recompute our spots
Vector toGoal = m_snipingGoalEntity->WorldSpaceCenter() - m_lastSnipingGoalEntityPosition;
if ( toGoal.IsLengthLessThan( tf_bot_sniper_goal_entity_move_tolerance.GetFloat() ) )
{
// already set up
return;
}
}
ClearSniperSpots();
int myTeam = GetTeamNumber();
int enemyTeam = ( myTeam == TF_TEAM_BLUE ) ? TF_TEAM_RED : TF_TEAM_BLUE;
bool isDefendingPoint = false;
CTFNavArea *goalEntityArea = NULL;
if ( TFGameRules()->GetGameType() == TF_GAMETYPE_ESCORT )
{
// the cart is owned by the invaders
isDefendingPoint = ( goalEntity->GetTeamNumber() != myTeam );
goalEntityArea = (CTFNavArea *)TheTFNavMesh()->GetNearestNavArea( goalEntity->WorldSpaceCenter(), GETNAVAREA_CHECK_GROUND, 500.0f );
}
else
{
isDefendingPoint = ( GetMyControlPoint()->GetOwner() == myTeam );
goalEntityArea = TheTFNavMesh()->GetControlPointCenterArea( GetMyControlPoint()->GetPointIndex() );
}
// we are sniping a different control point - setup for new point accumulation
m_sniperVantageAreaVector.RemoveAll();
m_sniperTheaterAreaVector.RemoveAll();
if ( !goalEntityArea )
{
return;
}
for( int i=0; i<TheNavAreas.Count(); ++i )
{
CTFNavArea *area = (CTFNavArea *)TheNavAreas[i];
if ( !area->IsReachableByTeam( myTeam ) || !area->IsReachableByTeam( enemyTeam ) )
{
continue;
}
if ( area->GetIncursionDistance( enemyTeam ) <= goalEntityArea->GetIncursionDistance( enemyTeam ) )
{
m_sniperTheaterAreaVector.AddToTail( area );
}
// if this is my point, I can stand on it, or go a bit beyond it
float myIncursionTolerance = tf_bot_sniper_spot_point_tolerance.GetFloat();
if ( !isDefendingPoint )
{
// not my point, keep back from it a bit
myIncursionTolerance *= -1.0f;
}
if ( area->GetIncursionDistance( myTeam ) <= goalEntityArea->GetIncursionDistance( myTeam ) + myIncursionTolerance )
{
m_sniperVantageAreaVector.AddToTail( area );
}
}
m_snipingGoalEntity = goalEntity;
m_lastSnipingGoalEntityPosition = goalEntity->WorldSpaceCenter();
}
//-----------------------------------------------------------------------------------------------------
// Randomly sample points within candidate areas to find good sniping positions
void CTFBot::AccumulateSniperSpots( void )
{
VPROF_BUDGET( "CTFBot::AccumulateSniperSpots", "NextBot" );
SetupSniperSpotAccumulation();
if ( m_sniperVantageAreaVector.Count() == 0 || m_sniperTheaterAreaVector.Count() == 0 )
{
// retry every so often to catch cases where the incursion data is invalid during setup time
// due to blocked/closed off areas, etc.
if ( m_retrySniperSpotSetupTimer.IsElapsed() )
{
// retry
ClearSniperSpots();
}
return;
}
SniperSpotInfo info;
for( int count=0; count<tf_bot_sniper_spot_search_count.GetInt(); ++count )
{
// pick a random vantage area to sample
int which = RandomInt( 0, m_sniperVantageAreaVector.Count()-1 );
info.m_vantageArea = m_sniperVantageAreaVector[ which ];
info.m_vantageSpot = info.m_vantageArea->GetRandomPoint();
// pick a random theater area to sample
which = RandomInt( 0, m_sniperTheaterAreaVector.Count()-1 );
info.m_theaterArea = m_sniperTheaterAreaVector[ which ];
info.m_theaterSpot = info.m_theaterArea->GetRandomPoint();
info.m_range = ( info.m_vantageSpot - info.m_theaterSpot ).Length();
if ( info.m_range < tf_bot_sniper_spot_min_range.GetFloat() )
{
// not long enough sightline
continue;
}
for( int i=0; i<m_sniperSpotVector.Count(); ++i )
{
if ( ( info.m_vantageSpot - m_sniperSpotVector[i].m_vantageSpot ).IsLengthLessThan( tf_bot_sniper_spot_epsilon.GetFloat() ) )
{
// too close to existing spot
continue;
}
}
Vector eyeOffset( 0, 0, 60.0f );
if ( IsLineOfFireClear( info.m_vantageSpot + eyeOffset, info.m_theaterSpot + eyeOffset ) )
{
// valid spot
// maximize the time it takes the enemy to get to us
info.m_advantage = info.m_vantageArea->GetIncursionDistance( GetEnemyTeam( GetTeamNumber() ) ) - info.m_theaterArea->GetIncursionDistance( GetEnemyTeam( GetTeamNumber() ) );
// if we have already maxxed out our sniper spots, replace the worst one if this is better
if ( m_sniperSpotVector.Count() >= tf_bot_sniper_spot_max_count.GetInt() )
{
int worst = -1;
for( int i=0; i<m_sniperSpotVector.Count(); ++i )
{
if ( worst < 0 || m_sniperSpotVector[i].m_advantage < m_sniperSpotVector[ worst ].m_advantage )
{
worst = i;
}
}
// if our new spot is better, replace it
if ( info.m_advantage > m_sniperSpotVector[ worst ].m_advantage )
{
m_sniperSpotVector[ worst ] = info;
}
}
else
{
m_sniperSpotVector.AddToTail( info );
}
}
}
if ( IsDebugging( NEXTBOT_BEHAVIOR ) )
{
for( int i=0; i<m_sniperSpotVector.Count(); ++i )
{
NDebugOverlay::Cross3D( m_sniperSpotVector[i].m_vantageSpot, 5.0f, 255, 0, 255, true, 0.1f );
NDebugOverlay::Line( m_sniperSpotVector[i].m_vantageSpot, m_sniperSpotVector[i].m_theaterSpot, 0, 200, 0, true, 0.1f );
}
}
}
//-----------------------------------------------------------------------------------------------------
void CTFBot::ClearSniperSpots( void )
{
m_sniperSpotVector.RemoveAll();
m_sniperVantageAreaVector.RemoveAll();
m_sniperTheaterAreaVector.RemoveAll();
m_snipingGoalEntity = NULL;
m_retrySniperSpotSetupTimer.Start( RandomFloat( 5.0f, 10.0f ) );
}
//---------------------------------------------------------------------------------------------
class CCollectReachableObjects : public ISearchSurroundingAreasFunctor
{
public:
CCollectReachableObjects( const CTFBot *me, float maxRange, const CUtlVector< CHandle< CBaseEntity > > &potentialVector, CUtlVector< CHandle< CBaseEntity > > *collectionVector ) : m_potentialVector( potentialVector )
{
m_me = me;
m_maxRange = maxRange;
m_collectionVector = collectionVector;
}
virtual bool operator() ( CNavArea *area, CNavArea *priorArea, float travelDistanceSoFar )
{
// do any of the potential objects overlap this area?
FOR_EACH_VEC( m_potentialVector, it )
{
CBaseEntity *obj = m_potentialVector[ it ];
if ( obj && area->Contains( obj->WorldSpaceCenter() ) )
{
// reachable - keep it
if ( !m_collectionVector->HasElement( obj ) )
{
m_collectionVector->AddToTail( obj );
}
}
}
return true;
}
virtual bool ShouldSearch( CNavArea *adjArea, CNavArea *currentArea, float travelDistanceSoFar )
{
if ( adjArea->IsBlocked( m_me->GetTeamNumber() ) )
{
return false;
}
if ( travelDistanceSoFar > m_maxRange )
{
// too far away
return false;
}
return currentArea->IsContiguous( adjArea );
}
const CTFBot *m_me;
float m_maxRange;
const CUtlVector< CHandle< CBaseEntity > > &m_potentialVector;
CUtlVector< CHandle< CBaseEntity > > *m_collectionVector;
};
//
// Search outwards from startSearchArea and collect all reachable objects from the given list that pass the given filter
// Items in selectedObjectVector will be approximately sorted in nearest-to-farthest order (because of SearchSurroundingAreas)
//
void CTFBot::SelectReachableObjects( const CUtlVector< CHandle< CBaseEntity > > &candidateObjectVector,
CUtlVector< CHandle< CBaseEntity > > *selectedObjectVector,
const INextBotFilter &filter,
CNavArea *startSearchArea,
float maxRange ) const
{
if ( startSearchArea == NULL || selectedObjectVector == NULL )
return;
selectedObjectVector->RemoveAll();
// filter candidate objects
CUtlVector< CHandle< CBaseEntity > > filteredObjectVector;
for( int i=0; i<candidateObjectVector.Count(); ++i )
{
if ( filter.IsSelected( candidateObjectVector[i] ) )
{
filteredObjectVector.AddToTail( candidateObjectVector[i] );
}
}
// only keep those that are reachable by us
CCollectReachableObjects collector( this, maxRange, filteredObjectVector, selectedObjectVector );
SearchSurroundingAreas( startSearchArea, collector );
}
//---------------------------------------------------------------------------------------------
bool CTFBot::IsAmmoLow( void ) const
{
CTFWeaponBase *myWeapon = m_Shared.GetActiveTFWeapon();
if ( myWeapon )
{
if ( myWeapon->GetWeaponID() == TF_WEAPON_WRENCH )
{
// wrench is special. it's a melee weapon that wants ammo - metal
return ( GetAmmoCount( TF_AMMO_METAL ) <= 0 );
}
if ( myWeapon->IsMeleeWeapon() )
{
// we never run out of ammo with a melee weapon
return false;
}
// no projectile, no ammo needed
const char *weaponAlias = WeaponIdToAlias( myWeapon->GetWeaponID() );
if ( weaponAlias )
{
WEAPON_FILE_INFO_HANDLE weaponInfoHandle = LookupWeaponInfoSlot( weaponAlias );
if ( weaponInfoHandle != GetInvalidWeaponInfoHandle() )
{
CTFWeaponInfo *weaponInfo = static_cast< CTFWeaponInfo * >( GetFileWeaponInfoFromHandle( weaponInfoHandle ) );
if ( weaponInfo && weaponInfo->GetWeaponData( TF_WEAPON_PRIMARY_MODE ).m_iProjectile == TF_PROJECTILE_NONE )
{
// we don't shoot anything, so we don't need ammo
return false;
}
}
}
float ratio = (float)GetAmmoCount( TF_AMMO_PRIMARY ) / (float)( const_cast< CTFBot * >( this )->GetMaxAmmo( TF_AMMO_PRIMARY ) );
if ( ratio < 0.2f )
{
return true;
}
//if ( !myWeapon->HasPrimaryAmmo() && myWeapon->GetWeaponID() != TF_WEAPON_BUILDER && myWeapon->GetWeaponID() != TF_WEAPON_MEDIGUN )
}
return false;
}
//-----------------------------------------------------------------------------------------------------
bool CTFBot::IsAmmoFull( void ) const
{
bool isPrimaryFull = GetAmmoCount( TF_AMMO_PRIMARY ) >= const_cast< CTFBot * >( this )->GetMaxAmmo( TF_AMMO_PRIMARY );
bool isSecondaryFull = GetAmmoCount( TF_AMMO_SECONDARY ) >= const_cast< CTFBot * >( this )->GetMaxAmmo( TF_AMMO_SECONDARY );
if ( IsPlayerClass( TF_CLASS_ENGINEER ) )
{
// wrench is special. it's a melee weapon that wants ammo - metal
return ( GetAmmoCount( TF_AMMO_METAL ) >= 200 ) && isPrimaryFull && isSecondaryFull;
}
return isPrimaryFull && isSecondaryFull;
/*
CTFWeaponBase *myWeapon = m_Shared.GetActiveTFWeapon();
if ( myWeapon )
{
if ( IsPlayerClass( TF_CLASS_ENGINEER ) )
{
// wrench is special. it's a melee weapon that wants ammo - metal
return ( GetAmmoCount( TF_AMMO_METAL ) >= 200 );
}
if ( myWeapon->IsMeleeWeapon() )
{
// we never run out of ammo with a melee weapon
return true;
}
// no projectile, no ammo needed
const char *weaponAlias = WeaponIdToAlias( myWeapon->GetWeaponID() );
if ( weaponAlias )
{
WEAPON_FILE_INFO_HANDLE weaponInfoHandle = LookupWeaponInfoSlot( weaponAlias );
if ( weaponInfoHandle != GetInvalidWeaponInfoHandle() )
{
CTFWeaponInfo *weaponInfo = static_cast< CTFWeaponInfo * >( GetFileWeaponInfoFromHandle( weaponInfoHandle ) );
if ( weaponInfo && weaponInfo->GetWeaponData( TF_WEAPON_PRIMARY_MODE ).m_iProjectile == TF_PROJECTILE_NONE )
{
// we don't shoot anything, so we don't need ammo
return true;
}
}
}
bool isPrimaryFull = GetAmmoCount( TF_AMMO_PRIMARY ) >= const_cast< CTFBot * >( this )->GetMaxAmmo( TF_AMMO_PRIMARY );
bool isSecondaryFull = GetAmmoCount( TF_AMMO_SECONDARY ) >= const_cast< CTFBot * >( this )->GetMaxAmmo( TF_AMMO_SECONDARY );
return isPrimaryFull && isSecondaryFull;
}
return false;
*/
}
bool CTFBot::IsDormantWhenDead( void ) const
{
return false;
}
//-----------------------------------------------------------------------------------------------------
/**
* When someone fires their weapon
*/
void CTFBot::OnWeaponFired( CBaseCombatCharacter *whoFired, CBaseCombatWeapon *weapon )
{
VPROF_BUDGET( "CTFBot::OnWeaponFired", "NextBot" );
BaseClass::OnWeaponFired( whoFired, weapon );
if ( !whoFired || !whoFired->IsAlive() )
return;
if ( IsRangeGreaterThan( whoFired, tf_bot_notice_gunfire_range.GetFloat() ) )
return;
int noticeChance = 100;
if ( IsQuietWeapon( (CTFWeaponBase *)weapon ) )
{
if ( IsRangeGreaterThan( whoFired, tf_bot_notice_quiet_gunfire_range.GetFloat() ) )
{
// too far away to hear in any event
return;
}
switch( GetDifficulty() )
{
case EASY:
noticeChance = 10;
break;
case NORMAL:
noticeChance = 30;
break;
case HARD:
noticeChance = 60;
break;
default:
case EXPERT:
noticeChance = 90;
break;
}
if ( IsEnvironmentNoisy() )
{
// less likely to notice with all the noise
noticeChance /= 2;
}
}
else if ( IsRangeLessThan( whoFired, 1000.0f ) )
{
// loud gunfire in our area - it's now "noisy" for a bit
m_noisyTimer.Start( 3.0f );
}
if ( RandomInt( 1, 100 ) > noticeChance )
{
return;
}
// notice the gunfire
GetVisionInterface()->AddKnownEntity( whoFired );
}
//-----------------------------------------------------------------------------------------------------
// Return true if we match the given debug symbol
bool CTFBot::IsDebugFilterMatch( const char *name ) const
{
// player classname
if ( !Q_strnicmp( name, const_cast< CTFBot * >( this )->GetPlayerClass()->GetName(), Q_strlen( name ) ) )
{
return true;
}
return BaseClass::IsDebugFilterMatch( name );
}
//-----------------------------------------------------------------------------------------------------
class CFindClosestPotentiallyVisibleAreaToPos
{
public:
CFindClosestPotentiallyVisibleAreaToPos( const Vector &pos )
{
m_pos = pos;
m_closeArea = NULL;
m_closeRangeSq = FLT_MAX;
}
bool operator() ( CNavArea *baseArea )
{
CTFNavArea *area = (CTFNavArea *)baseArea;
Vector close;
area->GetClosestPointOnArea( m_pos, &close );
float rangeSq = ( close - m_pos ).LengthSqr();
if ( rangeSq < m_closeRangeSq )
{
m_closeArea = area;
m_closeRangeSq = rangeSq;
}
return true;
}
Vector m_pos;
CTFNavArea *m_closeArea;
float m_closeRangeSq;
};
//-----------------------------------------------------------------------------------------------------
// Update our view to watch where members of the given team will be coming from
void CTFBot::UpdateLookingAroundForIncomingPlayers( bool lookForEnemies )
{
if ( !m_lookAtEnemyInvasionAreasTimer.IsElapsed() )
return;
const float maxLookInterval = 1.0f;
m_lookAtEnemyInvasionAreasTimer.Start( RandomFloat( 0.333f, maxLookInterval ) );
float minGazeRange = m_Shared.InCond( TF_COND_ZOOMED ) ? 750.0f : 150.0f;
CTFNavArea *myArea = GetLastKnownArea();
if ( myArea )
{
int team = GetTeamNumber();
// if we want to look where teammates come from, we need to pass in
// the *enemy* team, since the method collects *enemy* invasion areas
if ( !lookForEnemies )
{
team = GetEnemyTeam( team );
}
const CUtlVector< CTFNavArea * > &invasionAreaVector = myArea->GetEnemyInvasionAreaVector( team );
if ( invasionAreaVector.Count() > 0 )
{
// try to not look directly at walls
const int retryCount = 20.0f;
for( int r=0; r<retryCount; ++r )
{
int which = RandomInt( 0, invasionAreaVector.Count()-1 );
Vector gazeSpot = invasionAreaVector[ which ]->GetRandomPoint() + Vector( 0, 0, 0.75f * HumanHeight );
if ( IsRangeGreaterThan( gazeSpot, minGazeRange ) && GetVisionInterface()->IsLineOfSightClear( gazeSpot ) )
{
// use maxLookInterval so these looks override body aiming from path following
GetBodyInterface()->AimHeadTowards( gazeSpot, IBody::INTERESTING, maxLookInterval, NULL, "Looking toward enemy invasion areas" );
break;
}
}
}
}
}
//-----------------------------------------------------------------------------------------------------
/**
* Update our view to keep an eye on areas where the enemy will be coming from
*/
void CTFBot::UpdateLookingAroundForEnemies( void )
{
if ( !m_isLookingAroundForEnemies )
return;
if ( HasAttribute( CTFBot::IGNORE_ENEMIES ) )
return;
if ( m_Shared.IsControlStunned() )
return;
const float maxLookInterval = 1.0f;
const CKnownEntity *known = GetVisionInterface()->GetPrimaryKnownThreat();
if ( known )
{
if ( known->IsVisibleInFOVNow() )
{
if ( IsPlayerClass( TF_CLASS_SPY ) &&
GetDifficulty() >= CTFBot::HARD &&
m_Shared.InCond( TF_COND_DISGUISED ) &&
!m_Shared.IsStealthed() )
{
// smart Spies don't look at their victims until it's too late...
// look around at where *teammates* will be coming from to fool the enemy
UpdateLookingAroundForIncomingPlayers( LOOK_FOR_FRIENDS );
return;
}
// I see you!
GetBodyInterface()->AimHeadTowards( known->GetEntity(), IBody::CRITICAL, 1.0f, NULL, "Aiming at a visible threat" );
return;
}
/* apparently sounds update last known position...
if ( known->WasEverVisible() && known->GetTimeSinceLastSeen() < 3.0f )
{
// I saw you just a moment ago...
GetBodyInterface()->AimHeadTowards( known->GetLastKnownPosition() + GetClassEyeHeight(), IBody::IMPORTANT, 1.0f, NULL, "Aiming at a last known threat position" );
return;
}
*/
// known but not currently visible (I know you're around here somewhere)
// if there is unobstructed space between us, turn around
if ( IsLineOfSightClear( known->GetEntity(), IGNORE_ACTORS ) )
{
Vector toThreat = known->GetEntity()->GetAbsOrigin() - GetAbsOrigin();
float threatRange = toThreat.NormalizeInPlace();
float aimError = M_PI/6.0f;
float s, c;
FastSinCos( aimError, &s, &c );
float error = threatRange * s;
Vector imperfectAimSpot = known->GetEntity()->WorldSpaceCenter();
imperfectAimSpot.x += RandomFloat( -error, error );
imperfectAimSpot.y += RandomFloat( -error, error );
GetBodyInterface()->AimHeadTowards( imperfectAimSpot, IBody::IMPORTANT, 1.0f, NULL, "Turning around to find threat out of our FOV" );
return;
}
if ( !IsPlayerClass( TF_CLASS_SNIPER ) )
{
// look toward potentially visible area nearest the last known position
CTFNavArea *myArea = GetLastKnownArea();
if ( myArea )
{
const CTFNavArea *closeArea = NULL;
CFindClosestPotentiallyVisibleAreaToPos find( known->GetLastKnownPosition() );
myArea->ForAllPotentiallyVisibleAreas( find );
closeArea = find.m_closeArea;
if ( closeArea )
{
// try to not look directly at walls
const int retryCount = 10.0f;
for( int r=0; r<retryCount; ++r )
{
Vector gazeSpot = closeArea->GetRandomPoint() + Vector( 0, 0, 0.75f * HumanHeight );
if ( GetVisionInterface()->IsLineOfSightClear( gazeSpot ) )
{
// use maxLookInterval so these looks override body aiming from path following
GetBodyInterface()->AimHeadTowards( gazeSpot, IBody::IMPORTANT, maxLookInterval, NULL, "Looking toward potentially visible area near known but hidden threat" );
return;
}
}
// can't find a clear line to look along
if ( IsDebugging( NEXTBOT_VISION | NEXTBOT_ERRORS ) )
{
ConColorMsg( Color( 255, 255, 0, 255 ), "%3.2f: %s can't find clear line to look at potentially visible near known but hidden entity %s(#%d)\n",
gpGlobals->curtime,
GetDebugIdentifier(),
known->GetEntity()->GetClassname(),
known->GetEntity()->entindex() );
}
}
else if ( IsDebugging( NEXTBOT_VISION | NEXTBOT_ERRORS ) )
{
ConColorMsg( Color( 255, 255, 0, 255 ), "%3.2f: %s no potentially visible area to look toward known but hidden entity %s(#%d)\n",
gpGlobals->curtime,
GetDebugIdentifier(),
known->GetEntity()->GetClassname(),
known->GetEntity()->entindex() );
}
}
return;
}
}
// no known threat - look toward where enemies will come from
UpdateLookingAroundForIncomingPlayers( LOOK_FOR_ENEMIES );
}
//---------------------------------------------------------------------------------------------
class CFindVantagePoint : public ISearchSurroundingAreasFunctor
{
public:
CFindVantagePoint( int enemyTeamIndex )
{
m_enemyTeamIndex = enemyTeamIndex;
m_vantageArea = NULL;
}
virtual bool operator() ( CNavArea *baseArea, CNavArea *priorArea, float travelDistanceSoFar )
{
CTFNavArea *area = (CTFNavArea *)baseArea;
CTeam *enemyTeam = GetGlobalTeam( m_enemyTeamIndex );
for( int i=0; i<enemyTeam->GetNumPlayers(); ++i )
{
CTFPlayer *enemy = (CTFPlayer *)enemyTeam->GetPlayer(i);
if ( !enemy->IsAlive() || !enemy->GetLastKnownArea() )
continue;
CTFNavArea *enemyArea = (CTFNavArea *)enemy->GetLastKnownArea();
if ( enemyArea->IsCompletelyVisible( area ) )
{
// nearby area from which we can see the enemy team
m_vantageArea = area;
return false;
}
}
return true;
}
int m_enemyTeamIndex;
CTFNavArea *m_vantageArea;
};
//-----------------------------------------------------------------------------------------------------
// Return a nearby area where we can see a member of the enemy team
CTFNavArea *CTFBot::FindVantagePoint( float maxTravelDistance ) const
{
CFindVantagePoint find( GetTeamNumber() == TF_TEAM_BLUE ? TF_TEAM_RED : TF_TEAM_BLUE );
SearchSurroundingAreas( GetLastKnownArea(), find, maxTravelDistance );
return find.m_vantageArea;
}
//-----------------------------------------------------------------------------------------------------
/**
* Return perceived danger of threat (0=none, 1=immediate deadly danger)
* @todo: Move this to contextual query
* @todo: Differentiate between potential threats (that sentry up ahead along our route) and immediate threats (the sentry I'm in range of)
*/
float CTFBot::GetThreatDanger( CBaseCombatCharacter *who ) const
{
if ( who == NULL )
return 0.0f;
if ( IsPlayerClass( TF_CLASS_SNIPER ) )
{
if ( IsRangeGreaterThan( who, tf_bot_sniper_personal_space_range.GetFloat() ) )
{
// far away enemies are no threat to a Sniper
return 0.0f;
}
}
if ( who->IsPlayer() )
{
CTFPlayer *player = ToTFPlayer( who );
// ubers are scary
if ( player->m_Shared.IsInvulnerable() )
return 1.0f;
switch( player->GetPlayerClass()->GetClassIndex() )
{
case TF_CLASS_MEDIC:
return 0.2f; // 1/5
case TF_CLASS_ENGINEER:
case TF_CLASS_SNIPER:
return 0.4f; // 2/5
case TF_CLASS_SCOUT:
case TF_CLASS_SPY:
case TF_CLASS_DEMOMAN:
return 0.6f; // 3/5
case TF_CLASS_SOLDIER:
case TF_CLASS_HEAVYWEAPONS:
return 0.8f; // 4/5
case TF_CLASS_PYRO:
return 1.0f; // 5/5
}
}
else
{
// sentry gun
CObjectSentrygun *sentry = dynamic_cast< CObjectSentrygun * >( who );
if ( sentry )
{
if ( !sentry->IsAlive() || sentry->IsPlacing() || sentry->HasSapper() || sentry->IsPlasmaDisabled() || sentry->IsUpgrading() || sentry->IsBuilding() )
return 0.0f;
switch( sentry->GetUpgradeLevel() )
{
case 3: return 1.0f;
case 2: return 0.8f;
default: return 0.6f;
}
}
}
return 0.0f;
}
//-----------------------------------------------------------------------------------------------------
/**
* Return the max range at which we can effectively attack
*/
float CTFBot::GetMaxAttackRange( void ) const
{
CTFWeaponBase *myWeapon = m_Shared.GetActiveTFWeapon();
if ( !myWeapon )
return 0.0f;
if ( myWeapon->IsMeleeWeapon() )
{
return 100.0f;
}
if ( myWeapon->IsWeapon( TF_WEAPON_FLAMETHROWER ) )
{
if ( TFGameRules()->IsMannVsMachineMode() )
{
const float flameRange = 350.0f;
static CSchemaItemDefHandle pItemDef_GiantFlamethrower( "MVM Giant Flamethrower" );
if ( IsActiveTFWeapon( pItemDef_GiantFlamethrower ) )
{
return 2.5f * flameRange;
}
return flameRange;
}
return 250.0f;
}
if ( WeaponID_IsSniperRifle( myWeapon->GetWeaponID() ) )
{
// infinite
return FLT_MAX;
}
if ( myWeapon->IsWeapon( TF_WEAPON_ROCKETLAUNCHER ) )
{
return 3000.0f;
}
// bullet spray weapons, grenades, etc
// for now, default to infinite so bot always returns fire and doesn't look dumb
return FLT_MAX;
}
//-----------------------------------------------------------------------------------------------------
/**
* Return the ideal range at which we can effectively attack
*/
float CTFBot::GetDesiredAttackRange( void ) const
{
CTFWeaponBase *myWeapon = m_Shared.GetActiveTFWeapon();
if ( !myWeapon )
return 0.0f;
if ( myWeapon->IsWeapon( TF_WEAPON_KNIFE ) )
{
// get very close and stab
return 70.0f; // 60
}
if ( myWeapon->IsMeleeWeapon() )
{
return 100.0f;
}
if ( myWeapon->IsWeapon( TF_WEAPON_FLAMETHROWER ) )
{
return 100.0f;
}
if ( WeaponID_IsSniperRifle( myWeapon->GetWeaponID() ) )
{
// infinite
return FLT_MAX;
}
if ( myWeapon->IsWeapon( TF_WEAPON_ROCKETLAUNCHER ) && !TFGameRules()->IsMannVsMachineMode() )
{
return 1250.0f;
}
// bullet spray weapons, grenades, etc
return 500.0f;
}
//-----------------------------------------------------------------------------------------------------
// If we're required to equip a specific weapon, do it.
bool CTFBot::EquipRequiredWeapon( void )
{
// if we have a required weapon on our stack, it takes precedence (items, etc)
if ( m_requiredWeaponStack.Count() )
{
CBaseCombatWeapon *pWeapon = m_requiredWeaponStack.Top().Get();
return Weapon_Switch( pWeapon );
}
if ( TheTFBots().IsMeleeOnly() || TFGameRules()->IsInMedievalMode() || HasWeaponRestriction( MELEE_ONLY ) )
{
// force use of melee weapons
Weapon_Switch( Weapon_GetSlot( TF_WPN_TYPE_MELEE ) );
return true;
}
if ( HasWeaponRestriction( PRIMARY_ONLY ) )
{
Weapon_Switch( Weapon_GetSlot( TF_WPN_TYPE_PRIMARY ) );
return true;
}
if ( HasWeaponRestriction( SECONDARY_ONLY ) )
{
Weapon_Switch( Weapon_GetSlot( TF_WPN_TYPE_SECONDARY ) );
return true;
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Equip the best weapon we have to attack the given threat
void CTFBot::EquipBestWeaponForThreat( const CKnownEntity *threat )
{
if ( EquipRequiredWeapon() )
return;
#ifdef TF_RAID_MODE
if ( TFGameRules()->IsRaidMode() )
{
if ( HasAttribute( CTFBot::AGGRESSIVE ) )
{
// mobs never equip other weapons
return;
}
if ( GetPlayerClass()->GetClassIndex() == TF_CLASS_DEMOMAN && !IsInASquad() )
{
// wandering demomen use stickies only
Weapon_Switch( Weapon_GetSlot( TF_WPN_TYPE_SECONDARY ) );
return;
}
}
#endif // TF_RAID_MODE
CTFWeaponBase *primary = dynamic_cast< CTFWeaponBase *>( Weapon_GetSlot( TF_WPN_TYPE_PRIMARY ) );
if ( !IsCombatWeapon( primary ) )
{
primary = NULL;
}
CTFWeaponBase *secondary = dynamic_cast< CTFWeaponBase *>( Weapon_GetSlot( TF_WPN_TYPE_SECONDARY ) );
if ( !IsCombatWeapon( secondary ) )
{
secondary = NULL;
}
// no secondary weapons in MvM
if ( TFGameRules()->IsMannVsMachineMode() )
{
if ( IsPlayerClass( TF_CLASS_MEDIC ) && IsInASquad() && GetSquad() && !GetSquad()->IsLeader( this ) )
{
// always try to heal leader
Weapon_Switch( Weapon_GetSlot( TF_WPN_TYPE_SECONDARY ) );
return;
}
secondary = NULL;
}
CTFWeaponBase *melee = dynamic_cast< CTFWeaponBase *>( Weapon_GetSlot( TF_WPN_TYPE_MELEE ) );
if ( !IsCombatWeapon( melee ) )
{
melee = NULL;
}
CTFWeaponBase *gun = NULL;
if ( primary )
{
gun = primary;
}
else if ( secondary )
{
gun = secondary;
}
else
{
gun = melee;
}
if ( IsDifficulty( CTFBot::EASY ) )
{
// easy bots always use their primary weapon if they have one
if ( gun )
{
Weapon_Switch( gun );
}
return;
}
if ( !threat || !threat->WasEverVisible() || threat->GetTimeSinceLastSeen() > 5.0f )
{
// no threat - go back to primary weapon so it has a chance to reload
if ( gun )
{
Weapon_Switch( gun );
}
return;
}
// now filter weapons by available ammo
if ( GetAmmoCount( TF_AMMO_PRIMARY ) <= 0 )
{
primary = NULL;
}
if ( GetAmmoCount( TF_WPN_TYPE_SECONDARY ) <= 0 )
{
secondary = NULL;
}
// modify our gun choice based on threat situation (range, etc)
switch( GetPlayerClass()->GetClassIndex() )
{
case TF_CLASS_DEMOMAN:
case TF_CLASS_HEAVYWEAPONS:
case TF_CLASS_SPY:
case TF_CLASS_MEDIC:
case TF_CLASS_ENGINEER:
// primary
break;
case TF_CLASS_SCOUT:
{
if ( secondary )
{
if ( gun && !gun->Clip1() )
{
gun = secondary;
}
}
}
break;
case TF_CLASS_SOLDIER:
{
// if we've emptied our rocket launcher clip and are fighting a nearby threat, switch to our secondary if it is ready to fire
if ( gun && !gun->Clip1() )
{
if ( secondary && secondary->Clip1() )
{
const float closeSoldierRange = 500.0f;
if ( IsRangeLessThan( threat->GetLastKnownPosition(), closeSoldierRange ) )
{
gun = secondary;
}
}
}
}
break;
case TF_CLASS_SNIPER:
{
const float closeSniperRange = 750.0f;
if ( secondary && IsRangeLessThan( threat->GetLastKnownPosition(), closeSniperRange ) )
gun = secondary;
}
break;
case TF_CLASS_PYRO:
{
const float flameRange = 750.0f;
if ( secondary && IsRangeGreaterThan( threat->GetLastKnownPosition(), flameRange ) )
{
gun = secondary;
}
// keep flamethrower out to reflect projectiles
if ( threat->GetEntity() && threat->GetEntity()->IsPlayer() )
{
CTFPlayer *enemy = ToTFPlayer( threat->GetEntity() );
if ( enemy->IsPlayerClass( TF_CLASS_SOLDIER ) || enemy->IsPlayerClass( TF_CLASS_DEMOMAN ) )
{
gun = primary;
}
}
}
break;
}
if ( gun )
{
Weapon_Switch( gun );
}
}
//-----------------------------------------------------------------------------------------------------
// NOTE: This assumes default weapon loadouts
bool CTFBot::EquipLongRangeWeapon( void )
{
// no secondary weapons in MvM
if ( TFGameRules()->IsMannVsMachineMode() )
return false;
if ( IsPlayerClass( TF_CLASS_SOLDIER ) ||
IsPlayerClass( TF_CLASS_DEMOMAN ) ||
IsPlayerClass( TF_CLASS_HEAVYWEAPONS ) ||
IsPlayerClass( TF_CLASS_SNIPER ) )
{
CBaseCombatWeapon *primary = Weapon_GetSlot( TF_WPN_TYPE_PRIMARY );
if ( primary )
{
if ( GetAmmoCount( TF_AMMO_PRIMARY ) > 0 )
{
Weapon_Switch( primary );
return true;
}
}
}
// fall back to our secondary (or go right to it if its the only thing we have that has reach)
CBaseCombatWeapon *secondary = Weapon_GetSlot( TF_WPN_TYPE_SECONDARY );
if ( secondary )
{
if ( GetAmmoCount( TF_AMMO_SECONDARY ) > 0 )
{
Weapon_Switch( secondary );
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Force us to equip and use this weapon until popped off the required stack
void CTFBot::PushRequiredWeapon( CTFWeaponBase *weapon )
{
m_requiredWeaponStack.Push( weapon );
}
//-----------------------------------------------------------------------------------------------------
// Pop top required weapon off of stack and discard
void CTFBot::PopRequiredWeapon( void )
{
m_requiredWeaponStack.Pop();
}
//-----------------------------------------------------------------------------------------------------
// return true if given weapon can be used to attack
bool CTFBot::IsCombatWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN ) // MY_CURRENT_GUN == NULL
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_MEDIGUN:
case TF_WEAPON_PDA:
case TF_WEAPON_PDA_ENGINEER_BUILD:
case TF_WEAPON_PDA_ENGINEER_DESTROY:
case TF_WEAPON_PDA_SPY:
case TF_WEAPON_BUILDER:
case TF_WEAPON_DISPENSER:
case TF_WEAPON_INVIS:
case TF_WEAPON_LUNCHBOX:
case TF_WEAPON_BUFF_ITEM:
case TF_WEAPON_PUMPKIN_BOMB:
return false;
};
}
return true;
}
//-----------------------------------------------------------------------------------------------------
// return true if given weapon is a "hitscan" weapon
bool CTFBot::IsHitScanWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN ) // MY_CURRENT_GUN == NULL
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_SHOTGUN_PRIMARY:
case TF_WEAPON_SHOTGUN_SOLDIER:
case TF_WEAPON_SHOTGUN_HWG:
case TF_WEAPON_SHOTGUN_PYRO:
case TF_WEAPON_SCATTERGUN:
case TF_WEAPON_SNIPERRIFLE:
case TF_WEAPON_MINIGUN:
case TF_WEAPON_SMG:
case TF_WEAPON_CHARGED_SMG:
case TF_WEAPON_PISTOL:
case TF_WEAPON_PISTOL_SCOUT:
case TF_WEAPON_REVOLVER:
case TF_WEAPON_SENTRY_BULLET:
case TF_WEAPON_SENTRY_ROCKET:
case TF_WEAPON_SENTRY_REVENGE:
case TF_WEAPON_HANDGUN_SCOUT_PRIMARY:
case TF_WEAPON_HANDGUN_SCOUT_SECONDARY:
case TF_WEAPON_SODA_POPPER:
case TF_WEAPON_SNIPERRIFLE_DECAP:
case TF_WEAPON_PEP_BRAWLER_BLASTER:
case TF_WEAPON_SNIPERRIFLE_CLASSIC:
return true;
};
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// return true if given weapon "sprays" bullets/fire/etc continuously (ie: not individual rockets/etc)
bool CTFBot::IsContinuousFireWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN )
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( !IsCombatWeapon( weapon ) )
return false;
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_ROCKETLAUNCHER:
case TF_WEAPON_ROCKETLAUNCHER_DIRECTHIT:
case TF_WEAPON_GRENADELAUNCHER:
case TF_WEAPON_PIPEBOMBLAUNCHER:
case TF_WEAPON_PISTOL:
case TF_WEAPON_PISTOL_SCOUT:
case TF_WEAPON_FLAREGUN:
case TF_WEAPON_JAR:
case TF_WEAPON_COMPOUND_BOW:
return false;
};
}
return true;
}
//-----------------------------------------------------------------------------------------------------
// return true if given weapon launches explosive projectiles with splash damage
bool CTFBot::IsExplosiveProjectileWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN )
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_ROCKETLAUNCHER:
case TF_WEAPON_ROCKETLAUNCHER_DIRECTHIT:
case TF_WEAPON_GRENADELAUNCHER:
case TF_WEAPON_PIPEBOMBLAUNCHER:
case TF_WEAPON_JAR:
return true;
};
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// return true if given weapon has small clip and long reload cost (ie: rocket launcher, etc)
bool CTFBot::IsBarrageAndReloadWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN )
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_ROCKETLAUNCHER:
case TF_WEAPON_ROCKETLAUNCHER_DIRECTHIT:
case TF_WEAPON_GRENADELAUNCHER:
case TF_WEAPON_PIPEBOMBLAUNCHER:
case TF_WEAPON_SCATTERGUN:
return true;
};
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Return true if given weapon doesn't make much sound when used (ie: spy knife, etc)
bool CTFBot::IsQuietWeapon( CTFWeaponBase *weapon ) const
{
if ( weapon == MY_CURRENT_GUN )
{
weapon = m_Shared.GetActiveTFWeapon();
}
if ( weapon )
{
switch ( weapon->GetWeaponID() )
{
case TF_WEAPON_KNIFE:
case TF_WEAPON_FISTS:
case TF_WEAPON_PDA:
case TF_WEAPON_PDA_ENGINEER_BUILD:
case TF_WEAPON_PDA_ENGINEER_DESTROY:
case TF_WEAPON_PDA_SPY:
case TF_WEAPON_BUILDER:
case TF_WEAPON_MEDIGUN:
case TF_WEAPON_DISPENSER:
case TF_WEAPON_INVIS:
case TF_WEAPON_FLAREGUN:
case TF_WEAPON_LUNCHBOX:
case TF_WEAPON_JAR:
case TF_WEAPON_COMPOUND_BOW:
case TF_WEAPON_SWORD:
case TF_WEAPON_CROSSBOW:
return true;
};
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Return true if a weapon has no obstructions along the line between the given points
bool CTFBot::IsLineOfFireClear( const Vector &from, const Vector &to ) const
{
trace_t trace;
NextBotTraceFilterIgnoreActors botFilter( NULL, COLLISION_GROUP_NONE );
CTraceFilterIgnoreFriendlyCombatItems ignoreFriendlyCombatFilter( this, COLLISION_GROUP_NONE, GetTeamNumber() );
CTraceFilterChain filter( &botFilter, &ignoreFriendlyCombatFilter );
UTIL_TraceLine( from, to, MASK_SOLID_BRUSHONLY, &filter, &trace );
return !trace.DidHit();
}
//-----------------------------------------------------------------------------------------------------
// Return true if a weapon has no obstructions along the line from our eye to the given position
bool CTFBot::IsLineOfFireClear( const Vector &where ) const
{
return IsLineOfFireClear( const_cast< CTFBot * >( this )->EyePosition(), where );
}
//-----------------------------------------------------------------------------------------------------
// Return true if a weapon has no obstructions along the line between the given point and entity
bool CTFBot::IsLineOfFireClear( const Vector &from, CBaseEntity *who ) const
{
trace_t trace;
NextBotTraceFilterIgnoreActors botFilter( NULL, COLLISION_GROUP_NONE );
CTraceFilterIgnoreFriendlyCombatItems ignoreFriendlyCombatFilter( this, COLLISION_GROUP_NONE, GetTeamNumber() );
CTraceFilterChain filter( &botFilter, &ignoreFriendlyCombatFilter );
UTIL_TraceLine( from, who->WorldSpaceCenter(), MASK_SOLID_BRUSHONLY, &filter, &trace );
return !trace.DidHit() || trace.m_pEnt == who;
}
//-----------------------------------------------------------------------------------------------------
// Return true if a weapon has no obstructions along the line from our eye to the given entity
bool CTFBot::IsLineOfFireClear( CBaseEntity *who ) const
{
return IsLineOfFireClear( const_cast< CTFBot * >( this )->EyePosition(), who );
}
//-----------------------------------------------------------------------------------------------------
bool CTFBot::IsEntityBetweenTargetAndSelf( CBaseEntity *other, CBaseEntity *target )
{
Vector toTarget = target->GetAbsOrigin() - GetAbsOrigin();
float rangeToTarget = toTarget.NormalizeInPlace();
Vector toOther = other->GetAbsOrigin() - GetAbsOrigin();
float rangeToOther = toOther.NormalizeInPlace();
return rangeToOther < rangeToTarget && DotProduct( toTarget, toOther ) > 0.7071f;
}
//-----------------------------------------------------------------------------------------------------
// Return true if we are sure this player actually is an enemy spy
bool CTFBot::IsKnownSpy( CTFPlayer *player ) const
{
for( int i=0; i<m_knownSpyVector.Count(); ++i )
{
CTFPlayer *spy = m_knownSpyVector[i];
if ( spy && player->entindex() == spy->entindex() )
{
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Return true if we suspect this player might be an enemy spy
CTFBot::SuspectedSpyInfo_t* CTFBot::IsSuspectedSpy( CTFPlayer *pPlayer )
{
for( int i=0; i<m_suspectedSpyVector.Count(); ++i )
{
SuspectedSpyInfo_t* pSpyInfo = m_suspectedSpyVector[i];
CTFPlayer* pSpy = pSpyInfo->m_suspectedSpy;
if ( pSpy && pPlayer->entindex() == pSpy->entindex() )
{
return pSpyInfo;
}
}
return NULL;
}
//-----------------------------------------------------------------------------------------------------
// Note that this player might be a spy
void CTFBot::SuspectSpy( CTFPlayer *pPlayer )
{
SuspectedSpyInfo_t* pSpyInfo = IsSuspectedSpy( pPlayer );
// Start suspecting this spy if we're not aware of them until now
if( pSpyInfo == NULL )
{
// add to head for LRU effect
pSpyInfo = new SuspectedSpyInfo_t;
pSpyInfo->m_suspectedSpy = pPlayer;
m_suspectedSpyVector.AddToHead( pSpyInfo );
}
// Suspicious!
pSpyInfo->Suspect();
// Too suspicious?
if( pSpyInfo->TestForRealizing() )
{
RealizeSpy( pPlayer );
}
}
void CTFBot::SuspectedSpyInfo_t::Suspect()
{
int nCurTime = floor(gpGlobals->curtime);
// Add our new entry
m_touchTimes.AddToHead( nCurTime );
}
bool CTFBot::SuspectedSpyInfo_t::TestForRealizing()
{
// Remove any old entries
int nCurTime = floor(gpGlobals->curtime);
int nCutoffTime = nCurTime - tf_bot_suspect_spy_touch_interval.GetInt();
FOR_EACH_VEC_BACK( m_touchTimes, i )
{
if( m_touchTimes[i] <= nCutoffTime )
m_touchTimes.Remove( i );
}
// Add our new entry
m_touchTimes.AddToHead( nCurTime );
// Setup an array of bools representing the past few seconds that we want
// to look for suspicious activity
CUtlVector<bool> vecSeconds;
vecSeconds.SetSize( tf_bot_suspect_spy_touch_interval.GetInt() );
FOR_EACH_VEC( vecSeconds, i )
{
vecSeconds[i] = false;
}
// Go through each time chunk and mark if there was suspicious activity
FOR_EACH_VEC( m_touchTimes, i )
{
int nTouchTime = m_touchTimes[i];
int nTimeSlot = nCurTime - nTouchTime;
if( nTimeSlot >= 0 && nTimeSlot < vecSeconds.Count() )
{
vecSeconds[nTimeSlot] = true;
}
}
// If all are true, then the spy has been suspicious enough to warrant being realized
FOR_EACH_VEC( vecSeconds, i )
{
if( vecSeconds[i] == false )
{
return false;
}
}
return true;
}
bool CTFBot::SuspectedSpyInfo_t::IsCurrentlySuspected()
{
float flCutoffTime = gpGlobals->curtime - tf_bot_suspect_spy_forget_cooldown.GetFloat();
if( m_touchTimes.Count() && m_touchTimes.Head() > flCutoffTime )
{
return true;
}
return false;
}
//-----------------------------------------------------------------------------------------------------
// Note that this player *IS* a spy
void CTFBot::RealizeSpy( CTFPlayer *pPlayer )
{
// We already know about this spy
if ( IsKnownSpy( pPlayer ) )
return;
// add to head for LRU effect
m_knownSpyVector.AddToHead( pPlayer );
// inform my teammates
SpeakConceptIfAllowed( MP_CONCEPT_PLAYER_CLOAKEDSPY );
// If I am suspicious of this spy, make everyone around me know that
// they should be suspicious too
SuspectedSpyInfo_t* pSuspectInfo = IsSuspectedSpy( pPlayer );
if( pSuspectInfo && pSuspectInfo->IsCurrentlySuspected() )
{
// Tell others around us we've realized there's a spy
CUtlVector< CTFPlayer * > playerVector;
CollectPlayers( &playerVector, GetTeamNumber(), COLLECT_ONLY_LIVING_PLAYERS );
FOR_EACH_VEC( playerVector, i )
{
CTFPlayer* pOther = playerVector[i];
if( !pOther->IsBot() )
continue;
//Make sure they're close by
Vector vecBetween = EyePosition() - pOther->EyePosition();
if( vecBetween.IsLengthLessThan( 512.f ) )
{
// If they dont know about this spy
CTFBot* pOtherBot = static_cast<CTFBot*>( pOther );
if( !pOtherBot->IsKnownSpy( pPlayer ) )
{
// I was suspicious that they were a spy, make my friend suspicious as well.
// This will cause them to attack a disguised spy in MvM for a bit.
pOtherBot->SuspectSpy( pPlayer );
// Tell them about it
pOtherBot->RealizeSpy( pPlayer );
}
}
}
}
}
//-----------------------------------------------------------------------------------------------------
// Remove player from spy suspect system
void CTFBot::ForgetSpy( CTFPlayer *pPlayer )
{
StopSuspectingSpy( pPlayer );
m_knownSpyVector.FindAndFastRemove( pPlayer );
}
void CTFBot::StopSuspectingSpy( CTFPlayer *pPlayer )
{
// Find the entry matching this spy
for( int i=0; i<m_suspectedSpyVector.Count(); ++i )
{
SuspectedSpyInfo_t* pSpyInfo = m_suspectedSpyVector[i];
CTFPlayer* pSpy = pSpyInfo->m_suspectedSpy;
if ( pSpy && pPlayer->entindex() == pSpy->entindex() )
{
delete pSpyInfo;
m_suspectedSpyVector.Remove(i);
break;
}
}
}
//-----------------------------------------------------------------------------------------------------
// Return the nearest human player on the given team who is looking directly at me
CTFPlayer *CTFBot::GetClosestHumanLookingAtMe( int team ) const
{
CUtlVector< CTFPlayer * > otherVector;
CollectPlayers( &otherVector, team, COLLECT_ONLY_LIVING_PLAYERS );
float closeRange = FLT_MAX;
CTFPlayer *close = NULL;
for( int i=0; i<otherVector.Count(); ++i )
{
CTFPlayer *other = otherVector[i];
if ( other->IsBot() )
continue;
Vector otherEye, otherForward;
other->EyePositionAndVectors( &otherEye, &otherForward, NULL, NULL );
Vector toMe = const_cast< CTFBot * >( this )->EyePosition() - otherEye;
float range = toMe.NormalizeInPlace();
if ( range < closeRange )
{
const float cosTolerance = 0.98f;
if ( DotProduct( toMe, otherForward ) > cosTolerance )
{
// a human is looking toward me - check LOS
if ( IsLineOfSightClear( otherEye, IGNORE_NOTHING, other ) )
{
close = other;
closeRange = range;
}
}
}
}
return close;
}
//-----------------------------------------------------------------------------------------------------
// become a member of the given squad
void CTFBot::JoinSquad( CTFBotSquad *squad )
{
if ( squad )
{
squad->Join( this );
m_squad = squad;
}
}
//-----------------------------------------------------------------------------------------------------
// leave our current squad
void CTFBot::LeaveSquad( void )
{
if ( m_squad )
{
m_squad->Leave( this );
m_squad = NULL;
}
}
//-----------------------------------------------------------------------------------------------------
// leave our current squad
void CTFBot::DeleteSquad( void )
{
if ( m_squad )
{
m_squad = NULL;
}
}
//---------------------------------------------------------------------------------------------
bool CTFBot::IsWeaponRestricted( CTFWeaponBase *weapon ) const
{
if ( !weapon )
{
return false;
}
// Get the weapon's loadout slot
CEconItemView *pEconItemView = weapon->GetAttributeContainer()->GetItem();
if ( !pEconItemView )
return false;
CTFItemDefinition *pItemDef = pEconItemView->GetStaticData();
if ( !pItemDef )
return false;
int iLoadoutSlot = pItemDef->GetLoadoutSlot( GetPlayerClass()->GetClassIndex() );
if ( HasWeaponRestriction( MELEE_ONLY ) )
{
return (iLoadoutSlot != LOADOUT_POSITION_MELEE);
}
if ( HasWeaponRestriction( PRIMARY_ONLY ) )
{
return (iLoadoutSlot != LOADOUT_POSITION_PRIMARY);
}
if ( HasWeaponRestriction( SECONDARY_ONLY ) )
{
return (iLoadoutSlot != LOADOUT_POSITION_SECONDARY);
}
return false;
}
//---------------------------------------------------------------------------------------------
//
// Return true if there is something we want to reflect directly ahead of us
//
bool CTFBot::ShouldFireCompressionBlast( void )
{
if ( TFGameRules()->IsInTraining() )
{
// no reflection in training mode
return false;
}
if ( !tf_bot_pyro_always_reflect.GetBool() )
{
if ( IsDifficulty( CTFBot::EASY ) )
{
// easy bots can't reflect at all
return false;
}
if ( IsDifficulty( CTFBot::NORMAL ) )
{
// normal bots reflect some of the time
if ( TransientlyConsistentRandomValue( 1.0f ) < 0.5f )
{
return false;
}
}
if ( IsDifficulty( CTFBot::HARD ) )
{
// hard bots reflect most of the time
if ( TransientlyConsistentRandomValue( 1.0f ) < 0.1f )
{
return false;
}
}
}
bool shouldPushPlayers = !TFGameRules()->IsMannVsMachineMode();
if ( shouldPushPlayers )
{
const CKnownEntity *threat = GetVisionInterface()->GetPrimaryKnownThreat( true );
if ( threat && threat->GetEntity() && threat->GetEntity()->IsPlayer() )
{
CTFPlayer *pushVictim = ToTFPlayer( threat->GetEntity() );
if ( IsRangeLessThan( pushVictim, tf_bot_pyro_shove_away_range.GetFloat() ) )
{
// our threat is very close - shove them!
// always shove ubers
if ( pushVictim && pushVictim->m_Shared.IsInvulnerable() )
{
return true;
}
if ( pushVictim->GetGroundEntity() == NULL )
{
// they are in the air - juggle them some of the time
return ( TransientlyConsistentRandomValue( 0.5f ) < 0.5f );
}
if ( pushVictim->IsCapturingPoint() )
{
// push them off the point!
return true;
}
// be pushy sometimes
if ( TransientlyConsistentRandomValue( 3.0f ) < 0.5f )
{
return true;
}
}
}
}
Vector vecEye = EyePosition();
Vector vecForward, vecRight, vecUp;
AngleVectors( EyeAngles(), &vecForward, &vecRight, &vecUp );
Vector vecCenter = vecEye + vecForward * 128;
Vector vecSize = Vector( 128, 128, 64 );
const int maxCollectedEntities = 128;
CBaseEntity *pObjects[ maxCollectedEntities ];
int count = UTIL_EntitiesInBox( pObjects, maxCollectedEntities, vecCenter - vecSize, vecCenter + vecSize, FL_CLIENT | FL_GRENADE );
for ( int i = 0; i < count; i++ )
{
CBaseEntity *pObject = pObjects[i];
if ( pObject == this )
continue;
if ( pObject->GetTeamNumber() == GetTeamNumber() )
continue;
// should air blast player logic is already done before this loop
if ( pObject->IsPlayer() )
continue;
// is this something I want to deflect?
if ( !pObject->IsDeflectable() )
continue;
if ( FClassnameIs( pObject, "tf_projectile_rocket" ) || FClassnameIs( pObject, "tf_projectile_energy_ball" ) )
{
// is it headed right for me?
Vector vecThemUnitVel = pObject->GetAbsVelocity();
vecThemUnitVel.z = 0.0f;
vecThemUnitVel.NormalizeInPlace();
Vector horzForward( vecForward.x, vecForward.y, 0.0f );
horzForward.NormalizeInPlace();
if ( DotProduct( horzForward, vecThemUnitVel ) > -tf_bot_pyro_deflect_tolerance.GetFloat() )
continue;
}
// can I see it?
if ( !GetVisionInterface()->IsLineOfSightClear( pObject->WorldSpaceCenter() ) )
continue;
// bounce it!
return true;
}
return false;
}
//---------------------------------------------------------------------------------------------
// Compute a pseudo random value (0-1) that stays consistent for the
// given period of time, but changes unpredictably each period.
float CTFBot::TransientlyConsistentRandomValue( float period, int seedValue ) const
{
CNavArea *area = GetLastKnownArea();
if ( !area )
{
return 0.0f;
}
// this term stays stable for 'period' seconds, then changes in an unpredictable way
int timeMod = (int)( gpGlobals->curtime / period ) + 1;
return fabs( FastCos( (float)( seedValue + ( entindex() * area->GetID() * timeMod ) ) ) );
}
//---------------------------------------------------------------------------------------------
// Given a target entity, find a target within 'maxSplashRadius' that has clear line of fire
// to both the target entity and to me.
bool CTFBot::FindSplashTarget( CBaseEntity *target, float maxSplashRadius, Vector *splashTarget ) const
{
if ( !target || !splashTarget )
return false;
*splashTarget = target->WorldSpaceCenter();
const int retryCount = 50;
for( int i=0; i<retryCount; ++i )
{
Vector probe = target->WorldSpaceCenter() + RandomVector( -maxSplashRadius, maxSplashRadius );
trace_t trace;
NextBotTraceFilterIgnoreActors filter( NULL, COLLISION_GROUP_NONE );
UTIL_TraceLine( target->WorldSpaceCenter(), probe, MASK_SOLID_BRUSHONLY, &filter, &trace );
if ( trace.DidHitWorld() )
{
// can we shoot this spot?
if ( IsLineOfFireClear( trace.endpos ) )
{
// yes, found a corner-sticky target
*splashTarget = trace.endpos;
NDebugOverlay::Line( target->WorldSpaceCenter(), trace.endpos, 255, 0, 0, true, 60.0f );
NDebugOverlay::Cross3D( trace.endpos, 5.0f, 255, 255, 0, true, 60.0f );
return true;
}
}
}
return false;
}
//---------------------------------------------------------------------------------------------
// Restrict bot's attention to only this entity (or radius around this entity) to the exclusion of everything else
void CTFBot::SetAttentionFocus( CBaseEntity *focusOn )
{
m_attentionFocusEntity = focusOn;
}
//---------------------------------------------------------------------------------------------
// Remove attention focus restrictions
void CTFBot::ClearAttentionFocus( void )
{
m_attentionFocusEntity = NULL;
}
//---------------------------------------------------------------------------------------------
bool CTFBot::IsAttentionFocused( void ) const
{
return m_attentionFocusEntity != NULL;
}
//---------------------------------------------------------------------------------------------
bool CTFBot::IsAttentionFocusedOn( CBaseEntity *who ) const
{
if ( m_attentionFocusEntity == NULL || who == NULL )
{
return false;
}
if ( m_attentionFocusEntity->entindex() == who->entindex() )
{
// specifically focused on this entity
return true;
}
CTFBotActionPoint *actionPoint = dynamic_cast< CTFBotActionPoint * >( m_attentionFocusEntity.Get() );
if ( actionPoint )
{
// we attend to everything within the action point's radius
return actionPoint->IsWithinRange( who );
}
return false;
}
//---------------------------------------------------------------------------------------------
// Notice the given threat after the given number of seconds have elapsed
void CTFBot::DelayedThreatNotice( CHandle< CBaseEntity > who, float noticeDelay )
{
float when = gpGlobals->curtime + noticeDelay;
// if we already have a delayed notice for this threat, ignore the new one unless the delay is less
for( int i=0; i<m_delayedNoticeVector.Count(); ++i )
{
if ( m_delayedNoticeVector[i].m_who == who )
{
if ( m_delayedNoticeVector[i].m_when > when )
{
// update delay to shorter time
m_delayedNoticeVector[i].m_when = when;
}
return;
}
}
// new notice
DelayedNoticeInfo delay;
delay.m_who = who;
delay.m_when = when;
m_delayedNoticeVector.AddToTail( delay );
}
//---------------------------------------------------------------------------------------------
void CTFBot::UpdateDelayedThreatNotices( void )
{
for( int i=0; i<m_delayedNoticeVector.Count(); ++i )
{
if ( m_delayedNoticeVector[i].m_when <= gpGlobals->curtime )
{
// delay is up - notice this threat
CBaseEntity *who = m_delayedNoticeVector[i].m_who;
if ( who )
{
if ( who->IsPlayer() )
{
CTFPlayer *player = ToTFPlayer( who );
if ( player->IsPlayerClass( TF_CLASS_SPY ) )
{
RealizeSpy( player );
}
}
GetVisionInterface()->AddKnownEntity( who );
}
m_delayedNoticeVector.Remove( i );
--i;
}
}
}
//---------------------------------------------------------------------------------------------
void CTFBot::GiveRandomItem( loadout_positions_t loadoutPosition )
{
CUtlVector< const CEconItemDefinition * > itemVector;
const CEconItemSchema::ItemDefinitionMap_t& mapItemDefs = ItemSystem()->GetItemSchema()->GetItemDefinitionMap();
FOR_EACH_MAP_FAST( mapItemDefs, i )
{
const CTFItemDefinition *pItemDef = dynamic_cast< const CTFItemDefinition * >( mapItemDefs[i] );
if ( pItemDef && pItemDef->GetLoadoutSlot( GetPlayerClass()->GetClassIndex() ) == loadoutPosition )
{
itemVector.AddToTail( pItemDef );
}
}
if ( itemVector.Count() > 0 )
{
int which = RandomInt( 0, itemVector.Count()-1 );
/*
CBaseCombatWeapon *myMelee = me->Weapon_GetSlot( TF_WPN_TYPE_MELEE );
me->Weapon_Detach( myMelee );
UTIL_Remove( myMelee );
*/
const char *itemName = itemVector[ which ]->GetDefinitionName();
BotGenerateAndWearItem( this, itemName );
}
}
//---------------------------------------------------------------------------------------------
bool CTFBot::IsSquadmate( CTFPlayer *who ) const
{
if ( !m_squad || !who || !who->IsBotOfType( TF_BOT_TYPE ) )
return false;
return GetSquad() == ToTFBot( who )->GetSquad();
}
//---------------------------------------------------------------------------------------------
// Set Spy disguise to be a class that someone on the enemy team is actually using
void CTFBot::DisguiseAsMemberOfEnemyTeam( void )
{
CUtlVector< CTFPlayer * > enemyVector;
CollectPlayers( &enemyVector, GetEnemyTeam( GetTeamNumber() ) );
int disguise = RandomInt( TF_FIRST_NORMAL_CLASS, TF_LAST_NORMAL_CLASS-1 );
if ( enemyVector.Count() > 0 )
{
disguise = enemyVector[ RandomInt( 0, enemyVector.Count()-1 ) ]->GetPlayerClass()->GetClassIndex();
}
m_Shared.Disguise( GetEnemyTeam( GetTeamNumber() ), disguise );
}
//---------------------------------------------------------------------------------------------
void CTFBot::ClearTags( void )
{
m_tags.RemoveAll();
}
//---------------------------------------------------------------------------------------------
void CTFBot::AddTag( const char *tag )
{
if ( !HasTag( tag ) )
{
m_tags.AddToTail( CFmtStr( "%s", tag ) );
}
}
//---------------------------------------------------------------------------------------------
void CTFBot::RemoveTag( const char *tag )
{
for ( int i=0; i<m_tags.Count(); ++i )
{
if ( FStrEq( tag, m_tags[i] ) )
{
m_tags.Remove(i);
return;
}
}
}
//---------------------------------------------------------------------------------------------
// TODO: Make this an efficient lookup/match
bool CTFBot::HasTag( const char *tag )
{
for( int i=0; i<m_tags.Count(); ++i )
{
if ( FStrEq( tag, m_tags[i] ) )
{
return true;
}
}
return false;
}
//---------------------------------------------------------------------------------------------
CBaseObject *CTFBot::GetNearestKnownSappableTarget( void )
{
CUtlVector< CKnownEntity > knownVector;
GetVisionInterface()->CollectKnownEntities( &knownVector );
CBaseObject *closeObject = NULL;
float closeObjectRangeSq = 500.0f * 500.0f;
for( int i=0; i<knownVector.Count(); ++i )
{
CBaseObject *enemyObject = dynamic_cast< CBaseObject * >( knownVector[i].GetEntity() );
if ( enemyObject && !enemyObject->HasSapper() && IsEnemy( enemyObject ) )
{
float rangeSq = GetRangeSquaredTo( enemyObject );
if ( rangeSq < closeObjectRangeSq )
{
closeObjectRangeSq = rangeSq;
closeObject = enemyObject;
}
}
}
return closeObject;
}
//-----------------------------------------------------------------------------------------
Action< CTFBot > *CTFBot::OpportunisticallyUseWeaponAbilities( void )
{
if ( !m_opportunisticTimer.IsElapsed() )
{
return NULL;
}
m_opportunisticTimer.Start( RandomFloat( 0.1f, 0.2f ) );
// if I'm wearing a charge shield, use it!
if ( IsPlayerClass( TF_CLASS_DEMOMAN ) && m_Shared.IsShieldEquipped() )
{
Vector forward;
EyeVectors( &forward );
bool bShouldCharge = GetLocomotionInterface()->IsPotentiallyTraversable( GetAbsOrigin(), GetAbsOrigin() + 100.0f * forward, ILocomotion::IMMEDIATELY );
if ( HasAttribute( CTFBot::AIR_CHARGE_ONLY ) && ( GetGroundEntity() || GetAbsVelocity().z > 0 ) )
{
bShouldCharge = false;
}
if ( bShouldCharge )
{
PressAltFireButton();
}
}
// if I'm wearing parachute, check if I should activate my parachute
else if ( m_Shared.IsParachuteEquipped() )
{
bool bIsBurning = m_Shared.InCond( TF_COND_BURNING );
float flHealthPercent = (float)GetHealth() / GetMaxHealth();
const float flHealthThreshold = 0.5f;
// should I activate parachute?
if ( !m_Shared.InCond( TF_COND_PARACHUTE_DEPLOYED ) )
{
float flMinParachuteGroundDistance = 300.f;
// check if I'm falling, high enough off the ground to deploy parachute, and not burning
if ( flHealthPercent >= flHealthThreshold && !bIsBurning && GetAbsVelocity().z < 0 && GetLocomotionInterface()->IsPotentiallyTraversable( GetAbsOrigin(), GetAbsOrigin() + Vector( 0, 0, -flMinParachuteGroundDistance ), ILocomotion::IMMEDIATELY ) )
{
PressJumpButton();
}
}
// should I deactivate parachute?
else
{
float flCancelParachuteDistance = 150.f;
// if I'm burning or close enough to landing, deactivate the parachute or health less than some threshold
if ( flHealthPercent < flHealthThreshold || bIsBurning || !GetLocomotionInterface()->IsPotentiallyTraversable( GetAbsOrigin(), GetAbsOrigin() + Vector( 0, 0, -flCancelParachuteDistance ), ILocomotion::IMMEDIATELY ) )
{
PressJumpButton();
}
}
}
// don't use items if we have the flag, since most of them are unusable (unless we're a bomb carrier in MvM)
if ( HasTheFlag() && !TFGameRules()->IsMannVsMachineMode() )
{
return NULL;
}
for ( int w=0; w<MAX_WEAPONS; ++w )
{
CTFWeaponBase *weapon = ( CTFWeaponBase * )GetWeapon( w );
if ( !weapon )
continue;
// if I have some kind of buff banner - use it!
if ( weapon->GetWeaponID() == TF_WEAPON_BUFF_ITEM )
{
CTFBuffItem *buff = (CTFBuffItem *)weapon;
if ( buff->IsFull() )
{
return new CTFBotUseItem( buff );
}
}
else if ( weapon->GetWeaponID() == TF_WEAPON_LUNCHBOX )
{
// if we have an eatable (drink, sandvich, etc) - eat it!
CTFLunchBox *lunchbox = (CTFLunchBox *)weapon;
if ( lunchbox->HasAmmo() )
{
// scout lunchboxes are also gated by their energy drink meter
if ( !IsPlayerClass( TF_CLASS_SCOUT ) || m_Shared.GetScoutEnergyDrinkMeter() >= 100 )
{
return new CTFBotUseItem( lunchbox );
}
}
}
else if ( weapon->GetWeaponID() == TF_WEAPON_BAT_WOOD )
{
// sandman
if ( GetAmmoCount( TF_AMMO_GRENADES1 ) > 0 )
{
const CKnownEntity *threat = GetVisionInterface()->GetPrimaryKnownThreat();
if ( threat && threat->IsVisibleInFOVNow() )
{
// hit a stunball
PressAltFireButton();
}
}
}
}
return NULL;
}
//-----------------------------------------------------------------------------------------
// mostly for MvM - pick a random enemy player that is not in their spawn room
CTFPlayer *CTFBot::SelectRandomReachableEnemy( void )
{
CUtlVector< CTFPlayer * > livePlayerVector;
CollectPlayers( &livePlayerVector, GetEnemyTeam( GetTeamNumber() ), COLLECT_ONLY_LIVING_PLAYERS );
// only consider players who have left their spawn
CUtlVector< CTFPlayer * > playerVector;
for( int i=0; i<livePlayerVector.Count(); ++i )
{
CTFPlayer *player = livePlayerVector[i];
if ( !PointInRespawnRoom( player, player->WorldSpaceCenter() ) )
{
playerVector.AddToTail( player );
}
}
if ( playerVector.Count() > 0 )
{
return playerVector[ RandomInt( 0, playerVector.Count()-1 ) ];
}
return NULL;
}
//-----------------------------------------------------------------------------------------
// Different sized bots used different lookahead distances
float CTFBot::GetDesiredPathLookAheadRange( void ) const
{
return tf_bot_path_lookahead_range.GetFloat() * GetModelScale();
}
//-----------------------------------------------------------------------------------------
// Hack to apply idle loop sounds in MvM
void CTFBot::StartIdleSound( void )
{
StopIdleSound();
if ( TFGameRules() && !TFGameRules()->IsMannVsMachineMode() )
return;
// SHIELD YOUR EYES MIKEB!!!
if ( IsMiniBoss() )
{
const char *pszSoundName = NULL;
int iClass = GetPlayerClass()->GetClassIndex();
switch ( iClass )
{
case TF_CLASS_HEAVYWEAPONS:
{
pszSoundName = "MVM.GiantHeavyLoop";
break;
}
case TF_CLASS_SOLDIER:
{
pszSoundName = "MVM.GiantSoldierLoop";
break;
}
case TF_CLASS_DEMOMAN:
{
if ( m_mission == MISSION_DESTROY_SENTRIES )
{
pszSoundName = "MVM.SentryBusterLoop";
}
else
{
pszSoundName = "MVM.GiantDemomanLoop";
}
break;
}
case TF_CLASS_SCOUT:
{
pszSoundName = "MVM.GiantScoutLoop";
break;
}
case TF_CLASS_PYRO:
{
pszSoundName = "MVM.GiantPyroLoop";
break;
}
}
if ( pszSoundName )
{
CReliableBroadcastRecipientFilter filter;
CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
m_pIdleSound = controller.SoundCreate( filter, entindex(), pszSoundName );
controller.Play( m_pIdleSound, 1.0, 100 );
}
}
}
//-----------------------------------------------------------------------------------------
void CTFBot::StopIdleSound( void )
{
if ( m_pIdleSound )
{
CSoundEnvelopeController::GetController().SoundDestroy( m_pIdleSound );
m_pIdleSound = NULL;
}
}
bool CTFBot::ShouldAutoJump()
{
if ( !HasAttribute( CTFBot::AUTO_JUMP ) )
return false;
if ( !m_autoJumpTimer.HasStarted() )
{
m_autoJumpTimer.Start( RandomFloat( m_flAutoJumpMin, m_flAutoJumpMax ) );
return true;
}
else if ( m_autoJumpTimer.IsElapsed() )
{
m_autoJumpTimer.Start( RandomFloat( m_flAutoJumpMin, m_flAutoJumpMax ) );
return true;
}
return false;
}
void CTFBot::SetFlagTarget( CCaptureFlag* pFlag )
{
if ( m_hFollowingFlagTarget != pFlag )
{
if ( m_hFollowingFlagTarget )
{
m_hFollowingFlagTarget->RemoveFollower( this );
}
m_hFollowingFlagTarget = pFlag;
if ( m_hFollowingFlagTarget )
{
m_hFollowingFlagTarget->AddFollower( this );
}
}
}
int CTFBot::DrawDebugTextOverlays(void)
{
int offset = tf_bot_debug_tags.GetBool() ? 1 : BaseClass::DrawDebugTextOverlays();
CUtlString strTags = "Tags : ";
for( int i=0; i<m_tags.Count(); ++i )
{
strTags.Append( m_tags[i] );
strTags.Append( " " );
}
EntityText( offset, strTags.Get(), 0 );
offset++;
return offset;
}
void CTFBot::AddEventChangeAttributes( const CTFBot::EventChangeAttributes_t* newEvent )
{
m_eventChangeAttributes.AddToTail( newEvent );
}
const CTFBot::EventChangeAttributes_t* CTFBot::GetEventChangeAttributes( const char* pszEventName ) const
{
for ( int i=0; i<m_eventChangeAttributes.Count(); ++i )
{
if ( FStrEq( m_eventChangeAttributes[i]->m_eventName, pszEventName ) )
{
return m_eventChangeAttributes[i];
}
}
return NULL;
}
void CTFBot::OnEventChangeAttributes( const CTFBot::EventChangeAttributes_t* pEvent )
{
if ( pEvent )
{
SetDifficulty( pEvent->m_skill );
ClearWeaponRestrictions();
SetWeaponRestriction( pEvent->m_weaponRestriction );
SetMission( pEvent->m_mission );
ClearAllAttributes();
SetAttribute( pEvent->m_attributeFlags );
SetMaxVisionRangeOverride( pEvent->m_maxVisionRange );
if ( TFGameRules()->IsMannVsMachineMode() )
{
SetAttribute( CTFBot::BECOME_SPECTATOR_ON_DEATH );
SetAttribute( CTFBot::RETAIN_BUILDINGS );
}
// cache off health value before we clear attribute because ModifyMaxHealth adds new attribute and reset the health
int nHealth = GetHealth();
int nMaxHealth = GetMaxHealth();
// remove any player attributes
RemovePlayerAttributes( false );
// and add ones that we want specifically
FOR_EACH_VEC( pEvent->m_characterAttributes, i )
{
const CEconItemAttributeDefinition *pDef = pEvent->m_characterAttributes[i].GetAttributeDefinition();
if ( pDef )
{
Assert( GetAttributeList() );
GetAttributeList()->SetRuntimeAttributeValue( pDef, pEvent->m_characterAttributes[i].m_value.asFloat );
}
}
NetworkStateChanged();
// set health back to what it was before we clear bot's attributes
ModifyMaxHealth( nMaxHealth );
SetHealth( nHealth );
// give items to bot before apply attribute changes
FOR_EACH_VEC( pEvent->m_items, i )
{
AddItem( pEvent->m_items[i] );
}
// add attributes to equipped items
FOR_EACH_VEC( pEvent->m_itemsAttributes, i )
{
const CTFBot::EventChangeAttributes_t::item_attributes_t& itemAttributes = pEvent->m_itemsAttributes[i];
CSchemaItemDefHandle itemDef( itemAttributes.m_itemName );
if ( !itemDef )
{
Warning( "Unable to find item %s to update attribute.\n", itemAttributes.m_itemName.Get() );
}
for ( int iItemSlot = LOADOUT_POSITION_PRIMARY ; iItemSlot < CLASS_LOADOUT_POSITION_COUNT ; iItemSlot++ )
{
CEconEntity* pEntity = NULL;
CEconItemView *pCurItemData = CTFPlayerSharedUtils::GetEconItemViewByLoadoutSlot( this, iItemSlot, &pEntity );
if ( pCurItemData && itemDef && ( pCurItemData->GetItemDefIndex() == itemDef->GetDefinitionIndex() ) )
{
for ( int iAtt=0; iAtt<itemAttributes.m_attributes.Count(); ++iAtt )
{
const static_attrib_t& attrib = itemAttributes.m_attributes[iAtt];
CAttributeList *pAttribList = pCurItemData->GetAttributeList();
if ( pAttribList )
{
pAttribList->SetRuntimeAttributeValue( attrib.GetAttributeDefinition(), attrib.m_value.asFloat );
}
}
if ( pEntity )
{
// update model incase we change style
pEntity->UpdateModelToClass();
}
// move on to the next set of attributes
break;
}
} // for each slot
} // for each set of attributes
// tags
ClearTags();
for( int g=0; g<pEvent->m_tags.Count(); ++g )
{
AddTag( pEvent->m_tags[g] );
}
}
}
void CTFBot::AddItem( const char* pszItemName )
{
CItemSelectionCriteria criteria;
criteria.SetQuality( AE_USE_SCRIPT_VALUE );
criteria.BAddCondition( "name", k_EOperator_String_EQ, pszItemName, true );
CBaseEntity *pItem = ItemGeneration()->GenerateRandomItem( &criteria, WorldSpaceCenter(), vec3_angle );
if ( pItem )
{
CEconItemView *pScriptItem = static_cast< CBaseCombatWeapon * >( pItem )->GetAttributeContainer()->GetItem();
// If we already have an item in that slot, remove it
int iClass = GetPlayerClass()->GetClassIndex();
int iSlot = pScriptItem->GetStaticData()->GetLoadoutSlot( iClass );
equip_region_mask_t unNewItemRegionMask = pScriptItem->GetItemDefinition() ? pScriptItem->GetItemDefinition()->GetEquipRegionConflictMask() : 0;
if ( IsWearableSlot( iSlot ) )
{
// Remove any wearable that has a conflicting equip_region
for ( int wbl = 0; wbl < GetNumWearables(); wbl++ )
{
CEconWearable *pWearable = GetWearable( wbl );
if ( !pWearable )
continue;
equip_region_mask_t unWearableRegionMask = 0;
if ( pWearable->GetAttributeContainer()->GetItem() )
{
unWearableRegionMask = pWearable->GetAttributeContainer()->GetItem()->GetItemDefinition()->GetEquipRegionConflictMask();
}
if ( unWearableRegionMask & unNewItemRegionMask )
{
RemoveWearable( pWearable );
}
}
}
else
{
CBaseEntity *pEntity = GetEntityForLoadoutSlot( iSlot );
if ( pEntity )
{
CBaseCombatWeapon *pWpn = dynamic_cast< CBaseCombatWeapon * >( pEntity );
Weapon_Detach( pWpn );
UTIL_Remove( pEntity );
}
}
// Fake global id
pScriptItem->SetItemID( 1 );
DispatchSpawn( pItem );
CEconEntity *pNewItem = assert_cast<CEconEntity*>( pItem );
if ( pNewItem )
{
pNewItem->GiveTo( this );
}
PostInventoryApplication();
}
else
{
if ( pszItemName && pszItemName[0] )
{
DevMsg( "CTFBotSpawner::AddItemToBot: Invalid item %s.\n", pszItemName );
}
}
}
int CTFBot::GetUberHealthThreshold()
{
int iUberHealthThreshold = 0;
CALL_ATTRIB_HOOK_INT( iUberHealthThreshold, bot_medic_uber_health_threshold );
if ( iUberHealthThreshold > 0 )
{
return iUberHealthThreshold;
}
return 50;
}
float CTFBot::GetUberDeployDelayDuration()
{
float flDelayUberDuration = 0;
CALL_ATTRIB_HOOK_INT( flDelayUberDuration, bot_medic_uber_deploy_delay_duration );
if ( flDelayUberDuration > 0 )
{
return flDelayUberDuration;
}
return -1.f;
}