mirror of
https://github.com/dashr9230/SA-MP.git
synced 2025-01-12 03:52:07 +08:00
574 lines
12 KiB
C++
574 lines
12 KiB
C++
|
|
#include "main.h"
|
|
|
|
extern bool bQuitApp;
|
|
|
|
void con_exit()
|
|
{
|
|
bQuitApp = true;
|
|
}
|
|
|
|
void con_echo()
|
|
{
|
|
char* arg = strtok(NULL, "");
|
|
if (arg)
|
|
{
|
|
logprintf("%s", arg);
|
|
} else {
|
|
logprintf("");
|
|
}
|
|
}
|
|
|
|
void con_exec()
|
|
{
|
|
char* arg = strtok(NULL, " ");
|
|
if (arg)
|
|
{
|
|
char tmp[1025];
|
|
char tmpvarname[1025];
|
|
sprintf(tmp, "%s.cfg", arg);
|
|
FILE* f = fopen(tmp, "r");
|
|
if (!f)
|
|
{
|
|
logprintf("Unable to exec file '%s'.", tmp);
|
|
} else {
|
|
memset(tmp, 0, sizeof(tmp));
|
|
while (fgets(tmp, 1024, f))
|
|
{
|
|
int len = strlen(tmp);
|
|
if (len > 1024) continue;
|
|
if (len > 0 && tmp[len-1] == '\n')
|
|
tmp[strlen(tmp)-1] = 0;
|
|
len = strlen(tmp);
|
|
if (len > 0 && tmp[len-1] == '\r')
|
|
tmp[strlen(tmp)-1] = 0;
|
|
if (tmp[0] == 0)
|
|
continue;
|
|
// If the line has a comment, finish it there.
|
|
for (size_t i=0; i<strlen(tmp)-1; i++)
|
|
{
|
|
if (((tmp[i] == '/') && (tmp[i+1] == '/')) || tmp[i] == '#')
|
|
{
|
|
tmp[i] = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (strlen(tmp) > 2)
|
|
{
|
|
if ((tmp[0] != '/') && (tmp[1] != '/')) {
|
|
memset(tmpvarname, 0, sizeof(tmpvarname));
|
|
for(int i = 0; tmp[i] != '\0'; i++)
|
|
{
|
|
if (tmp[i] == ' ')
|
|
{
|
|
tmpvarname[i] = '\0';
|
|
break;
|
|
}
|
|
tmpvarname[i] = tmp[i];
|
|
}
|
|
if (pConsole->FindVariable(tmpvarname))
|
|
{
|
|
pConsole->Execute(tmp);
|
|
}
|
|
}
|
|
}
|
|
memset(tmp, 0, sizeof(tmp));
|
|
}
|
|
fclose(f);
|
|
}
|
|
} else {
|
|
logprintf("Usage:");
|
|
logprintf(" exec <filename>");
|
|
}
|
|
}
|
|
|
|
void con_kick() {} // TODO: con_kick W: 0048A530 L: 0809ECB0
|
|
void con_ban() {} // TODO: con_ban W: 0048A5D0 L: 0809ED80
|
|
void con_banip() {} // TODO: con_banip W: 0048A740 L: 0809EF40
|
|
void con_unbanip() {} // TODO: con_unbanip W: 0048A790 L: 0809EFB0
|
|
void con_gmx() {} // TODO: con_gmx W: 0048A7D0 L: 0809F000
|
|
void con_changemode() {} // TODO: con_changemode W: 0048A800 L: 0809F050
|
|
|
|
void con_cmdlist();
|
|
|
|
void con_varlist()
|
|
{
|
|
pConsole->PrintVariableList();
|
|
}
|
|
|
|
void con_say() {} // TODO: con_say W: 0048A830 L: 0809F0A0
|
|
void con_reloadbans() {} // TODO: con_reloadbans W: 0048A880 L: 0809F110
|
|
|
|
void con_reloadlog() {
|
|
LoadLogFile();
|
|
}
|
|
|
|
void con_players() {} // TODO: con_players W: 0048A8A0 L: 0809F140
|
|
void con_gravity() {} // TODO: con_gravity W: 0048A950 L: 0809F220
|
|
void con_weather() {} // TODO: con_weather W: 0048A980 L: 0809F260
|
|
|
|
void con_loadfs()
|
|
{
|
|
PLAYERID Temp = RconUser;
|
|
// Stop sending all the filterscripts prints to the user if they used in-game RCON
|
|
RconUser = INVALID_ID;
|
|
char* arg = strtok(NULL, "");
|
|
if (arg)
|
|
{
|
|
if(!pNetGame->GetFilterScripts()->LoadFilterScript(arg))
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' load failed.", arg);
|
|
}
|
|
else
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' loaded.", arg);
|
|
}
|
|
}
|
|
}
|
|
|
|
void con_reloadfs()
|
|
{
|
|
PLAYERID Temp = RconUser;
|
|
// Stop sending all the filterscripts prints to the user if they used in-game RCON
|
|
RconUser = INVALID_ID;
|
|
char* arg = strtok(NULL, "");
|
|
if (arg)
|
|
{
|
|
if(pNetGame->GetFilterScripts()->UnloadOneFilterScript(arg))
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' unloaded.", arg);
|
|
}
|
|
else
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' unload failed.", arg);
|
|
}
|
|
|
|
RconUser = INVALID_ID;
|
|
if(!pNetGame->GetFilterScripts()->LoadFilterScript(arg))
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' load failed'.", arg);
|
|
}
|
|
else
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' loaded.", arg);
|
|
}
|
|
}
|
|
}
|
|
|
|
void con_unloadfs()
|
|
{
|
|
PLAYERID Temp = RconUser;
|
|
// Stop sending all the filterscripts prints to the user if they used in-game RCON
|
|
RconUser = INVALID_ID;
|
|
char* arg = strtok(NULL, "");
|
|
if (arg)
|
|
{
|
|
if(pNetGame->GetFilterScripts()->UnloadOneFilterScript(arg))
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' unloaded.", arg);
|
|
}
|
|
else
|
|
{
|
|
RconUser = Temp;
|
|
logprintf(" Filterscript '%s.amx' unload failed.", arg);
|
|
}
|
|
}
|
|
}
|
|
|
|
#define CON_CMDFLAG_DEBUG 1
|
|
|
|
struct ConsoleCommand_s
|
|
{
|
|
char CmdName[255];
|
|
DWORD CmdFlags;
|
|
void (*CmdFunc)();
|
|
} ConsoleCommands[] = {
|
|
{"echo", 0, con_echo},
|
|
{"exec", 0, con_exec},
|
|
{"cmdlist", 0, con_cmdlist},
|
|
{"varlist", 0, con_varlist},
|
|
{"exit", 0, con_exit},
|
|
{"kick", 0, con_kick},
|
|
{"ban", 0, con_ban},
|
|
{"gmx", 0, con_gmx},
|
|
{"changemode", 0, con_changemode},
|
|
{"say", 0, con_say},
|
|
{"reloadbans", 0, con_reloadbans},
|
|
{"reloadlog", 0, con_reloadlog},
|
|
{"players", 0, con_players},
|
|
{"banip", 0, con_banip},
|
|
{"unbanip", 0, con_unbanip},
|
|
{"gravity", 0, con_gravity},
|
|
{"weather", 0, con_weather},
|
|
{"loadfs", 0, con_loadfs},
|
|
{"unloadfs", 0, con_unloadfs},
|
|
{"reloadfs", 0, con_reloadfs},
|
|
};
|
|
|
|
void con_cmdlist()
|
|
{
|
|
logprintf("Console Commands:");
|
|
for (int i=0; i<ARRAY_SIZE(ConsoleCommands); i++)
|
|
{
|
|
logprintf(" %s%s", ConsoleCommands[i].CmdName, (ConsoleCommands[i].CmdFlags & CON_CMDFLAG_DEBUG)?" (DEBUG)":"");
|
|
}
|
|
logprintf("");
|
|
}
|
|
|
|
char* strrtrim(char* str)
|
|
{
|
|
for (int i=strlen(str)-1; i>=0; i--)
|
|
{
|
|
if ((str[i] == ' ') || (str[i] == '\t'))
|
|
{
|
|
str[i] = 0;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return str;
|
|
}
|
|
|
|
CConsole::CConsole()
|
|
{
|
|
|
|
}
|
|
|
|
CConsole::~CConsole()
|
|
{
|
|
ConsoleVariables.clear();
|
|
}
|
|
|
|
ConsoleVariable_s* CConsole::FindVariable(char* pVarName)
|
|
{
|
|
char VarName[256];
|
|
strncpy(VarName, pVarName, 255);
|
|
strlwr(VarName);
|
|
|
|
StringConvarMap::iterator itor;
|
|
itor = ConsoleVariables.find(VarName);
|
|
if (itor != ConsoleVariables.end())
|
|
{
|
|
return itor->second;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void CConsole::PrintVariableList()
|
|
{
|
|
char val[1034]; // Max str size of 1024 + the (string) tag.
|
|
logprintf("Console Variables:");
|
|
StringConvarMap::iterator itor;
|
|
for (itor = ConsoleVariables.begin(); itor != ConsoleVariables.end(); itor++)
|
|
{
|
|
val[0] = 0;
|
|
switch (itor->second->VarType)
|
|
{
|
|
case CON_VARTYPE_FLOAT:
|
|
sprintf(val, "%f (float)", *(float*)itor->second->VarPtr);
|
|
break;
|
|
case CON_VARTYPE_INT:
|
|
sprintf(val, "%d (int)", *(int*)itor->second->VarPtr);
|
|
break;
|
|
case CON_VARTYPE_BOOL:
|
|
sprintf(val, "%d (bool)", *(bool*)itor->second->VarPtr);
|
|
break;
|
|
case CON_VARTYPE_STRING:
|
|
sprintf(val, "\"%s\" (string)", (char*)itor->second->VarPtr);
|
|
break;
|
|
}
|
|
const char* VarName = itor->first.c_str();
|
|
logprintf(" %s\t%s= %s%s%s%s", VarName, (strlen(VarName)<6)?"\t":"", val,
|
|
(itor->second->VarFlags & CON_VARFLAG_READONLY)?" (read-only)":"",
|
|
(itor->second->VarFlags & CON_VARFLAG_DEBUG)?" (debug)":"",
|
|
(itor->second->VarFlags & CON_VARFLAG_RULE)?" (rule)":"");
|
|
}
|
|
logprintf("");
|
|
}
|
|
|
|
void CConsole::AddVariable(char* pVarName, CON_VARTYPE VarType, DWORD VarFlags, void* VarPtr,
|
|
VARCHANGEFUNC VarChangeFunc)
|
|
{
|
|
char VarName[256];
|
|
strncpy(VarName, pVarName, 255);
|
|
strlwr(VarName);
|
|
|
|
if (FindVariable(VarName) != NULL)
|
|
{
|
|
logprintf("Unable to add console variable '%s'. Already exists.", VarName);
|
|
return;
|
|
}
|
|
|
|
ConsoleVariable_s* ConVar = new ConsoleVariable_s;
|
|
ConVar->VarType = VarType;
|
|
ConVar->VarFlags = VarFlags;
|
|
ConVar->VarPtr = VarPtr;
|
|
ConVar->VarChangeFunc = VarChangeFunc;
|
|
|
|
if (VarChangeFunc)
|
|
VarChangeFunc();
|
|
|
|
ConsoleVariables.insert(StringConvarMap::value_type(VarName, ConVar));
|
|
}
|
|
|
|
char* CConsole::AddStringVariable(char* pVarName, DWORD VarFlags, char* pInitStr,
|
|
VARCHANGEFUNC VarChangeFunc)
|
|
{
|
|
char* str;
|
|
if (!pInitStr)
|
|
{
|
|
// Empty string
|
|
str = (char*)malloc(1);
|
|
str[0] = 0;
|
|
} else {
|
|
str = (char*)malloc(strlen(pInitStr)+1);
|
|
strcpy(str,pInitStr);
|
|
}
|
|
AddVariable(pVarName, CON_VARTYPE_STRING, VarFlags, (void*)str, VarChangeFunc);
|
|
return str;
|
|
}
|
|
|
|
void CConsole::RemoveVariable(char* pVarName)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_STRING)
|
|
{
|
|
if (ConVar->VarPtr != NULL)
|
|
free(ConVar->VarPtr);
|
|
}
|
|
char VarName[255];
|
|
strncpy(VarName, pVarName, 255);
|
|
strlwr(VarName);
|
|
delete ConVar;
|
|
ConsoleVariables.erase(VarName);
|
|
}
|
|
}
|
|
|
|
void CConsole::ModifyVariableFlags(char* pVarName, DWORD VarFlags)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
ConVar->VarFlags = VarFlags;
|
|
}
|
|
}
|
|
|
|
char* CConsole::GetStringVariable(char* pVarName)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_STRING)
|
|
return (char*)ConVar->VarPtr;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void CConsole::SetStringVariable(char* pVarName, char* pString)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_STRING)
|
|
{
|
|
if (ConVar->VarPtr != NULL)
|
|
free(ConVar->VarPtr);
|
|
|
|
char* str = (char*)malloc(strlen(pString)+1);
|
|
strcpy(str, pString);
|
|
ConVar->VarPtr = str;
|
|
}
|
|
}
|
|
}
|
|
|
|
float CConsole::GetFloatVariable(char* pVarName)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_FLOAT)
|
|
return *(float*)ConVar->VarPtr;
|
|
}
|
|
return 0.0f;
|
|
}
|
|
|
|
void CConsole::SetFloatVariable(char* pVarName, float fFloat)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_FLOAT)
|
|
*(float*)ConVar->VarPtr = fFloat;
|
|
}
|
|
}
|
|
|
|
int CConsole::GetIntVariable(char* pVarName)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_INT)
|
|
return *(int*)ConVar->VarPtr;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void CConsole::SetIntVariable(char* pVarName, int iInt)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_INT)
|
|
*(int*)ConVar->VarPtr = iInt;
|
|
}
|
|
}
|
|
|
|
bool CConsole::GetBoolVariable(char* pVarName)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_BOOL)
|
|
return *(bool*)ConVar->VarPtr;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CConsole::SetBoolVariable(char* pVarName, bool bBool)
|
|
{
|
|
ConsoleVariable_s* ConVar = FindVariable(pVarName);
|
|
if (ConVar != NULL)
|
|
{
|
|
if (ConVar->VarType == CON_VARTYPE_BOOL)
|
|
*(bool*)ConVar->VarPtr = bBool;
|
|
}
|
|
}
|
|
|
|
void CConsole::Execute(char* pExecLine)
|
|
{
|
|
if (!pExecLine) return;
|
|
|
|
// Ya can't strtok on a read-only string!
|
|
char cpy[256];
|
|
memset(cpy,0,sizeof(cpy));
|
|
strncpy(cpy, pExecLine, 255);
|
|
char* tmp = strtok(cpy, " ");
|
|
if(!tmp) return;
|
|
char* cmd = strlwr(tmp);
|
|
|
|
for (int i=0; i<ARRAY_SIZE(ConsoleCommands); i++)
|
|
{
|
|
if (stricmp(cmd, ConsoleCommands[i].CmdName) == 0)
|
|
{
|
|
if (ConsoleCommands[i].CmdFlags & CON_CMDFLAG_DEBUG)
|
|
{
|
|
#ifndef _DEBUG
|
|
logprintf("Unable to execute command '%s'. Debug mode required.", ConsoleCommands[i].CmdName);
|
|
return;
|
|
#endif
|
|
}
|
|
ConsoleCommands[i].CmdFunc();
|
|
return;
|
|
}
|
|
}
|
|
|
|
ConsoleVariable_s* ConVar = FindVariable(cmd);
|
|
if (ConVar != NULL)
|
|
{
|
|
bool readonly = (ConVar->VarFlags & CON_VARFLAG_READONLY) == CON_VARFLAG_READONLY;
|
|
if (ConVar->VarFlags & CON_VARFLAG_DEBUG)
|
|
{
|
|
#ifndef _DEBUG
|
|
logprintf("Unable to access variable '%s'. Debug mode required.", cmd);
|
|
return;
|
|
#endif
|
|
}
|
|
bool bChangedVar = false;
|
|
char* arg = strtok(NULL, " ");
|
|
switch (ConVar->VarType)
|
|
{
|
|
case CON_VARTYPE_FLOAT:
|
|
if ((arg) && (!readonly))
|
|
{
|
|
*(float*)ConVar->VarPtr = (float)atof(arg);
|
|
bChangedVar = true;
|
|
} else {
|
|
logprintf("%s = %f (float%s)", cmd, *(float*)ConVar->VarPtr, readonly?", read-only":"");
|
|
}
|
|
break;
|
|
case CON_VARTYPE_INT:
|
|
if ((arg) && (!readonly))
|
|
{
|
|
*(int*)ConVar->VarPtr = atoi(arg);
|
|
bChangedVar = true;
|
|
} else {
|
|
logprintf("%s = %d (int%s)", cmd, *(int*)ConVar->VarPtr, readonly?", read-only":"");
|
|
}
|
|
break;
|
|
case CON_VARTYPE_BOOL:
|
|
if ((arg) && (!readonly))
|
|
{
|
|
*(bool*)ConVar->VarPtr = (atoi(arg) > 0);
|
|
bChangedVar = true;
|
|
} else {
|
|
logprintf("%s = %d (bool%s)", cmd, *(bool*)ConVar->VarPtr, readonly?", read-only":"");
|
|
}
|
|
break;
|
|
case CON_VARTYPE_STRING:
|
|
if ((arg) && (!readonly))
|
|
{
|
|
if (ConVar->VarPtr != NULL)
|
|
free(ConVar->VarPtr);
|
|
|
|
char* str;
|
|
char* ext = strtok(NULL, "");
|
|
if (ext)
|
|
{
|
|
str = (char*)malloc(strlen(arg)+strlen(ext)+2);
|
|
strcpy(str, arg);
|
|
strcat(str, " ");
|
|
strrtrim(ext);
|
|
strcat(str, ext);
|
|
} else {
|
|
str = (char*)malloc(strlen(arg)+1);
|
|
strcpy(str, arg);
|
|
}
|
|
if (strlen(str) > 1023) str[1022] = 0;
|
|
ConVar->VarPtr = str;
|
|
bChangedVar = true;
|
|
} else {
|
|
logprintf("%s = \"%s\" (string%s)", cmd, (char*)ConVar->VarPtr, readonly?", read-only":"");
|
|
}
|
|
break;
|
|
}
|
|
if (bChangedVar)
|
|
{
|
|
if (ConVar->VarChangeFunc)
|
|
{
|
|
ConVar->VarChangeFunc();
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (!pNetGame->GetFilterScripts()->OnRconCommand(pExecLine))
|
|
{
|
|
if (pNetGame->GetGameMode())
|
|
{
|
|
if (!pNetGame->GetGameMode()->OnRconCommand(pExecLine))
|
|
{
|
|
logprintf("Unknown command or variable:\n %s", cmd);
|
|
}
|
|
}
|
|
}
|
|
}
|