2014-08-04 10:11:56 +00:00
|
|
|
// vim: set ts=4 sw=4 tw=99 noet:
|
|
|
|
//
|
|
|
|
// AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO").
|
|
|
|
// Copyright (C) The AMX Mod X Development Team.
|
|
|
|
//
|
|
|
|
// This software is licensed under the GNU General Public License, version 3 or higher.
|
|
|
|
// Additional exceptions apply. For full license details, see LICENSE.txt or visit:
|
|
|
|
// https://alliedmods.net/amxmodx-license
|
|
|
|
|
|
|
|
//
|
|
|
|
// Fun Module
|
|
|
|
//
|
|
|
|
|
2018-08-30 17:16:19 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <amxxmodule.h>
|
2004-02-11 15:43:26 +00:00
|
|
|
|
|
|
|
// Fun-specific defines below
|
2004-02-18 11:10:54 +00:00
|
|
|
#define GETCLIENTLISTENING (*g_engfuncs.pfnVoice_GetClientListening)
|
|
|
|
#define SETCLIENTLISTENING (*g_engfuncs.pfnVoice_SetClientListening)
|
2004-03-05 14:39:18 +00:00
|
|
|
#define SETCLIENTMAXSPEED (*g_engfuncs.pfnSetClientMaxspeed)
|
2004-03-10 21:58:47 +00:00
|
|
|
#define GETPLAYERAUTHID (*g_engfuncs.pfnGetPlayerAuthId)
|
2004-02-18 11:10:54 +00:00
|
|
|
#define SF_NORESPAWN (1 << 30)// !!!set this bit on guns and stuff that should never respawn.
|
|
|
|
#define STANDARDTIMESTEPSOUND 400
|
2004-02-11 15:43:26 +00:00
|
|
|
|
2004-02-18 11:10:54 +00:00
|
|
|
#define HITGROUP_GENERIC 0 // none
|
2004-03-10 21:58:47 +00:00
|
|
|
#define HITGROUP_HEAD 1 // 1 << 1 = 2
|
|
|
|
#define HITGROUP_CHEST 2 // 1 << 2 = 4
|
|
|
|
#define HITGROUP_STOMACH 3 // 8
|
|
|
|
#define HITGROUP_LEFTARM 4 // 16
|
|
|
|
#define HITGROUP_RIGHTARM 5 // 32
|
|
|
|
#define HITGROUP_LEFTLEG 6 // 64
|
|
|
|
#define HITGROUP_RIGHTLEG 7 // 128
|
2018-08-30 17:16:19 +00:00
|
|
|
#define HITGROUP_MAX 8
|
|
|
|
|
|
|
|
extern DLL_FUNCTIONS *g_pFunctionTable;
|
|
|
|
extern enginefuncs_t *g_pengfuncsTable_Post;
|
|
|
|
|
|
|
|
void PlayerPreThink(edict_t *pEntity);
|
|
|
|
void TraceLine_Post(const float *v1, const float *v2, int fNoMonsters, edict_t *shooter, TraceResult *ptr);
|
|
|
|
|
|
|
|
static const auto kHitGroupsBits = (1 << HITGROUP_MAX) - 1;
|
|
|
|
static const auto kMaxClients = 32u;
|
|
|
|
|
|
|
|
class CPlayer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
CPlayer()
|
|
|
|
{
|
|
|
|
Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
bool HasBodyHits() const
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
if (GetBodyHits(i) != kHitGroupsBits)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GetBodyHits(const int other) const
|
|
|
|
{
|
|
|
|
return bodyHits_[other];
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetBodyHits(const int other, const int flags)
|
|
|
|
{
|
|
|
|
bodyHits_[other] = flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetBodyHits(const int flags)
|
|
|
|
{
|
|
|
|
memset(bodyHits_, flags, sizeof bodyHits_);
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
bool HasSilentFootsteps() const
|
|
|
|
{
|
|
|
|
return silentFootsteps_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetSilentFootsteps(const bool state)
|
|
|
|
{
|
|
|
|
silentFootsteps_ = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
void Clear()
|
|
|
|
{
|
|
|
|
SetBodyHits(kHitGroupsBits);
|
|
|
|
SetSilentFootsteps(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
int bodyHits_[kMaxClients + 1] {};
|
|
|
|
bool silentFootsteps_ {};
|
|
|
|
};
|
|
|
|
|
|
|
|
class CPlayers
|
|
|
|
{
|
|
|
|
using Internal = CPlayer;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
bool HaveBodyHits() const
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
if (players_[i].HasBodyHits())
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetBodyHits(const int attacker, const int target, const int flags)
|
|
|
|
{
|
|
|
|
players_[attacker].SetBodyHits(target, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetTargetsBodyHits(const int attacker, const int flags)
|
|
|
|
{
|
|
|
|
players_[attacker].SetBodyHits(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetAttackersBodyHits(const int target, const int flags)
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
players_[i].SetBodyHits(target, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetEveryoneBodyHits(const int flags)
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
players_[i].SetBodyHits(flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
bool HaveSilentFootsteps() const
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
if (players_[i].HasSilentFootsteps())
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
void Clear()
|
|
|
|
{
|
|
|
|
for (auto i = 1; i <= gpGlobals->maxClients; ++i)
|
|
|
|
{
|
|
|
|
players_[i].Clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Internal& operator [](const size_t index)
|
|
|
|
{
|
|
|
|
return players_[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
const Internal& operator [](const size_t index) const
|
|
|
|
{
|
|
|
|
return players_[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
Internal players_[kMaxClients + 1];
|
|
|
|
};
|
2004-10-03 20:19:47 +00:00
|
|
|
|
|
|
|
#define CHECK_ENTITY(x) \
|
2018-08-30 17:16:19 +00:00
|
|
|
if ((x) < 0 || (x) > gpGlobals->maxEntities) { \
|
2004-10-03 20:19:47 +00:00
|
|
|
MF_LogError(amx, AMX_ERR_NATIVE, "Entity out of range (%d)", x); \
|
2004-10-04 05:51:27 +00:00
|
|
|
return 0; \
|
2004-10-03 20:19:47 +00:00
|
|
|
} else { \
|
2018-08-30 17:16:19 +00:00
|
|
|
if ((x) <= gpGlobals->maxClients) { \
|
2004-10-03 20:19:47 +00:00
|
|
|
if (!MF_IsPlayerIngame(x)) { \
|
|
|
|
MF_LogError(amx, AMX_ERR_NATIVE, "Invalid player %d (not in-game)", x); \
|
2004-10-04 05:51:27 +00:00
|
|
|
return 0; \
|
2004-10-03 20:19:47 +00:00
|
|
|
} \
|
|
|
|
} else { \
|
2018-08-30 17:16:19 +00:00
|
|
|
if ((x) != 0 && FNullEnt(TypeConversion.id_to_edict(x))) { \
|
2004-10-03 20:19:47 +00:00
|
|
|
MF_LogError(amx, AMX_ERR_NATIVE, "Invalid entity %d", x); \
|
2004-10-04 05:51:27 +00:00
|
|
|
return 0; \
|
2004-10-03 20:19:47 +00:00
|
|
|
} \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CHECK_PLAYER(x) \
|
2018-08-30 17:16:19 +00:00
|
|
|
if ((x) < 1 || (x) > gpGlobals->maxClients) { \
|
2004-10-03 20:19:47 +00:00
|
|
|
MF_LogError(amx, AMX_ERR_NATIVE, "Player out of range (%d)", x); \
|
2004-10-04 05:51:27 +00:00
|
|
|
return 0; \
|
2004-10-03 20:19:47 +00:00
|
|
|
} else { \
|
2015-10-06 22:26:14 +00:00
|
|
|
if (!MF_IsPlayerIngame(x) || FNullEnt(TypeConversion.id_to_edict(x))) { \
|
2004-10-03 20:19:47 +00:00
|
|
|
MF_LogError(amx, AMX_ERR_NATIVE, "Invalid player %d", x); \
|
2004-10-04 05:51:27 +00:00
|
|
|
return 0; \
|
2004-10-03 20:19:47 +00:00
|
|
|
} \
|
|
|
|
}
|