#include namespace Components { std::unordered_map> Command::FunctionMap; std::unordered_map> Command::FunctionMapSV; std::string Command::Params::join(const int index) const { std::string result; for (auto i = index; i < this->size(); i++) { if (i > index) result.append(" "); result.append(this->get(i)); } return result; } Command::ClientParams::ClientParams() : nesting_(Game::cmd_args->nesting) { assert(Game::cmd_args->nesting < Game::CMD_MAX_NESTING); } int Command::ClientParams::size() const noexcept { return Game::cmd_args->argc[this->nesting_]; } const char* Command::ClientParams::get(const int index) const noexcept { if (index >= this->size()) { return ""; } return Game::cmd_args->argv[this->nesting_][index]; } Command::ServerParams::ServerParams() : nesting_(Game::sv_cmd_args->nesting) { assert(Game::sv_cmd_args->nesting < Game::CMD_MAX_NESTING); } int Command::ServerParams::size() const noexcept { return Game::sv_cmd_args->argc[this->nesting_]; } const char* Command::ServerParams::get(const int index) const noexcept { if (index >= this->size()) { return ""; } return Game::sv_cmd_args->argv[this->nesting_][index]; } void Command::Add(const char* name, const std::function& callback) { Add(name, [callback]([[maybe_unused]] const Params* params) { callback(); }); } void Command::Add(const char* name, const std::function& callback) { const auto command = Utils::String::ToLower(name); if (!FunctionMap.contains(command)) { AddRaw(name, MainCallback); } FunctionMap.insert_or_assign(command, callback); } void Command::AddSV(const char* name, const std::function& callback) { if (Loader::IsPregame()) { MessageBoxA(nullptr, "Registering server commands in pregame state is illegal!", nullptr, MB_ICONERROR); #ifdef _DEBUG __debugbreak(); #endif return; } const auto command = Utils::String::ToLower(name); if (!FunctionMapSV.contains(command)) { AddRawSV(name, MainCallbackSV); } FunctionMapSV.insert_or_assign(command, callback); } void Command::AddRaw(const char* name, void(*callback)(), bool key) { Game::Cmd_AddCommand(name, callback, Allocate(), key); } void Command::AddRawSV(const char* name, void(*callback)()) { Game::Cmd_AddServerCommand(name, callback, Allocate()); // If the main command is registered as Cbuf_AddServerText, the command will be redirected to the SV handler AddRaw(name, Game::Cbuf_AddServerText_f, false); } void Command::Execute(std::string command, bool sync) { command.append("\n"); // Make sure it's terminated assert(command.size() < Game::MAX_CMD_LINE); if (sync) { Game::Cmd_ExecuteSingleCommand(0, 0, command.data()); } else { Game::Cbuf_AddText(0, command.data()); } } Game::cmd_function_s* Command::Find(const std::string& command) { auto* cmdFunction = *Game::cmd_functions; while (cmdFunction) { if (cmdFunction->name && Utils::String::Compare(cmdFunction->name, command)) { return cmdFunction; } cmdFunction = cmdFunction->next; } return nullptr; } Game::cmd_function_s* Command::Allocate() { return Utils::Memory::GetAllocator()->allocate(); } void Command::MainCallback() { ClientParams params; const auto command = Utils::String::ToLower(params[0]); if (const auto itr = FunctionMap.find(command); itr != FunctionMap.end()) { itr->second(¶ms); } } void Command::MainCallbackSV() { ServerParams params; const auto command = Utils::String::ToLower(params[0]); if (const auto itr = FunctionMapSV.find(command); itr != FunctionMapSV.end()) { itr->second(¶ms); } } const std::vector& Command::GetExceptions() { static const auto exceptions = []() -> std::vector { std::vector values = { "cmd", "exec", "map", }; if (Flags::HasFlag("disable-notifies")) { values.emplace_back("vstr"); values.emplace_back("wait"); } return values; }(); return exceptions; } bool Command::CL_ShouldSendNotify_Hk(const char* cmd) { if (!cmd) { return false; } const auto& exceptions = GetExceptions(); for (const auto& entry : exceptions) { if (Utils::String::Compare(cmd, entry)) { return false; } } return true; } Command::Command() { // Protect players from invasive servers Utils::Hook(0x434BD4, CL_ShouldSendNotify_Hk, HOOK_CALL).install()->quick(); // CL_CheckNotify } }