ReGameDLL_CS/regamedll/dlls/bot/cs_bot_statemachine.cpp

427 lines
9.2 KiB
C++
Raw Normal View History

2015-06-30 15:46:07 +06:00
#include "precompiled.h"
// This method is the ONLY legal way to change a bot's current state
void CCSBot::SetState(BotState *state)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
PrintIfWatched("SetState: %s -> %s\n", m_state ? m_state->GetName() : "NULL", state->GetName());
// if we changed state from within the special Attack state, we are no longer attacking
if (m_isAttacking)
StopAttacking();
2017-10-12 21:50:56 +07:00
if (m_state)
m_state->OnExit(this);
state->OnEnter(this);
m_state = state;
m_stateTimestamp = gpGlobals->time;
2015-06-30 15:46:07 +06:00
}
void CCSBot::Idle()
2015-06-30 15:46:07 +06:00
{
SetTask(SEEK_AND_DESTROY);
SetState(&m_idleState);
2015-06-30 15:46:07 +06:00
}
void CCSBot::EscapeFromBomb()
2015-06-30 15:46:07 +06:00
{
SetTask(ESCAPE_FROM_BOMB);
SetState(&m_escapeFromBombState);
2015-06-30 15:46:07 +06:00
}
void CCSBot::Follow(CBasePlayer *player)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
if (!player)
return;
// note when we began following
if (!m_isFollowing || m_leader != player)
m_followTimestamp = gpGlobals->time;
m_isFollowing = true;
m_leader = player;
SetTask(FOLLOW);
m_followState.SetLeader(player);
SetState(&m_followState);
2015-06-30 15:46:07 +06:00
}
// Continue following our leader after finishing what we were doing
void CCSBot::ContinueFollowing()
2015-06-30 15:46:07 +06:00
{
SetTask(FOLLOW);
m_followState.SetLeader(m_leader);
SetState(&m_followState);
2015-06-30 15:46:07 +06:00
}
// Stop following
void CCSBot::StopFollowing()
2015-06-30 15:46:07 +06:00
{
m_isFollowing = false;
2017-10-12 21:50:56 +07:00
m_leader = nullptr;
m_allowAutoFollowTime = gpGlobals->time + 10.0f;
2015-06-30 15:46:07 +06:00
}
// Begin process of rescuing hostages
void CCSBot::RescueHostages()
2015-06-30 15:46:07 +06:00
{
SetTask(RESCUE_HOSTAGES);
2015-06-30 15:46:07 +06:00
}
// Use the entity
void CCSBot::UseEntity(CBaseEntity *entity)
2015-06-30 15:46:07 +06:00
{
m_useEntityState.SetEntity(entity);
SetState(&m_useEntityState);
2015-06-30 15:46:07 +06:00
}
// DEPRECATED: Use TryToHide() instead.
// Move to a hiding place.
// If 'searchFromArea' is non-NULL, hiding spots are looked for from that area first.
void CCSBot::Hide(CNavArea *searchFromArea, float duration, float hideRange, bool holdPosition)
2015-06-30 15:46:07 +06:00
{
DestroyPath();
CNavArea *source;
Vector sourcePos;
if (searchFromArea)
{
source = searchFromArea;
sourcePos = *searchFromArea->GetCenter();
}
else
{
source = m_lastKnownArea;
sourcePos = pev->origin;
}
2017-10-12 21:50:56 +07:00
if (!source)
{
PrintIfWatched("Hide from area is NULL.\n");
Idle();
return;
}
m_hideState.SetSearchArea(source);
m_hideState.SetSearchRange(hideRange);
m_hideState.SetDuration(duration);
m_hideState.SetHoldPosition(holdPosition);
// search around source area for a good hiding spot
Vector useSpot;
const Vector *pos = FindNearbyHidingSpot(this, &sourcePos, source, hideRange, IsSniper());
2017-10-12 21:50:56 +07:00
if (!pos)
{
PrintIfWatched("No available hiding spots.\n");
// hide at our current position
useSpot = pev->origin;
}
else
{
useSpot = *pos;
}
m_hideState.SetHidingSpot(useSpot);
// build a path to our new hiding spot
if (ComputePath(TheNavAreaGrid.GetNavArea(&useSpot), &useSpot, FASTEST_ROUTE) == false)
{
PrintIfWatched("Can't pathfind to hiding spot\n");
Idle();
return;
}
SetState(&m_hideState);
2015-06-30 15:46:07 +06:00
}
// Move to the given hiding place
void CCSBot::Hide(const Vector *hidingSpot, float duration, bool holdPosition)
2015-06-30 15:46:07 +06:00
{
CNavArea *hideArea = TheNavAreaGrid.GetNearestNavArea(hidingSpot);
2017-10-12 21:50:56 +07:00
if (!hideArea)
{
PrintIfWatched("Hiding spot off nav mesh\n");
Idle();
return;
}
DestroyPath();
m_hideState.SetSearchArea(hideArea);
m_hideState.SetSearchRange(750.0f);
m_hideState.SetDuration(duration);
m_hideState.SetHoldPosition(holdPosition);
m_hideState.SetHidingSpot(*hidingSpot);
// build a path to our new hiding spot
if (ComputePath(hideArea, hidingSpot, FASTEST_ROUTE) == false)
{
PrintIfWatched("Can't pathfind to hiding spot\n");
Idle();
return;
}
SetState(&m_hideState);
2015-06-30 15:46:07 +06:00
}
// Try to hide nearby. Return true if hiding, false if can't hide here.
// If 'searchFromArea' is non-NULL, hiding spots are looked for from that area first.
bool CCSBot::TryToHide(CNavArea *searchFromArea, float duration, float hideRange, bool holdPosition, bool useNearest)
2015-06-30 15:46:07 +06:00
{
CNavArea *source;
Vector sourcePos;
if (searchFromArea)
{
source = searchFromArea;
sourcePos = *searchFromArea->GetCenter();
}
else
{
source = m_lastKnownArea;
sourcePos = pev->origin;
}
2017-10-12 21:50:56 +07:00
if (!source)
{
PrintIfWatched("Hide from area is NULL.\n");
return false;
}
m_hideState.SetSearchArea(source);
m_hideState.SetSearchRange(hideRange);
m_hideState.SetDuration(duration);
m_hideState.SetHoldPosition(holdPosition);
// search around source area for a good hiding spot
const Vector *pos = FindNearbyHidingSpot(this, &sourcePos, source, hideRange, IsSniper(), useNearest);
2017-10-12 21:50:56 +07:00
if (!pos)
{
PrintIfWatched("No available hiding spots.\n");
return false;
}
m_hideState.SetHidingSpot(*pos);
// build a path to our new hiding spot
if (ComputePath(TheNavAreaGrid.GetNavArea(pos), pos, FASTEST_ROUTE) == false)
{
PrintIfWatched("Can't pathfind to hiding spot\n");
return false;
}
SetState(&m_hideState);
return true;
2015-06-30 15:46:07 +06:00
}
// Retreat to a nearby hiding spot, away from enemies
bool CCSBot::TryToRetreat()
2015-06-30 15:46:07 +06:00
{
const float maxRange = 1000.0f;
const Vector *spot = FindNearbyRetreatSpot(this, maxRange);
2017-10-12 21:50:56 +07:00
if (spot)
{
// ignore enemies for a second to give us time to hide
// reaching our hiding spot clears our disposition
IgnoreEnemies(10.0f);
float holdTime = RANDOM_FLOAT(3.0f, 15.0f);
StandUp();
Run();
Hide(spot, holdTime);
PrintIfWatched("Retreating to a safe spot!\n");
return true;
}
return false;
2015-06-30 15:46:07 +06:00
}
void CCSBot::Hunt()
2015-06-30 15:46:07 +06:00
{
SetState(&m_huntState);
2015-06-30 15:46:07 +06:00
}
// Attack our the given victim
// NOTE: Attacking does not change our task.
void CCSBot::Attack(CBasePlayer *victim)
2015-06-30 15:46:07 +06:00
{
2017-10-12 21:50:56 +07:00
if (!victim)
2015-06-30 15:46:07 +06:00
return;
// zombies never attack
if (cv_bot_zombie.value != 0.0f)
2015-06-30 15:46:07 +06:00
return;
// cannot attack if we are reloading
if (IsActiveWeaponReloading())
return;
// change enemy
2015-08-20 16:35:01 +06:00
SetEnemy(victim);
2015-06-30 15:46:07 +06:00
// Do not "re-enter" the attack state if we are already attacking
if (IsAttacking())
return;
// if we are currently hiding, increase our chances of crouching and holding position
2015-06-30 15:46:07 +06:00
if (IsAtHidingSpot())
2015-08-20 16:35:01 +06:00
m_attackState.SetCrouchAndHold((RANDOM_FLOAT(0, 100) < 60.0f) != 0);
2015-06-30 15:46:07 +06:00
else
m_attackState.SetCrouchAndHold(false);
2015-06-30 15:46:07 +06:00
PrintIfWatched("ATTACK BEGIN (reaction time = %g (+ update time), surprise time = %g, attack delay = %g)\n"
#ifdef REGAMEDLL_FIXES
, GetProfile()->GetReactionTime(), m_surpriseDelay, GetProfile()->GetAttackDelay()
#endif
);
2015-06-30 15:46:07 +06:00
m_isAttacking = true;
m_attackState.OnEnter(this);
2015-06-30 15:46:07 +06:00
// cheat a bit and give the bot the initial location of its victim
m_lastEnemyPosition = victim->pev->origin;
m_lastSawEnemyTimestamp = gpGlobals->time;
m_aimSpreadTimestamp = gpGlobals->time;
// compute the angle difference between where are looking, and where we need to look
Vector toEnemy = victim->pev->origin - pev->origin;
Vector idealAngle = UTIL_VecToAngles(toEnemy);
2015-06-30 15:46:07 +06:00
float_precision deltaYaw = float_precision(Q_abs(int64(m_lookYaw - idealAngle.y)));
2015-06-30 15:46:07 +06:00
while (deltaYaw > 180.0f)
deltaYaw -= 360.0f;
if (deltaYaw < 0.0f)
deltaYaw = -deltaYaw;
2015-06-30 15:46:07 +06:00
// immediately aim at enemy - accuracy penalty depending on how far we must turn to aim
// accuracy is halved if we have to turn 180 degrees
float turn = deltaYaw * 0.0055555557;/// 180.0f;
2015-06-30 15:46:07 +06:00
float accuracy = GetProfile()->GetSkill() / (1.0f + turn);
2015-08-20 16:35:01 +06:00
SetAimOffset(accuracy);
2015-06-30 15:46:07 +06:00
// define time when aim offset will automatically be updated
// longer time the more we had to turn (surprise)
2015-08-20 16:35:01 +06:00
m_aimOffsetTimestamp = gpGlobals->time + RANDOM_FLOAT(0.25f + turn, 1.5f);
2015-06-30 15:46:07 +06:00
}
// Exit the Attack state
void CCSBot::StopAttacking()
2015-06-30 15:46:07 +06:00
{
PrintIfWatched("ATTACK END\n");
m_attackState.OnExit(this);
m_isAttacking = false;
// if we are following someone, go to the Idle state after the attack to decide whether we still want to follow
if (IsFollowing())
{
Idle();
}
2015-06-30 15:46:07 +06:00
}
bool CCSBot::IsAttacking() const
2015-06-30 15:46:07 +06:00
{
return m_isAttacking;
}
// Return true if we are escaping from the bomb
bool CCSBot::IsEscapingFromBomb() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_escapeFromBombState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Return true if we are defusing the bomb
bool CCSBot::IsDefusingBomb() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_defuseBombState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Return true if we are hiding
bool CCSBot::IsHiding() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_hideState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Return true if we are hiding and at our hiding spot
bool CCSBot::IsAtHidingSpot() const
2015-06-30 15:46:07 +06:00
{
if (!IsHiding())
return false;
return m_hideState.IsAtSpot();
}
// Return true if we are huting
bool CCSBot::IsHunting() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_huntState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Return true if we are in the MoveTo state
bool CCSBot::IsMovingTo() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_moveToState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Return true if we are buying
bool CCSBot::IsBuying() const
2015-06-30 15:46:07 +06:00
{
if (m_state == static_cast<const BotState *>(&m_buyState))
return true;
return false;
2015-06-30 15:46:07 +06:00
}
// Move to potentially distant position
void CCSBot::MoveTo(const Vector *pos, RouteType route)
2015-06-30 15:46:07 +06:00
{
m_moveToState.SetGoalPosition(*pos);
m_moveToState.SetRouteType(route);
SetState(&m_moveToState);
2015-06-30 15:46:07 +06:00
}
void CCSBot::PlantBomb()
2015-06-30 15:46:07 +06:00
{
SetState(&m_plantBombState);
2015-06-30 15:46:07 +06:00
}
// Bomb has been dropped - go get it
void CCSBot::FetchBomb()
2015-06-30 15:46:07 +06:00
{
SetState(&m_fetchBombState);
2015-06-30 15:46:07 +06:00
}
void CCSBot::DefuseBomb()
2015-06-30 15:46:07 +06:00
{
SetState(&m_defuseBombState);
2015-06-30 15:46:07 +06:00
}
// Investigate recent enemy noise
void CCSBot::InvestigateNoise()
2015-06-30 15:46:07 +06:00
{
SetState(&m_investigateNoiseState);
2015-06-30 15:46:07 +06:00
}