Vaqtincha 3ff0f0cbf9
Bots can switch weapons underwater (#631)
* Bots can switch weapons underwater
2021-06-06 19:13:55 +07:00

443 lines
12 KiB
C++

/*
*
* 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.
*
*/
#pragma once
#include "gamerules.h"
// 30 times per second, just like human clients
constexpr float g_flBotCommandInterval = 1.0 / 30.0;
// full AI only 10 times per second
constexpr float g_flBotFullThinkInterval = 1.0 / 10.0;
class BotProfile;
template <class T, class TWrap>
T *CreateBot(const BotProfile *profile)
{
edict_t *pentBot;
if (UTIL_ClientsInGame() >= gpGlobals->maxClients)
{
CONSOLE_ECHO("Unable to create bot: Server is full (%d/%d clients).\n", UTIL_ClientsInGame(), gpGlobals->maxClients);
return nullptr;
}
char netname[64];
UTIL_ConstructBotNetName(netname, sizeof(netname), profile);
pentBot = CREATE_FAKE_CLIENT(netname);
if (FNullEnt(pentBot))
{
CONSOLE_ECHO("Unable to create bot: pfnCreateFakeClient() returned null.\n");
return nullptr;
}
else
{
T *pBot = nullptr;
#ifdef REGAMEDLL_FIXES
auto name = pentBot->v.netname;
Q_memset(&pentBot->v, 0, sizeof(pentBot->v)); // Reset entvars data
pentBot->v.netname = name;
pentBot->v.flags = FL_FAKECLIENT | FL_CLIENT;
pentBot->v.pContainingEntity = pentBot;
#endif
FREE_PRIVATE(pentBot);
pBot = GetClassPtr<TWrap>((T *)VARS(pentBot));
pBot->Initialize(profile);
return pBot;
}
}
// The base bot class from which bots for specific games are derived
class CBot: public CBasePlayer
{
public:
// constructor initializes all values to zero
CBot();
virtual void Spawn();
// invoked when injured by something
virtual BOOL TakeDamage(entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType)
{
return CBasePlayer::TakeDamage(pevInflictor, pevAttacker, flDamage, bitsDamageType);
}
// invoked when killed
virtual void Killed(entvars_t *pevAttacker, int iGib)
{
CBasePlayer::Killed(pevAttacker, iGib);
}
virtual void Think() {};
virtual BOOL IsBot() { return TRUE; }
virtual Vector GetAutoaimVector(float flDelta);
// invoked when in contact with a CWeaponBox
virtual void OnTouchingWeapon(CWeaponBox *box) {}
// prepare bot for action
virtual bool Initialize(const BotProfile *profile);
virtual void SpawnBot() = 0;
// lightweight maintenance, invoked frequently
virtual void Upkeep() = 0;
// heavyweight algorithms, invoked less often
virtual void Update() = 0;
virtual void Run();
virtual void Walk();
virtual void Crouch();
virtual void StandUp();
virtual void MoveForward();
virtual void MoveBackward();
virtual void StrafeLeft();
virtual void StrafeRight();
// returns true if jump was started
#define MUST_JUMP true
virtual bool Jump(bool mustJump = false);
// zero any MoveForward(), Jump(), etc
virtual void ClearMovement();
// Weapon interface
virtual void UseEnvironment();
virtual void PrimaryAttack();
virtual void ClearPrimaryAttack();
virtual void TogglePrimaryAttack();
virtual void SecondaryAttack();
virtual void Reload();
// invoked when event occurs in the game (some events have NULL entities)
virtual void OnEvent(GameEventType event, CBaseEntity *pEntity = nullptr, CBaseEntity *pOther = nullptr) {};
// return true if we can see the point
virtual bool IsVisible(const Vector *pos, bool testFOV = false) const = 0;
// return true if we can see any part of the player
virtual bool IsVisible(CBasePlayer *pPlayer, bool testFOV = false, unsigned char *visParts = nullptr) const = 0;
enum VisiblePartType : uint8
{
NONE = 0x00,
CHEST = 0x01,
HEAD = 0x02,
LEFT_SIDE = 0x04, // the left side of the object from our point of view (not their left side)
RIGHT_SIDE = 0x08, // the right side of the object from our point of view (not their right side)
FEET = 0x10
};
// if enemy is visible, return the part we see
virtual bool IsEnemyPartVisible(VisiblePartType part) const = 0;
// return true if player is facing towards us
virtual bool IsPlayerFacingMe(CBasePlayer *pOther) const;
// returns true if other player is pointing right at us
virtual bool IsPlayerLookingAtMe(CBasePlayer *pOther) const;
virtual void ExecuteCommand();
virtual void SetModel(const char *modelName);
public:
// return bot's unique ID
unsigned int GetID() const { return m_id; }
bool IsRunning() const { return m_isRunning; }
bool IsCrouching() const { return m_isCrouching; }
// push the current posture context onto the top of the stack
void PushPostureContext();
// restore the posture context to the next context on the stack
void PopPostureContext();
bool IsJumping();
// return time last jump began
float GetJumpTimestamp() const { return m_jumpTimestamp; }
// returns ratio of ammo left to max ammo (1 = full clip, 0 = empty)
float GetActiveWeaponAmmoRatio() const;
// return true if active weapon has any empty clip
bool IsActiveWeaponClipEmpty() const;
// return true if active weapon has no ammo at all
bool IsActiveWeaponOutOfAmmo() const;
// is the weapon in the middle of a reload
bool IsActiveWeaponReloading() const;
bool IsActiveWeaponCanShootUnderwater() const;
// return true if active weapon's bullet spray has become large and inaccurate
bool IsActiveWeaponRecoilHigh() const;
// return the weapon the bot is currently using
CBasePlayerWeapon *GetActiveWeapon() const;
// return true if looking thru weapon's scope
bool IsUsingScope() const;
// returns TRUE if given entity is our enemy
bool IsEnemy(CBaseEntity *pEntity) const;
// return number of enemies left alive
int GetEnemiesRemaining() const;
// return number of friends left alive
int GetFriendsRemaining() const;
// return true if local player is observing this bot
bool IsLocalPlayerWatchingMe() const;
// output message to console
void Print(char *format,...) const;
// output message to console if we are being watched by the local player
void PrintIfWatched(char *format,...) const;
void BotThink();
#ifdef REGAMEDLL_FIXES
BOOL IsNetClient() { return FALSE; }
#else
// The ambiguous function because there is a virtual function in inherited classes.
bool IsNetClient() const { return false; }
int Save(CSave &save) const;
int Restore(CRestore &restore) const;
#endif
// return our personality profile
const BotProfile *GetProfile() const { return m_profile; }
enum BotRelationshipTeam: uint8
{
BOT_TEAMMATE = 0,
BOT_ENEMY
};
BotRelationshipTeam BotRelationship(CBasePlayer *pTarget) const;
protected:
#ifndef REGAMEDLL_FIXES
// Do a "client command" - useful for invoking menu choices, etc.
void ClientCommand(const char *cmd, const char *arg1 = nullptr, const char *arg2 = nullptr, const char *arg3 = nullptr);
#endif
// the "personality" profile of this bot
const BotProfile *m_profile;
private:
void ResetCommand();
byte ThrottledMsec() const;
// returns current movement speed (for walk/run)
float GetMoveSpeed();
// unique bot ID
unsigned int m_id;
// Think mechanism variables
float m_flNextBotThink;
float m_flNextFullBotThink;
// Command interface variables
float m_flPreviousCommandTime;
// run/walk mode
bool m_isRunning;
// true if crouching (ducking)
bool m_isCrouching;
float m_forwardSpeed;
float m_strafeSpeed;
float m_verticalSpeed;
// bitfield of movement buttons
unsigned short m_buttonFlags;
// time when we last began a jump
float m_jumpTimestamp;
// the PostureContext represents the current settings of walking and crouching
struct PostureContext
{
bool isRunning;
bool isCrouching;
};
enum { MAX_POSTURE_STACK = 8 };
PostureContext m_postureStack[MAX_POSTURE_STACK];
// index of top of stack
int m_postureStackIndex;
};
inline void CBot::SetModel(const char *modelName)
{
SET_CLIENT_KEY_VALUE(entindex(), GET_INFO_BUFFER(edict()), "model", (char *)modelName);
}
inline float CBot::GetMoveSpeed()
{
if (m_isRunning || m_isCrouching)
return pev->maxspeed;
return 0.4f * pev->maxspeed;
}
inline void CBot::Run()
{
m_isRunning = true;
}
inline void CBot::Walk()
{
m_isRunning = false;
}
inline CBasePlayerWeapon *CBot::GetActiveWeapon() const
{
return static_cast<CBasePlayerWeapon *>(m_pActiveItem);
}
inline bool CBot::IsActiveWeaponCanShootUnderwater() const
{
CBasePlayerWeapon *pCurrentWeapon = GetActiveWeapon();
if (pCurrentWeapon && !(pCurrentWeapon->iFlags() & ITEM_FLAG_NOFIREUNDERWATER))
return true;
return false;
}
inline bool CBot::IsActiveWeaponReloading() const
{
CBasePlayerWeapon *pCurrentWeapon = GetActiveWeapon();
if (!pCurrentWeapon)
return false;
return (pCurrentWeapon->m_fInReload || pCurrentWeapon->m_fInSpecialReload) != 0;
}
inline bool CBot::IsActiveWeaponRecoilHigh() const
{
CBasePlayerWeapon *pCurrentWeapon = GetActiveWeapon();
if (pCurrentWeapon)
{
const float highRecoil = 0.4f;
return (pCurrentWeapon->m_flAccuracy > highRecoil) != 0;
}
return false;
}
inline void CBot::PushPostureContext()
{
if (m_postureStackIndex == MAX_POSTURE_STACK)
{
if (pev)
{
PrintIfWatched("PushPostureContext() overflow error!\n");
}
return;
}
m_postureStack[m_postureStackIndex].isRunning = m_isRunning;
m_postureStack[m_postureStackIndex].isCrouching = m_isCrouching;
m_postureStackIndex++;
}
inline void CBot::PopPostureContext()
{
if (m_postureStackIndex == 0)
{
if (pev)
{
PrintIfWatched("PopPostureContext() underflow error!\n");
}
m_isRunning = true;
m_isCrouching = false;
return;
}
m_postureStackIndex--;
m_isRunning = m_postureStack[m_postureStackIndex].isRunning;
m_isCrouching = m_postureStack[m_postureStackIndex].isCrouching;
}
inline bool CBot::IsPlayerFacingMe(CBasePlayer *pOther) const
{
Vector toOther = pOther->pev->origin - pev->origin;
UTIL_MakeVectors(pOther->pev->v_angle + pOther->pev->punchangle);
Vector otherDir = gpGlobals->v_forward;
if (otherDir.x * toOther.x + otherDir.y * toOther.y < 0.0f)
return true;
return false;
}
inline bool CBot::IsPlayerLookingAtMe(CBasePlayer *pOther) const
{
Vector toOther = pOther->pev->origin - pev->origin;
toOther.NormalizeInPlace();
UTIL_MakeVectors(pOther->pev->v_angle + pOther->pev->punchangle);
Vector otherDir = gpGlobals->v_forward;
const float lookAtCos = 0.9f;
if (otherDir.x * toOther.x + otherDir.y * toOther.y < -lookAtCos)
{
Vector vec(pOther->EyePosition());
if (IsVisible(&vec))
return true;
}
return false;
}
inline CBot::BotRelationshipTeam CBot::BotRelationship(CBasePlayer *pTarget) const
{
#ifdef REGAMEDLL_ADD
if (CSGameRules()->IsFreeForAll())
return BOT_ENEMY;
#endif
return pTarget->m_iTeam == m_iTeam ? BOT_TEAMMATE : BOT_ENEMY;
}
extern const char *BotArgs[4];
extern bool UseBotArgs;
extern bool AreBotsAllowed();