bot: stop shoot breakable task in case enemy exists

manager: fix crash due miss-swap of max and min difficulty levels on bot adding (thx @stalin_alex)
This commit is contained in:
jeefo 2025-12-20 00:44:21 +03:00
commit 113cb5e916
No known key found for this signature in database
GPG key ID: D696786B81B667C8
5 changed files with 123 additions and 20 deletions

View file

@ -154,6 +154,12 @@ void Bot::checkBreakable (edict_t *touch) {
if (!game.hasBreakables ()) { if (!game.hasBreakables ()) {
return; return;
} }
const bool hasEnemy = !game.isNullEntity (m_enemy);
// do n ot track for breakables if has some enemies
if (hasEnemy) {
return;
}
if (game.isNullEntity (touch)) { if (game.isNullEntity (touch)) {
auto breakable = lookupBreakable (); auto breakable = lookupBreakable ();

View file

@ -2193,21 +2193,114 @@ BotControl::BotControl () {
m_menuServerFillTeam = 5; m_menuServerFillTeam = 5;
m_printQueueFlushTimestamp = 0.0f; m_printQueueFlushTimestamp = 0.0f;
m_cmds.emplace ("add/addbot/add_ct/addbot_ct/add_t/addbot_t/addhs/addhs_t/addhs_ct", "add [difficulty] [personality] [team] [model] [name]", "Adding specific bot into the game.", &BotControl::cmdAddBot); m_cmds = {
m_cmds.emplace ("kick/kickone/kick_ct/kick_t/kickbot_ct/kickbot_t", "kick [team]", "Kicks off the random bot from the game.", &BotControl::cmdKickBot); {
m_cmds.emplace ("removebots/kickbots/kickall/kickall_ct/kickall_t", "removebots [instant] [team]", "Kicks all the bots from the game.", &BotControl::cmdKickBots); "add/addbot/add_ct/addbot_ct/add_t/addbot_t/addhs/addhs_t/addhs_ct",
m_cmds.emplace ("kill/killbots/killall/kill_ct/kill_t", "kill [team] [silent]", "Kills the specified team / all the bots.", &BotControl::cmdKillBots); "add [difficulty] [personality] [team] [model] [name]",
m_cmds.emplace ("fill/fillserver", "fill [team] [count] [difficulty] [personality]", "Fill the server (add bots) with specified parameters.", &BotControl::cmdFill); "Adding specific bot into the game.",
m_cmds.emplace ("vote/votemap", "vote [map_id]", "Forces all the bot to vote to specified map.", &BotControl::cmdVote);
m_cmds.emplace ("weapons/weaponmode", "weapons [knife|pistol|shotgun|smg|rifle|sniper|standard]", "Sets the bots weapon mode to use.", &BotControl::cmdWeaponMode); &BotControl::cmdAddBot
m_cmds.emplace ("menu/botmenu", "menu [cmd]", "Opens the main bot menu, or command menu if specified.", &BotControl::cmdMenu); },
m_cmds.emplace ("version/ver/about", "version [no arguments]", "Displays version information about bot build.", &BotControl::cmdVersion); {
m_cmds.emplace ("graphmenu/wpmenu/wptmenu", "graphmenu [noarguments]", "Opens and displays bots graph editor.", &BotControl::cmdNodeMenu); "kick/kickone/kick_ct/kick_t/kickbot_ct/kickbot_t",
m_cmds.emplace ("list/listbots", "list [noarguments]", "Lists the bots currently playing on server.", &BotControl::cmdList); "kick [team]",
m_cmds.emplace ("graph/g/w/wp/wpt/waypoint", "graph [help]", "Handles graph operations.", &BotControl::cmdNode); "Kicks off the random bot from the game.",
m_cmds.emplace ("cvars", "cvars [save|save_map|cvar|defaults]", "Display all the cvars with their descriptions.", &BotControl::cmdCvars);
m_cmds.emplace ("show_custom", "show_custom [noarguments]", "Shows the current values from custom.cfg.", &BotControl::cmdShowCustom, false); &BotControl::cmdKickBot
m_cmds.emplace ("exec", "exec [user_id] [command]", "Executes a client command on bot entity.", &BotControl::cmdExec); },
{
"removebots/kickbots/kickall/kickall_ct/kickall_t",
"removebots [instant] [team]",
"Kicks all the bots from the game.",
&BotControl::cmdKickBots
},
{
"kill/killbots/killall/kill_ct/kill_t",
"kill [team] [silent]",
"Kills the specified team / all the bots.",
&BotControl::cmdKillBots
},
{
"fill/fillserver",
"fill [team] [count] [difficulty] [personality]",
"Fill the server (add bots) with specified parameters.",
&BotControl::cmdFill
},
{
"vote/votemap",
"vote [map_id]",
"Forces all the bots to vote for the specified map.",
&BotControl::cmdVote
},
{
"weapons/weaponmode",
"weapons [knife|pistol|shotgun|smg|rifle|sniper|standard]",
"Sets the bots' weapon mode to use.",
&BotControl::cmdWeaponMode
},
{
"menu/botmenu",
"menu [cmd]",
"Opens the main bot menu, or command menu if specified.",
&BotControl::cmdMenu
},
{
"version/ver/about",
"version [no arguments]",
"Displays version information about bot build.",
&BotControl::cmdVersion
},
{
"graphmenu/wpmenu/wptmenu",
"graphmenu [noarguments]",
"Opens and displays bots graph editor.",
&BotControl::cmdNodeMenu
},
{
"list/listbots",
"list [noarguments]",
"Lists the bots currently playing on server.",
&BotControl::cmdList
},
{
"graph/g/w/wp/wpt/waypoint",
"graph [help]",
"Handles graph operations.",
&BotControl::cmdNode
},
{
"cvars",
"cvars [save|save_map|cvar|defaults]",
"Display all the cvars with their descriptions.",
&BotControl::cmdCvars
},
{
"show_custom",
"show_custom [noarguments]",
"Shows the current values from custom.cfg.",
&BotControl::cmdShowCustom,
false
},
{
"exec",
"exec [user_id] [command]",
"Executes a client command on bot entity.",
&BotControl::cmdExec
}
};
// declare the menus // declare the menus
createMenus (); createMenus ();

View file

@ -1472,7 +1472,7 @@ bool Game::isBreakableEntity (edict_t *ent, bool initialSeed) const {
const auto limit = cv_breakable_health_limit.as <float> (); const auto limit = cv_breakable_health_limit.as <float> ();
// not shoot-able // not shoot-able
if (ent->v.health >= limit) { if (ent->v.health < 5 || ent->v.health >= limit) {
return false; return false;
} }
constexpr auto kFuncBreakable = StringRef::fnv1a32 ("func_breakable"); constexpr auto kFuncBreakable = StringRef::fnv1a32 ("func_breakable");

View file

@ -1093,6 +1093,7 @@ void BotManager::updateBotDifficulties () {
void BotManager::balanceBotDifficulties () { void BotManager::balanceBotDifficulties () {
// difficulty changing once per round (time) // difficulty changing once per round (time)
auto updateDifficulty = [] (Bot *bot, int32_t offset) { auto updateDifficulty = [] (Bot *bot, int32_t offset) {
game.print ("offset = %d", offset);
bot->setNewDifficulty (cr::clamp (static_cast <Difficulty> (bot->m_difficulty + offset), Difficulty::Noob, Difficulty::Expert)); bot->setNewDifficulty (cr::clamp (static_cast <Difficulty> (bot->m_difficulty + offset), Difficulty::Noob, Difficulty::Expert));
}; };
@ -1203,7 +1204,7 @@ Bot::Bot (edict_t *bot, int difficulty, int personality, int team, int skin) {
// if we're have min/max difficulty specified, choose value from they // if we're have min/max difficulty specified, choose value from they
if (minDifficulty != Difficulty::Invalid && maxDifficulty != Difficulty::Invalid) { if (minDifficulty != Difficulty::Invalid && maxDifficulty != Difficulty::Invalid) {
if (maxDifficulty > minDifficulty) { if (minDifficulty > maxDifficulty) {
cr::swap (maxDifficulty, minDifficulty); cr::swap (maxDifficulty, minDifficulty);
} }
setNewDifficulty (rg (minDifficulty, maxDifficulty)); setNewDifficulty (rg (minDifficulty, maxDifficulty));
@ -1830,8 +1831,10 @@ void Bot::markStale () {
void Bot::setNewDifficulty (int32_t newDifficulty) { void Bot::setNewDifficulty (int32_t newDifficulty) {
if (newDifficulty < Difficulty::Noob || newDifficulty > Difficulty::Expert) { if (newDifficulty < Difficulty::Noob || newDifficulty > Difficulty::Expert) {
m_difficulty = Difficulty::Hard; const auto difficlutyDefault = Difficulty::Hard;;
m_difficultyData = conf.getDifficultyTweaks (Difficulty::Hard);
m_difficulty = difficlutyDefault;
m_difficultyData = conf.getDifficultyTweaks (difficlutyDefault);
} }
m_difficulty = newDifficulty; m_difficulty = newDifficulty;

View file

@ -1469,9 +1469,10 @@ void Bot::escapeFromBomb_ () {
} }
void Bot::shootBreakable_ () { void Bot::shootBreakable_ () {
const bool hasEnemy = !game.isNullEntity (m_enemy);
// breakable destroyed? // breakable destroyed?
if (!game.isBreakableEntity (m_breakableEntity)) { if (hasEnemy || !game.isBreakableEntity (m_breakableEntity)) {
completeTask (); completeTask ();
return; return;
} }