Address some parts of review
This commit is contained in:
parent
6687d73f20
commit
cfc540991c
@ -107,8 +107,7 @@ namespace Components
|
|||||||
Loader::Register(new Movement());
|
Loader::Register(new Movement());
|
||||||
Loader::Register(new Elevators());
|
Loader::Register(new Elevators());
|
||||||
Loader::Register(new ClientCommand());
|
Loader::Register(new ClientCommand());
|
||||||
|
Loader::Register(new ScriptExtension());
|
||||||
Loader::Register(new Client());
|
|
||||||
|
|
||||||
Loader::Pregame = false;
|
Loader::Pregame = false;
|
||||||
}
|
}
|
||||||
|
@ -137,4 +137,4 @@ namespace Components
|
|||||||
#include "Modules/ClientCommand.hpp"
|
#include "Modules/ClientCommand.hpp"
|
||||||
|
|
||||||
#include "Modules/Gamepad.hpp"
|
#include "Modules/Gamepad.hpp"
|
||||||
#include "Modules/Client.hpp"
|
#include "Modules/ScriptExtension.hpp"
|
||||||
|
@ -1,74 +1,51 @@
|
|||||||
#include "STDInclude.hpp"
|
#include "STDInclude.hpp"
|
||||||
|
|
||||||
#define KEY_MASK_FIRE 1
|
|
||||||
#define KEY_MASK_SPRINT 2
|
|
||||||
#define KEY_MASK_MELEE 4
|
|
||||||
#define KEY_MASK_RELOAD 16
|
|
||||||
#define KEY_MASK_LEANLEFT 64
|
|
||||||
#define KEY_MASK_LEANRIGHT 128
|
|
||||||
#define KEY_MASK_PRONE 256
|
|
||||||
#define KEY_MASK_CROUCH 512
|
|
||||||
#define KEY_MASK_JUMP 1024
|
|
||||||
#define KEY_MASK_ADS_MODE 2048
|
|
||||||
#define KEY_MASK_TEMP_ACTION 4096
|
|
||||||
#define KEY_MASK_HOLDBREATH 8192
|
|
||||||
#define KEY_MASK_FRAG 16384
|
|
||||||
#define KEY_MASK_SMOKE 32768
|
|
||||||
#define KEY_MASK_NIGHTVISION 262144
|
|
||||||
#define KEY_MASK_ADS 524288
|
|
||||||
#define KEY_MASK_USE 0x28
|
|
||||||
|
|
||||||
#define MAX_G_BOTAI_ENTRIES 18
|
|
||||||
|
|
||||||
namespace Components
|
namespace Components
|
||||||
{
|
{
|
||||||
std::vector<std::string> Bots::BotNames;
|
std::vector<std::string> Bots::BotNames;
|
||||||
|
|
||||||
typedef struct BotMovementInfo_t
|
struct BotMovementInfo_t
|
||||||
{
|
{
|
||||||
/* Actions */
|
int buttons; // Actions
|
||||||
int buttons;
|
|
||||||
/* Movement */
|
|
||||||
int8_t forward;
|
int8_t forward;
|
||||||
int8_t right;
|
int8_t right;
|
||||||
/* Weapon */
|
|
||||||
uint16_t weapon;
|
uint16_t weapon;
|
||||||
} BotMovementInfo_t;
|
};
|
||||||
|
|
||||||
static BotMovementInfo_t g_botai[MAX_G_BOTAI_ENTRIES];
|
static BotMovementInfo_t g_botai[18];
|
||||||
|
|
||||||
struct BotAction_t
|
struct BotAction
|
||||||
{
|
{
|
||||||
const char* action;
|
const char* action;
|
||||||
int key;
|
int key;
|
||||||
};
|
};
|
||||||
|
|
||||||
static const BotAction_t BotActions[] =
|
static const BotAction BotActions[] =
|
||||||
{
|
{
|
||||||
{ "gostand", KEY_MASK_JUMP },
|
{ "gostand", Bots::JUMP },
|
||||||
{ "gocrouch", KEY_MASK_CROUCH },
|
{ "gocrouch", Bots::CROUCH },
|
||||||
{ "goprone", KEY_MASK_PRONE },
|
{ "goprone", Bots::PRONE },
|
||||||
{ "fire", KEY_MASK_FIRE },
|
{ "fire", Bots::FIRE },
|
||||||
{ "melee", KEY_MASK_MELEE },
|
{ "melee", Bots::FIRE},
|
||||||
{ "frag", KEY_MASK_FRAG },
|
{ "frag", Bots::THROW_EQUIPMENT },
|
||||||
{ "smoke", KEY_MASK_SMOKE },
|
{ "smoke", Bots::THROW_SPECIAL },
|
||||||
{ "reload", KEY_MASK_RELOAD },
|
{ "reload", Bots::RELOAD },
|
||||||
{ "sprint", KEY_MASK_SPRINT },
|
{ "sprint", Bots::SPRINT },
|
||||||
{ "leanleft", KEY_MASK_LEANLEFT },
|
{ "leanleft", Bots::LEAN_LEFT},
|
||||||
{ "leanright", KEY_MASK_LEANRIGHT },
|
{ "leanright", Bots::LEAN_RIGHT },
|
||||||
{ "ads", KEY_MASK_ADS_MODE },
|
{ "ads", Bots::ADS_MODE },
|
||||||
{ "holdbreath", KEY_MASK_HOLDBREATH },
|
{ "holdbreath", Bots::HOLD_BREATH },
|
||||||
{ "use", KEY_MASK_USE },
|
{ "use", Bots::USE },
|
||||||
{ "0", 8 },
|
{ "0", Bots::NUM_0 },
|
||||||
{ "1", 32 },
|
{ "1", Bots::NUM_1 },
|
||||||
{ "2", 65536 },
|
{ "2", Bots::NUM_2 },
|
||||||
{ "3", 131072 },
|
{ "3", Bots::NUM_3 },
|
||||||
{ "4", 1048576 },
|
{ "4", Bots::NUM_4},
|
||||||
{ "5", 2097152 },
|
{ "5", Bots::NUM_5 },
|
||||||
{ "6", 4194304 },
|
{ "6", Bots::NUM_6 },
|
||||||
{ "7", 8388608 },
|
{ "7", Bots::NUM_7 },
|
||||||
{ "8", 16777216 },
|
{ "8", Bots::NUM_8 },
|
||||||
{ "9", 33554432 },
|
{ "9", Bots::NUM_9 },
|
||||||
};
|
};
|
||||||
|
|
||||||
void Bots::BuildConnectString(char* buffer, const char* connectString, int num, int, int protocol, int checksum, int statVer, int statStuff, int port)
|
void Bots::BuildConnectString(char* buffer, const char* connectString, int num, int, int protocol, int checksum, int statVer, int statStuff, int port)
|
||||||
@ -110,9 +87,9 @@ namespace Components
|
|||||||
_snprintf_s(buffer, 0x400, _TRUNCATE, connectString, num, botName, protocol, checksum, statVer, statStuff, port);
|
_snprintf_s(buffer, 0x400, _TRUNCATE, connectString, num, botName, protocol, checksum, statVer, statStuff, port);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Bots::Spawn(int count)
|
void Bots::Spawn(unsigned int count)
|
||||||
{
|
{
|
||||||
for (auto i = 0; i < count; ++i)
|
for (auto i = 0u; i < count; ++i)
|
||||||
{
|
{
|
||||||
Scheduler::OnDelay([]()
|
Scheduler::OnDelay([]()
|
||||||
{
|
{
|
||||||
@ -188,6 +165,12 @@ namespace Components
|
|||||||
{
|
{
|
||||||
const auto* weapon = Game::Scr_GetString(0);
|
const auto* weapon = Game::Scr_GetString(0);
|
||||||
|
|
||||||
|
if (weapon == nullptr)
|
||||||
|
{
|
||||||
|
Game::Scr_ParamError(0, "^1BotWeapon: Illegal parameter!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const auto* gentity = Script::GetEntFromEntRef(entref);
|
const auto* gentity = Script::GetEntFromEntRef(entref);
|
||||||
const auto* client = Script::GetClientFromEnt(gentity);
|
const auto* client = Script::GetClientFromEnt(gentity);
|
||||||
|
|
||||||
@ -204,13 +187,19 @@ namespace Components
|
|||||||
}
|
}
|
||||||
|
|
||||||
const auto weapId = Game::G_GetWeaponIndexForName(weapon);
|
const auto weapId = Game::G_GetWeaponIndexForName(weapon);
|
||||||
g_botai[entref.entnum].weapon = static_cast<uint16_t>(weapId);
|
g_botai[entref.entnum].weapon = static_cast<unsigned short>(weapId);
|
||||||
});
|
});
|
||||||
|
|
||||||
Script::AddFunction("BotAction", [](Game::scr_entref_t entref) // Usage: <bot> BotAction(<str action>);
|
Script::AddFunction("BotAction", [](Game::scr_entref_t entref) // Usage: <bot> BotAction(<str action>);
|
||||||
{
|
{
|
||||||
const auto* action = Game::Scr_GetString(0);
|
const auto* action = Game::Scr_GetString(0);
|
||||||
|
|
||||||
|
if (action == nullptr)
|
||||||
|
{
|
||||||
|
Game::Scr_ParamError(0, "^1BotAction: Illegal parameter!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const auto* gentity = Script::GetEntFromEntRef(entref);
|
const auto* gentity = Script::GetEntFromEntRef(entref);
|
||||||
const auto* client = Script::GetClientFromEnt(gentity);
|
const auto* client = Script::GetClientFromEnt(gentity);
|
||||||
|
|
||||||
@ -226,9 +215,9 @@ namespace Components
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto i = 0u; i < sizeof(BotActions) / sizeof(BotAction_t); ++i)
|
for (auto i = 0u; i < std::extent_v<decltype(BotActions)>; ++i)
|
||||||
{
|
{
|
||||||
if (strcmp(&action[1], BotActions[i].action))
|
if (strcmp(&action[1], BotActions[i].action) != 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (action[0] == '+')
|
if (action[0] == '+')
|
||||||
@ -256,17 +245,8 @@ namespace Components
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (forwardInt > 127)
|
forwardInt = std::clamp(forwardInt, -128, 127);
|
||||||
forwardInt = 127;
|
rightInt = std::clamp(rightInt, -128, 127);
|
||||||
|
|
||||||
if (forwardInt < -127)
|
|
||||||
forwardInt = -127;
|
|
||||||
|
|
||||||
if (rightInt > 127)
|
|
||||||
rightInt = 127;
|
|
||||||
|
|
||||||
if (rightInt < -127)
|
|
||||||
rightInt = -127;
|
|
||||||
|
|
||||||
g_botai[entref.entnum].forward = static_cast<int8_t>(forwardInt);
|
g_botai[entref.entnum].forward = static_cast<int8_t>(forwardInt);
|
||||||
g_botai[entref.entnum].right = static_cast<int8_t>(rightInt);
|
g_botai[entref.entnum].right = static_cast<int8_t>(rightInt);
|
||||||
@ -286,10 +266,10 @@ namespace Components
|
|||||||
Utils::Hook(0x627241, 0x4BB9B0, HOOK_CALL).install()->quick();
|
Utils::Hook(0x627241, 0x4BB9B0, HOOK_CALL).install()->quick();
|
||||||
|
|
||||||
// Zero the bot command array
|
// Zero the bot command array
|
||||||
for (auto i = 0; i < MAX_G_BOTAI_ENTRIES; i++)
|
for (auto i = 0u; i < std::extent_v<decltype(g_botai)>; i++)
|
||||||
{
|
{
|
||||||
g_botai[i] = {0};
|
g_botai[i] = {0};
|
||||||
g_botai[i].weapon = 1; // Prevent the bots from defaulting to the 'none' weapon
|
g_botai[i].weapon = 1; // Prevent the bots from defaulting to the 'none' weapon
|
||||||
}
|
}
|
||||||
|
|
||||||
// Have the bots perform the command every server frame
|
// Have the bots perform the command every server frame
|
||||||
@ -325,17 +305,30 @@ namespace Components
|
|||||||
|
|
||||||
Command::Add("spawnBot", [](Command::Params* params)
|
Command::Add("spawnBot", [](Command::Params* params)
|
||||||
{
|
{
|
||||||
auto count = 1;
|
auto count = 1u;
|
||||||
|
|
||||||
if (params->length() > 1)
|
if (params->length() > 1)
|
||||||
{
|
{
|
||||||
if (params->get(1) == "all"s)
|
if (params->get(1) == "all"s)
|
||||||
|
{
|
||||||
count = *Game::svs_numclients;
|
count = *Game::svs_numclients;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
count = std::atoi(params->get(1));
|
{
|
||||||
|
char* endptr;
|
||||||
|
const auto* input = params->get(1);
|
||||||
|
count = std::strtoul(input, &endptr, 10);
|
||||||
|
|
||||||
|
if (input == endptr)
|
||||||
|
{
|
||||||
|
Logger::Print("Warning: %s is not a valid input\n"
|
||||||
|
"Usage: %s optional <number of bots> or optional <all>\n",
|
||||||
|
input, params->get(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
count = std::min(*Game::svs_numclients, count);
|
count = std::min(static_cast<unsigned int>(*Game::svs_numclients), count);
|
||||||
|
|
||||||
// Check if ingame and host
|
// Check if ingame and host
|
||||||
if (!Game::SV_Loaded())
|
if (!Game::SV_Loaded())
|
||||||
@ -353,9 +346,4 @@ namespace Components
|
|||||||
|
|
||||||
Bots::AddMethods();
|
Bots::AddMethods();
|
||||||
}
|
}
|
||||||
|
|
||||||
Bots::~Bots()
|
|
||||||
{
|
|
||||||
Bots::BotNames.clear();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -6,14 +6,46 @@ namespace Components
|
|||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Bots();
|
Bots();
|
||||||
~Bots();
|
|
||||||
|
enum PlayerKeyFlag
|
||||||
|
{
|
||||||
|
FIRE = 0x1,
|
||||||
|
SPRINT = 0x2,
|
||||||
|
MELEE = 0x4,
|
||||||
|
RELOAD = 0x10,
|
||||||
|
LEAN_LEFT = 0x40,
|
||||||
|
LEAN_RIGHT = 0x80,
|
||||||
|
PRONE = 0x100,
|
||||||
|
CROUCH = 0x200,
|
||||||
|
JUMP = 0x400,
|
||||||
|
ADS_MODE = 0x800,
|
||||||
|
TEMP_ACTION = 0x1000,
|
||||||
|
HOLD_BREATH = 0x2000,
|
||||||
|
THROW_EQUIPMENT = 0x4000,
|
||||||
|
THROW_SPECIAL = 0x8000,
|
||||||
|
NIGHT_VISION = 0x40000,
|
||||||
|
ADS = 0x80000,
|
||||||
|
|
||||||
|
NUM_0 = 0x8,
|
||||||
|
NUM_1 = 0x20,
|
||||||
|
NUM_2 = 0x10000,
|
||||||
|
NUM_3 = 0x20000,
|
||||||
|
NUM_4 = 0x100000,
|
||||||
|
NUM_5 = 0x200000,
|
||||||
|
NUM_6 = 0x400000,
|
||||||
|
NUM_7 = 0x800000,
|
||||||
|
NUM_8 = 0x1000000,
|
||||||
|
NUM_9 = 0x2000000,
|
||||||
|
|
||||||
|
USE = 0x28,
|
||||||
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static std::vector<std::string> BotNames;
|
static std::vector<std::string> BotNames;
|
||||||
|
|
||||||
static void BuildConnectString(char* buffer, const char* connectString, int num, int, int protocol, int checksum, int statVer, int statStuff, int port);
|
static void BuildConnectString(char* buffer, const char* connectString, int num, int, int protocol, int checksum, int statVer, int statStuff, int port);
|
||||||
|
|
||||||
static void Spawn(int count);
|
static void Spawn(unsigned int count);
|
||||||
|
|
||||||
static void AddMethods();
|
static void AddMethods();
|
||||||
};
|
};
|
||||||
|
@ -150,7 +150,7 @@ namespace Components
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Game::Scr_GetNumParam() == 1u)
|
if (Game::Scr_GetNumParam() >= 1u)
|
||||||
{
|
{
|
||||||
if (Game::Scr_GetInt(0))
|
if (Game::Scr_GetInt(0))
|
||||||
{
|
{
|
||||||
@ -177,7 +177,7 @@ namespace Components
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Game::Scr_GetNumParam() == 1u)
|
if (Game::Scr_GetNumParam() >= 1u)
|
||||||
{
|
{
|
||||||
if (Game::Scr_GetInt(0))
|
if (Game::Scr_GetInt(0))
|
||||||
{
|
{
|
||||||
@ -198,7 +198,7 @@ namespace Components
|
|||||||
{
|
{
|
||||||
auto* ent = Script::GetEntFromEntRef(entref);
|
auto* ent = Script::GetEntFromEntRef(entref);
|
||||||
|
|
||||||
if (Game::Scr_GetNumParam() == 1u)
|
if (Game::Scr_GetNumParam() >= 1u)
|
||||||
{
|
{
|
||||||
if (Game::Scr_GetInt(0))
|
if (Game::Scr_GetInt(0))
|
||||||
{
|
{
|
||||||
@ -219,7 +219,7 @@ namespace Components
|
|||||||
{
|
{
|
||||||
auto* ent = Script::GetEntFromEntRef(entref);
|
auto* ent = Script::GetEntFromEntRef(entref);
|
||||||
|
|
||||||
if (Game::Scr_GetNumParam() == 1u)
|
if (Game::Scr_GetNumParam() >= 1u)
|
||||||
{
|
{
|
||||||
if (Game::Scr_GetInt(0))
|
if (Game::Scr_GetInt(0))
|
||||||
{
|
{
|
||||||
@ -240,7 +240,7 @@ namespace Components
|
|||||||
{
|
{
|
||||||
auto* ent = Script::GetEntFromEntRef(entref);
|
auto* ent = Script::GetEntFromEntRef(entref);
|
||||||
|
|
||||||
if (Game::Scr_GetNumParam() == 1u)
|
if (Game::Scr_GetNumParam() >= 1u)
|
||||||
{
|
{
|
||||||
if (Game::Scr_GetInt(0))
|
if (Game::Scr_GetInt(0))
|
||||||
{
|
{
|
||||||
|
@ -964,13 +964,20 @@ namespace Components
|
|||||||
Download::ScriptDownloads.clear();
|
Download::ScriptDownloads.clear();
|
||||||
});
|
});
|
||||||
|
|
||||||
Script::AddFunction("httpGet", [](Game::scr_entref_t)
|
Script::AddFunction("HttpGet", [](Game::scr_entref_t)
|
||||||
{
|
{
|
||||||
if (!Dedicated::IsEnabled() && !Flags::HasFlag("scriptablehttp")) return;
|
if (!Dedicated::IsEnabled() && !Flags::HasFlag("scriptablehttp")) return;
|
||||||
if (Game::Scr_GetNumParam() < 1u) return;
|
if (Game::Scr_GetNumParam() < 1u) return;
|
||||||
|
|
||||||
std::string url = Game::Scr_GetString(0);
|
std::string url = Game::Scr_GetString(0);
|
||||||
unsigned int object = Game::AllocObject();
|
|
||||||
|
if (url.empty())
|
||||||
|
{
|
||||||
|
Game::Scr_ParamError(0, "^1HttpGet: Illegal parameters!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto object = Game::AllocObject();
|
||||||
|
|
||||||
Game::Scr_AddObject(object);
|
Game::Scr_AddObject(object);
|
||||||
|
|
||||||
|
@ -50,24 +50,25 @@ namespace Components
|
|||||||
const auto developer = Dvar::Var("developer").get<int>();
|
const auto developer = Dvar::Var("developer").get<int>();
|
||||||
|
|
||||||
// Allow error messages to be printed if developer mode is on
|
// Allow error messages to be printed if developer mode is on
|
||||||
// Should check scrVarPub.developer but it's absent in this version of the game
|
// Should check scrVarPub.developer but it's absent
|
||||||
|
// in this version of the game so let's check the dvar
|
||||||
if (!Game::scrVmPub->terminal_error && !developer)
|
if (!Game::scrVmPub->terminal_error && !developer)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// If were are developing let's just print a brief message
|
// If were are developing let's call RuntimeErrorInternal
|
||||||
// scrVmPub.debugCode seems to be always false
|
// scrVmPub.debugCode seems to be always false
|
||||||
if (Game::scrVmPub->debugCode || Game::scrVarPub->developer_script)
|
if (Game::scrVmPub->debugCode || Game::scrVarPub->developer_script)
|
||||||
{
|
{
|
||||||
Logger::Print(23, "%s\n", msg);
|
Game::RuntimeErrorInternal(23, codePos, index, msg);
|
||||||
|
|
||||||
if (!Game::scrVmPub->terminal_error)
|
if (!Game::scrVmPub->terminal_error)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Game::RuntimeErrorInternal(23, codePos, index, msg);
|
Logger::Print(23, "%s\n", msg);
|
||||||
// Let's not throw error unless we have to
|
// Let's not throw error unless we have to
|
||||||
if (!Game::scrVmPub->abort_on_error && !Game::scrVmPub->terminal_error)
|
if (Game::scrVmPub->abort_on_error && !Game::scrVmPub->terminal_error)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -522,14 +523,12 @@ namespace Components
|
|||||||
|
|
||||||
Game::gentity_t* Script::GetEntFromEntRef(const Game::scr_entref_t entref)
|
Game::gentity_t* Script::GetEntFromEntRef(const Game::scr_entref_t entref)
|
||||||
{
|
{
|
||||||
if (entref.classnum != 0)
|
if (entref.classnum != 0 || entref.entnum >= Game::MAX_GENTITIES)
|
||||||
{
|
{
|
||||||
Game::Scr_ObjectError("Not an entity");
|
Game::Scr_ObjectError("Not an entity");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(entref.entnum < Game::MAX_GENTITIES);
|
|
||||||
|
|
||||||
return &Game::g_entities[entref.entnum];
|
return &Game::g_entities[entref.entnum];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -541,6 +540,12 @@ namespace Components
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (gentity->s.number >= *Game::svs_numclients)
|
||||||
|
{
|
||||||
|
Game::Scr_ObjectError(Utils::String::VA("Entity %i is out of bounds", gentity->s.number));
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
return &Game::svs_clients[gentity->s.number];
|
return &Game::svs_clients[gentity->s.number];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -580,14 +585,28 @@ namespace Components
|
|||||||
// Print to console, even without being in 'developer 1'.
|
// Print to console, even without being in 'developer 1'.
|
||||||
Script::AddFunction("PrintConsole", [](Game::scr_entref_t) // gsc: PrintConsole(<string>)
|
Script::AddFunction("PrintConsole", [](Game::scr_entref_t) // gsc: PrintConsole(<string>)
|
||||||
{
|
{
|
||||||
const auto str = Game::Scr_GetString(0);
|
const auto* str = Game::Scr_GetString(0);
|
||||||
Game::Com_Printf(0, str);
|
|
||||||
|
if (str == nullptr)
|
||||||
|
{
|
||||||
|
Game::Scr_ParamError(0, "^1PrintConsole: Illegal parameters!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger::Print(0, "%s", str);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Executes command to the console
|
// Executes command to the console
|
||||||
Script::AddFunction("Exec", [](Game::scr_entref_t) // gsc: Exec(<string>)
|
Script::AddFunction("Exec", [](Game::scr_entref_t) // gsc: Exec(<string>)
|
||||||
{
|
{
|
||||||
const auto str = Game::Scr_GetString(0);
|
const auto str = Game::Scr_GetString(0);
|
||||||
|
|
||||||
|
if (str == nullptr)
|
||||||
|
{
|
||||||
|
Game::Scr_ParamError(0, "^1Exec: Illegal parameters!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
Command::Execute(str, false);
|
Command::Execute(str, false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -51,7 +51,7 @@ namespace Components
|
|||||||
|
|
||||||
static void FunctionError();
|
static void FunctionError();
|
||||||
static void StoreFunctionNameStub();
|
static void StoreFunctionNameStub();
|
||||||
static void RuntimeError(const char*, unsigned int, const char*, const char*);
|
static void RuntimeError(const char* codePos, unsigned int index, const char* msg, const char* dialogMessage);
|
||||||
|
|
||||||
static void StoreScriptName(const char* name);
|
static void StoreScriptName(const char* name);
|
||||||
static void StoreScriptNameStub();
|
static void StoreScriptNameStub();
|
||||||
|
@ -2,19 +2,25 @@
|
|||||||
|
|
||||||
namespace Components
|
namespace Components
|
||||||
{
|
{
|
||||||
void Client::AddFunctions()
|
void ScriptExtension::AddFunctions()
|
||||||
{
|
{
|
||||||
//File functions
|
//File functions
|
||||||
|
|
||||||
Script::AddFunction("fileWrite", [](Game::scr_entref_t) // gsc: fileWrite(<filepath>, <string>, <mode>)
|
Script::AddFunction("FileWrite", [](Game::scr_entref_t) // gsc: FileWrite(<filepath>, <string>, <mode>)
|
||||||
{
|
{
|
||||||
std::string path = Game::Scr_GetString(0);
|
const std::string path = Game::Scr_GetString(0);
|
||||||
auto text = Game::Scr_GetString(1);
|
auto* text = Game::Scr_GetString(1);
|
||||||
auto mode = Game::Scr_GetString(2);
|
auto* mode = Game::Scr_GetString(2);
|
||||||
|
|
||||||
if (path.empty())
|
if (path.empty())
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileWrite: filepath not defined!\n");
|
Game::Scr_ParamError(0, "^1FileWrite: filepath not defined!\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (text == nullptr || mode == nullptr)
|
||||||
|
{
|
||||||
|
Game::Scr_Error("^1FileWrite: Illegal parameters!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -23,14 +29,14 @@ namespace Components
|
|||||||
{
|
{
|
||||||
if (path.find(queryStrings[i]) != std::string::npos)
|
if (path.find(queryStrings[i]) != std::string::npos)
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileWrite: directory traversal is not allowed!\n");
|
Logger::Print("^1FileWrite: directory traversal is not allowed!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mode != "append"s && mode != "write"s)
|
if (mode != "append"s && mode != "write"s)
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^3fileWrite: mode not defined or was wrong, defaulting to 'write'\n");
|
Logger::Print("^3FileWrite: mode not defined or was wrong, defaulting to 'write'\n");
|
||||||
mode = "write";
|
mode = "write";
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -44,13 +50,13 @@ namespace Components
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Script::AddFunction("fileRead", [](Game::scr_entref_t) // gsc: fileRead(<filepath>)
|
Script::AddFunction("FileRead", [](Game::scr_entref_t) // gsc: FileRead(<filepath>)
|
||||||
{
|
{
|
||||||
std::string path = Game::Scr_GetString(0);
|
std::string path = Game::Scr_GetString(0);
|
||||||
|
|
||||||
if (path.empty())
|
if (path.empty())
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileRead: filepath not defined!\n");
|
Game::Scr_ParamError(0, "^1FileRead: filepath not defined!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,27 +65,27 @@ namespace Components
|
|||||||
{
|
{
|
||||||
if (path.find(queryStrings[i]) != std::string::npos)
|
if (path.find(queryStrings[i]) != std::string::npos)
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileRead: directory traversal is not allowed!\n");
|
Logger::Print("^1FileRead: directory traversal is not allowed!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!FileSystem::FileReader(path).exists())
|
if (!FileSystem::FileReader(path).exists())
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileRead: file not found!\n");
|
Logger::Print("^1FileRead: file not found!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Game::Scr_AddString(FileSystem::FileReader(path).getBuffer().data());
|
Game::Scr_AddString(FileSystem::FileReader(path).getBuffer().data());
|
||||||
});
|
});
|
||||||
|
|
||||||
Script::AddFunction("fileExists", [](Game::scr_entref_t) // gsc: fileExists(<filepath>)
|
Script::AddFunction("FileExists", [](Game::scr_entref_t) // gsc: FileExists(<filepath>)
|
||||||
{
|
{
|
||||||
std::string path = Game::Scr_GetString(0);
|
std::string path = Game::Scr_GetString(0);
|
||||||
|
|
||||||
if (path.empty())
|
if (path.empty())
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileExists: filepath not defined!\n");
|
Game::Scr_ParamError(0, "^1FileExists: filepath not defined!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -88,7 +94,7 @@ namespace Components
|
|||||||
{
|
{
|
||||||
if (path.find(queryStrings[i]) != std::string::npos)
|
if (path.find(queryStrings[i]) != std::string::npos)
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileExists: directory traversal is not allowed!\n");
|
Logger::Print("^1FileExists: directory traversal is not allowed!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -96,13 +102,13 @@ namespace Components
|
|||||||
Game::Scr_AddInt(FileSystem::FileReader(path).exists());
|
Game::Scr_AddInt(FileSystem::FileReader(path).exists());
|
||||||
});
|
});
|
||||||
|
|
||||||
Script::AddFunction("fileRemove", [](Game::scr_entref_t) // gsc: fileRemove(<filepath>)
|
Script::AddFunction("FileRemove", [](Game::scr_entref_t) // gsc: FileRemove(<filepath>)
|
||||||
{
|
{
|
||||||
std::string path = Game::Scr_GetString(0);
|
std::string path = Game::Scr_GetString(0);
|
||||||
|
|
||||||
if (path.empty())
|
if (path.empty())
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileRemove: filepath not defined!\n");
|
Game::Scr_ParamError(0, "^1FileRemove: filepath not defined!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +117,7 @@ namespace Components
|
|||||||
{
|
{
|
||||||
if (path.find(queryStrings[i]) != std::string::npos)
|
if (path.find(queryStrings[i]) != std::string::npos)
|
||||||
{
|
{
|
||||||
Game::Com_Printf(0, "^1fileRemove: directory traversal is not allowed!\n");
|
Logger::Print("^1fileRemove: directory traversal is not allowed!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -123,9 +129,9 @@ namespace Components
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void Client::AddMethods()
|
void ScriptExtension::AddMethods()
|
||||||
{
|
{
|
||||||
// Client methods
|
// ScriptExtension methods
|
||||||
Script::AddFunction("GetIp", [](Game::scr_entref_t entref) // gsc: self GetIp()
|
Script::AddFunction("GetIp", [](Game::scr_entref_t entref) // gsc: self GetIp()
|
||||||
{
|
{
|
||||||
const auto* gentity = Script::GetEntFromEntRef(entref);
|
const auto* gentity = Script::GetEntFromEntRef(entref);
|
||||||
@ -133,8 +139,10 @@ namespace Components
|
|||||||
|
|
||||||
std::string ip = Game::NET_AdrToString(client->netchan.remoteAddress);
|
std::string ip = Game::NET_AdrToString(client->netchan.remoteAddress);
|
||||||
|
|
||||||
if (ip.find_first_of(":") != std::string::npos)
|
const auto pos = ip.find_first_of(":");
|
||||||
ip.erase(ip.begin() + ip.find_first_of(":"), ip.end()); // Erase port
|
|
||||||
|
if (pos != std::string::npos)
|
||||||
|
ip.erase(ip.begin() + pos, ip.end()); // Erase port
|
||||||
|
|
||||||
Game::Scr_AddString(ip.data());
|
Game::Scr_AddString(ip.data());
|
||||||
});
|
});
|
||||||
@ -148,13 +156,9 @@ namespace Components
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Client::Client()
|
ScriptExtension::ScriptExtension()
|
||||||
{
|
|
||||||
Client::AddFunctions();
|
|
||||||
Client::AddMethods();
|
|
||||||
}
|
|
||||||
|
|
||||||
Client::~Client()
|
|
||||||
{
|
{
|
||||||
|
ScriptExtension::AddFunctions();
|
||||||
|
ScriptExtension::AddMethods();
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -2,11 +2,10 @@
|
|||||||
|
|
||||||
namespace Components
|
namespace Components
|
||||||
{
|
{
|
||||||
class Client : public Component
|
class ScriptExtension : public Component
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Client();
|
ScriptExtension();
|
||||||
~Client();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
@ -1218,11 +1218,11 @@ namespace Game
|
|||||||
{
|
{
|
||||||
pushad
|
pushad
|
||||||
|
|
||||||
mov eax, [esp + 0x10 + 0x20]
|
mov eax, [esp + 0x10 + 0x20] // msg
|
||||||
mov edi, [esp + 0x4 + 0x20]
|
mov edi, [esp + 0x4 + 0x20] // channel
|
||||||
|
|
||||||
push [esp + 0xC + 0x20]
|
push [esp + 0xC + 0x20] // index
|
||||||
push [esp + 0xC + 0x20]
|
push [esp + 0xC + 0x20] // codePos
|
||||||
|
|
||||||
mov edx, 0x61ABE0
|
mov edx, 0x61ABE0
|
||||||
call edx
|
call edx
|
||||||
|
@ -20,11 +20,11 @@ namespace Game
|
|||||||
typedef vec_t vec3_t[3];
|
typedef vec_t vec3_t[3];
|
||||||
typedef vec_t vec4_t[4];
|
typedef vec_t vec4_t[4];
|
||||||
|
|
||||||
typedef struct
|
struct scr_entref_t
|
||||||
{
|
{
|
||||||
unsigned __int16 entnum;
|
unsigned __int16 entnum;
|
||||||
unsigned __int16 classnum;
|
unsigned __int16 classnum;
|
||||||
} scr_entref_t;
|
};
|
||||||
|
|
||||||
typedef void(__cdecl * scr_function_t)(scr_entref_t);
|
typedef void(__cdecl * scr_function_t)(scr_entref_t);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user