230 lines
4.7 KiB
C++
230 lines
4.7 KiB
C++
#include <STDInclude.hpp>
|
|
|
|
namespace Components
|
|
{
|
|
std::unordered_map<std::string, std::function<void(Command::Params*)>> Command::FunctionMap;
|
|
std::unordered_map<std::string, std::function<void(Command::Params*)>> 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<void()>& callback)
|
|
{
|
|
Add(name, [callback]([[maybe_unused]] const Params* params)
|
|
{
|
|
callback();
|
|
});
|
|
}
|
|
|
|
void Command::Add(const char* name, const std::function<void(Params*)>& 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<void(Params*)>& 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)
|
|
{
|
|
if (command.empty())
|
|
{
|
|
return;
|
|
}
|
|
|
|
command.push_back('\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<Game::cmd_function_s>();
|
|
}
|
|
|
|
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<std::string>& Command::GetExceptions()
|
|
{
|
|
static const auto exceptions = []() -> std::vector<std::string>
|
|
{
|
|
std::vector<std::string> 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
|
|
}
|
|
}
|