ReGameDLL_CS/regamedll/dlls/bot/cs_bot_event.cpp

414 lines
12 KiB
C++
Raw Normal View History

/*
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* In addition, as a special exception, the author gives permission to
* link the code of this program with the Half-Life Game Engine ("HL
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
* L.L.C ("Valve"). You must obey the GNU General Public License in all
* respects for all of the code used other than the HL Engine and MODs
* from Valve. If you modify this file, you may extend this exception
* to your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from your
* version.
*
*/
2015-06-30 15:46:07 +06:00
#include "precompiled.h"
2017-11-23 00:27:55 +07:00
void CCSBot::OnEvent(GameEventType event, CBaseEntity *pEntity, CBaseEntity *pOther)
2015-12-09 04:39:54 +06:00
{
2017-11-23 00:27:55 +07:00
GetGameState()->OnEvent(event, pEntity, pOther);
GetChatter()->OnEvent(event, pEntity, pOther);
2016-01-25 23:02:57 +06:00
// Morale adjustments happen even for dead players
switch (event)
{
case EVENT_TERRORISTS_WIN:
if (m_iTeam == CT)
{
DecreaseMorale();
}
else
{
IncreaseMorale();
}
break;
case EVENT_CTS_WIN:
if (m_iTeam == CT)
{
IncreaseMorale();
}
else
{
DecreaseMorale();
}
break;
2016-01-25 23:02:57 +06:00
}
if (!IsAlive())
return;
2017-11-23 00:27:55 +07:00
CBasePlayer *pPlayer = static_cast<CBasePlayer *>(pEntity);
2016-01-25 23:02:57 +06:00
// If we just saw a nearby friend die, and we haven't yet acquired an enemy
// automatically acquire our dead friend's killer
if (!IsAttacking() && (GetDisposition() == ENGAGE_AND_INVESTIGATE || GetDisposition() == OPPORTUNITY_FIRE))
{
if (event == EVENT_PLAYER_DIED)
{
2017-10-12 21:50:56 +07:00
if (BotRelationship(pPlayer) == BOT_TEAMMATE)
{
2017-11-23 00:27:55 +07:00
CBasePlayer *pKiller = static_cast<CBasePlayer *>(pOther);
// check that attacker is an enemy (for friendly fire, etc)
2017-10-12 21:50:56 +07:00
if (pKiller && pKiller->IsPlayer())
{
2016-01-25 23:02:57 +06:00
// check if we saw our friend die - dont check FOV - assume we're aware of our surroundings in combat
// snipers stay put
2017-10-12 21:50:56 +07:00
if (!IsSniper() && IsVisible(&pPlayer->pev->origin))
{
2016-01-25 23:02:57 +06:00
// people are dying - we should hurry
Hurry(RANDOM_FLOAT(10.0f, 15.0f));
// if we're hiding with only our knife, be a little more cautious
const float knifeAmbushChance = 50.0f;
if (!IsHiding() || !IsUsingKnife() || RANDOM_FLOAT(0, 100) < knifeAmbushChance)
{
PrintIfWatched("Attacking our friend's killer!\n");
2017-10-12 21:50:56 +07:00
Attack(pKiller);
2016-01-25 23:02:57 +06:00
return;
}
}
}
}
}
2016-01-25 23:02:57 +06:00
}
switch (event)
{
2017-11-23 00:27:55 +07:00
case EVENT_PLAYER_DIED:
{
CBasePlayer *pVictim = pPlayer;
CBasePlayer *pKiller = (pOther && pOther->IsPlayer()) ? static_cast<CBasePlayer *>(pOther) : nullptr;
// if the human player died in the single player game, tell the team
if (CSGameRules()->IsCareer() && !pVictim->IsBot() && BotRelationship(pVictim) == BOT_TEAMMATE)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
GetChatter()->Say("CommanderDown", 20.0f);
}
2016-01-25 23:02:57 +06:00
2017-11-23 00:27:55 +07:00
// keep track of the last player we killed
if (pKiller == this)
{
m_lastVictimID = pVictim->entindex();
}
// react to teammate death
if (BotRelationship(pVictim) == BOT_TEAMMATE)
{
// chastise friendly fire from humans
if (pKiller && !pKiller->IsBot() && BotRelationship(pKiller) == BOT_TEAMMATE && pKiller != this)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
GetChatter()->KilledFriend();
2016-01-25 23:02:57 +06:00
}
2017-11-23 00:27:55 +07:00
if (IsHunting())
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
PrintIfWatched("Rethinking hunt due to teammate death\n");
Idle();
return;
2016-01-25 23:02:57 +06:00
}
2017-11-23 00:27:55 +07:00
if (IsAttacking())
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
if (GetTimeSinceLastSawEnemy() > 0.4f)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
PrintIfWatched("Rethinking my attack due to teammate death\n");
2016-01-25 23:02:57 +06:00
2017-11-23 00:27:55 +07:00
// allow us to sneak past windows, doors, etc
IgnoreEnemies(1.0f);
2016-01-25 23:02:57 +06:00
2017-11-23 00:27:55 +07:00
// move to last known position of enemy - this could cause us to flank if
// the danger has changed due to our teammate's recent death
SetTask(MOVE_TO_LAST_KNOWN_ENEMY_POSITION, GetEnemy());
MoveTo(&GetLastKnownEnemyPosition());
return;
2016-01-25 23:02:57 +06:00
}
}
2017-11-23 00:27:55 +07:00
}
// an enemy was killed
else
{
if (pKiller && BotRelationship(pKiller) == BOT_TEAMMATE)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
// only chatter about enemy kills if we see them occur, and they were the last one we see
if (GetNearbyEnemyCount() <= 1)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
// report if number of enemies left is few and we killed the last one we saw locally
GetChatter()->EnemiesRemaining();
2016-01-25 23:02:57 +06:00
2017-11-23 00:27:55 +07:00
if (IsVisible(&pVictim->pev->origin, CHECK_FOV))
{
// congratulate teammates on their kills
if (pKiller != this)
{
2017-11-23 00:27:55 +07:00
float delay = RANDOM_FLOAT(2.0f, 3.0f);
if (pKiller->IsBot())
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
if (RANDOM_FLOAT(0.0f, 100.0f) < 40.0f)
GetChatter()->Say("NiceShot", 3.0f, delay);
}
else
{
// humans get the honorific
if (CSGameRules()->IsCareer())
GetChatter()->Say("NiceShotCommander", 3.0f, delay);
2016-01-25 23:02:57 +06:00
else
2017-11-23 00:27:55 +07:00
GetChatter()->Say("NiceShotSir", 3.0f, delay);
2016-01-25 23:02:57 +06:00
}
}
}
}
}
}
2017-11-23 00:27:55 +07:00
return;
}
case EVENT_TERRORISTS_WIN:
if (m_iTeam == TERRORIST)
GetChatter()->CelebrateWin();
return;
case EVENT_CTS_WIN:
if (m_iTeam == CT)
GetChatter()->CelebrateWin();
return;
case EVENT_BOMB_DEFUSED:
if (m_iTeam == CT && TheCSBots()->GetBombTimeLeft() < 2.0)
GetChatter()->Say("BarelyDefused");
return;
case EVENT_BOMB_PICKED_UP:
{
if (m_iTeam == CT && pPlayer)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
// check if we're close enough to hear it
const float bombPickupHearRangeSq = 1000.0f * 1000.0f;
if ((pev->origin - pPlayer->pev->origin).LengthSquared() < bombPickupHearRangeSq)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
GetChatter()->TheyPickedUpTheBomb();
2016-01-25 23:02:57 +06:00
}
}
2017-11-23 00:27:55 +07:00
return;
}
case EVENT_BOMB_BEEP:
{
// if we don't know where the bomb is, but heard it beep, we've discovered it
if (GetGameState()->IsPlantedBombLocationKnown() == false)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
// check if we're close enough to hear it
const float bombBeepHearRangeSq = 1000.0f * 1000.0f;
if ((pev->origin - pEntity->pev->origin).LengthSquared() < bombBeepHearRangeSq)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
// radio the news to our team
if (m_iTeam == CT && GetGameState()->GetPlantedBombsite() == CSGameState::UNKNOWN)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
const CCSBotManager::Zone *zone = TheCSBots()->GetZone(&pEntity->pev->origin);
if (zone)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
GetChatter()->FoundPlantedBomb(zone->m_index);
2016-01-25 23:02:57 +06:00
}
}
2017-11-23 00:27:55 +07:00
// remember where the bomb is
GetGameState()->UpdatePlantedBomb(&pEntity->pev->origin);
2016-01-25 23:02:57 +06:00
}
}
2017-11-23 00:27:55 +07:00
return;
}
case EVENT_BOMB_PLANTED:
{
// if we're a CT, forget what we're doing and go after the bomb
if (m_iTeam == CT)
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
Idle();
2016-01-25 23:02:57 +06:00
}
2017-11-23 00:27:55 +07:00
// if we are following someone, stop following
if (IsFollowing())
2016-01-25 23:02:57 +06:00
{
2017-11-23 00:27:55 +07:00
StopFollowing();
Idle();
2016-01-25 23:02:57 +06:00
}
2017-11-23 00:27:55 +07:00
OnEvent(EVENT_BOMB_BEEP, pOther);
return;
}
case EVENT_BOMB_DEFUSE_ABORTED:
PrintIfWatched("BOMB DEFUSE ABORTED\n");
return;
case EVENT_WEAPON_FIRED:
case EVENT_WEAPON_FIRED_ON_EMPTY:
case EVENT_WEAPON_RELOADED:
{
if (m_enemy == pEntity && IsUsingKnife())
ForceRun(5.0f);
break;
}
default:
break;
2016-01-25 23:02:57 +06:00
}
// Process radio events from our team
2017-10-12 21:50:56 +07:00
if (pPlayer && BotRelationship(pPlayer) == BOT_TEAMMATE && event > EVENT_START_RADIO_1 && event < EVENT_END_RADIO)
2016-01-25 23:02:57 +06:00
{
// TODO: Distinguish between radio commands and responses
if (event != EVENT_RADIO_AFFIRMATIVE && event != EVENT_RADIO_NEGATIVE && event != EVENT_RADIO_REPORTING_IN)
{
m_lastRadioCommand = event;
m_lastRadioRecievedTimestamp = gpGlobals->time;
2017-10-12 21:50:56 +07:00
m_radioSubject = pPlayer;
m_radioPosition = pPlayer->pev->origin;
2016-01-25 23:02:57 +06:00
}
}
// player_follows needs a player
2017-10-12 21:50:56 +07:00
if (!pPlayer)
2016-01-25 23:02:57 +06:00
return;
if (!IsRogue() && event == EVENT_HOSTAGE_CALLED_FOR_HELP && m_iTeam == CT && IsHunting())
{
2017-11-23 00:27:55 +07:00
if ((pEntity->pev->origin - pev->origin).IsLengthGreaterThan(1000.0f))
2016-01-25 23:02:57 +06:00
return;
2017-11-23 00:27:55 +07:00
if (IsVisible(&pEntity->Center()))
2016-01-25 23:02:57 +06:00
{
m_task = COLLECT_HOSTAGES;
2017-10-12 21:50:56 +07:00
m_taskEntity = nullptr;
2016-01-25 23:02:57 +06:00
Run();
2017-11-23 00:27:55 +07:00
m_goalEntity = pEntity;
2016-01-25 23:02:57 +06:00
2017-11-23 00:27:55 +07:00
MoveTo(&pEntity->pev->origin, m_hostageEscortCount == 0 ? SAFEST_ROUTE : FASTEST_ROUTE);
2016-01-25 23:02:57 +06:00
PrintIfWatched("I'm fetching a hostage that called out to me\n");
return;
}
}
// don't pay attention to noise that friends make
2017-10-12 21:50:56 +07:00
if (!IsEnemy(pPlayer))
2016-01-25 23:02:57 +06:00
return;
float range;
PriorityType priority;
bool isHostile;
2017-11-23 00:27:55 +07:00
if (IsGameEventAudible(event, pEntity, pOther, &range, &priority, &isHostile) == false)
2016-01-25 23:02:57 +06:00
return;
if (event == EVENT_HOSTAGE_USED)
{
if (m_iTeam == CT)
return;
2017-11-23 00:27:55 +07:00
if ((pEntity->pev->origin - pev->origin).IsLengthGreaterThan(range))
2016-01-25 23:02:57 +06:00
return;
GetChatter()->HostagesBeingTaken();
if (!GetGameState()->GetNearestVisibleFreeHostage() && m_task != GUARD_HOSTAGE_RESCUE_ZONE && GuardRandomZone())
{
m_task = GUARD_HOSTAGE_RESCUE_ZONE;
2017-10-12 21:50:56 +07:00
m_taskEntity = nullptr;
2016-01-25 23:02:57 +06:00
SetDisposition(OPPORTUNITY_FIRE);
PrintIfWatched("Trying to beat them to an escape zone!\n");
}
}
// check if noise is close enough for us to hear
2017-10-12 21:50:56 +07:00
const Vector *newNoisePosition = &pPlayer->pev->origin;
2016-01-25 23:02:57 +06:00
float newNoiseDist = (pev->origin - *newNoisePosition).Length();
if (newNoiseDist < range)
{
// we heard the sound
if ((IsLocalPlayerWatchingMe() && cv_bot_debug.value == 3.0f) || cv_bot_debug.value == 4.0f)
{
PrintIfWatched("Heard noise (%s from %s, pri %s, time %3.1f)\n",
(event == EVENT_WEAPON_FIRED) ? "Weapon fire " : "",
2017-10-12 21:50:56 +07:00
STRING(pPlayer->pev->netname),
2016-01-25 23:02:57 +06:00
(priority == PRIORITY_HIGH) ? "HIGH" : ((priority == PRIORITY_MEDIUM) ? "MEDIUM" : "LOW"),
gpGlobals->time);
}
if (event == EVENT_PLAYER_FOOTSTEP && IsUsingSniperRifle() && newNoiseDist < 300.0)
EquipPistol();
// should we pay attention to it
// if noise timestamp is zero, there is no prior noise
if (m_noiseTimestamp > 0.0f)
{
// only overwrite recent sound if we are louder (closer), or more important - if old noise was long ago, its faded
const float shortTermMemoryTime = 3.0f;
if (gpGlobals->time - m_noiseTimestamp < shortTermMemoryTime)
{
// prior noise is more important - ignore new one
if (priority < m_noisePriority)
return;
float oldNoiseDist = (pev->origin - m_noisePosition).Length();
if (newNoiseDist >= oldNoiseDist)
return;
}
}
// find the area in which the noise occured
// TODO: Better handle when noise occurs off the nav mesh
// TODO: Make sure noise area is not through a wall or ceiling from source of noise
// TODO: Change GetNavTravelTime to better deal with NULL destination areas
CNavArea *noiseArea = TheNavAreaGrid.GetNavArea(newNoisePosition);
2017-10-12 21:50:56 +07:00
if (!noiseArea)
2016-01-25 23:02:57 +06:00
noiseArea = TheNavAreaGrid.GetNearestNavArea(newNoisePosition);
2017-10-12 21:50:56 +07:00
if (!noiseArea)
2016-01-25 23:02:57 +06:00
{
PrintIfWatched(" *** Noise occurred off the nav mesh - ignoring!\n");
return;
}
m_noiseArea = noiseArea;
// remember noise priority
m_noisePriority = priority;
// randomize noise position in the area a bit - hearing isn't very accurate
// the closer the noise is, the more accurate our placement
// TODO: Make sure not to pick a position on the opposite side of ourselves.
const float maxErrorRadius = 400.0f;
const float maxHearingRange = 2000.0f;
float errorRadius = maxErrorRadius * newNoiseDist / maxHearingRange;
m_noisePosition.x = newNoisePosition->x + RANDOM_FLOAT(-errorRadius, errorRadius);
m_noisePosition.y = newNoisePosition->y + RANDOM_FLOAT(-errorRadius, errorRadius);
// make sure noise position remains in the same area
m_noiseArea->GetClosestPointOnArea(&m_noisePosition, &m_noisePosition);
m_isNoiseTravelRangeChecked = false;
2017-10-12 21:50:56 +07:00
2016-01-25 23:02:57 +06:00
// note when we heard the noise
m_noiseTimestamp = gpGlobals->time;
}
2015-06-30 15:46:07 +06:00
}