mirror of
https://github.com/alliedmodders/amxmodx.git
synced 2025-01-01 09:45:37 +03:00
b77c8e253b
Changed format with formatex where needed, removed many hardcoded contents in plugin and added semicolons.
561 lines
12 KiB
SourcePawn
Executable File
561 lines
12 KiB
SourcePawn
Executable File
// 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
|
|
|
|
//
|
|
// Commands Menu Plugin
|
|
//
|
|
|
|
#include <amxmodx>
|
|
#include <amxmisc>
|
|
|
|
// Precache sounds from speech.ini - comment this line to disable
|
|
#define PRECACHE_SPEECHINI
|
|
|
|
/* Commands Menus */
|
|
|
|
#define MAX_CMDS_LAYERS 3
|
|
|
|
new g_cmdMenuName[MAX_CMDS_LAYERS][] =
|
|
{
|
|
"CMD_MENU",
|
|
"CONF_MENU",
|
|
"SPE_MENU"
|
|
};
|
|
|
|
new g_cmdMenuCmd[MAX_CMDS_LAYERS][] =
|
|
{
|
|
"amx_cmdmenu",
|
|
"amx_cfgmenu",
|
|
"amx_speechmenu"
|
|
};
|
|
|
|
new g_cmdMenuCfg[MAX_CMDS_LAYERS][] =
|
|
{
|
|
"cmds.ini",
|
|
"configs.ini",
|
|
"speech.ini"
|
|
};
|
|
|
|
new g_cmdMenuHelp[MAX_CMDS_LAYERS][] =
|
|
{
|
|
"- displays commands menu",
|
|
"- displays configs menu",
|
|
"- displays speech menu"
|
|
};
|
|
|
|
/* End of Commands Menu */
|
|
|
|
#define MAX_CMDS 64
|
|
#define MAX_CVARS 48
|
|
|
|
new g_cmdName[MAX_CMDS*MAX_CMDS_LAYERS][32];
|
|
new g_cmdCmd[MAX_CMDS*MAX_CMDS_LAYERS][64];
|
|
new g_cmdMisc[MAX_CMDS*MAX_CMDS_LAYERS][2];
|
|
new g_cmdNum[MAX_CMDS_LAYERS];
|
|
|
|
new g_cvarNames[MAX_CVARS][32];
|
|
new g_cvarMisc[MAX_CVARS][3];
|
|
new g_cvarCmd[MAX_CVARS*5][32];
|
|
new g_cvarCmdNum;
|
|
new g_cvarNum;
|
|
|
|
new g_menuPosition[MAX_PLAYERS + 1];
|
|
new g_menuSelect[MAX_PLAYERS + 1][MAX_CMDS];
|
|
new g_menuSelectNum[MAX_PLAYERS + 1];
|
|
new g_menuLayer[MAX_PLAYERS + 1];
|
|
|
|
new g_coloredMenus;
|
|
|
|
|
|
public plugin_init()
|
|
{
|
|
register_plugin("Commands Menu", AMXX_VERSION_STR, "AMXX Dev Team");
|
|
register_dictionary("cmdmenu.txt");
|
|
register_dictionary("common.txt");
|
|
|
|
new configsDir[64], config[64];
|
|
get_configsdir(configsDir, charsmax(configsDir));
|
|
|
|
for (new a = 0; a < MAX_CMDS_LAYERS; ++a)
|
|
{
|
|
new MenuName[64];
|
|
|
|
formatex(MenuName, charsmax(MenuName), "%L", "en", g_cmdMenuName[a]);
|
|
register_menucmd(register_menuid(MenuName), 1023, "actionCmdMenu");
|
|
register_clcmd(g_cmdMenuCmd[a], "cmdCmdMenu", ADMIN_MENU, g_cmdMenuHelp[a]);
|
|
formatex(config, charsmax(config), "%s/%s", configsDir, g_cmdMenuCfg[a]);
|
|
loadCmdSettings(config, a);
|
|
}
|
|
|
|
register_menucmd(register_menuid("Cvars Menu"), 1023, "actionCvarMenu");
|
|
register_clcmd("amx_cvarmenu", "cmdCvarMenu", ADMIN_CVAR, "- displays cvars menu");
|
|
|
|
new cvars_ini_file[64];
|
|
formatex(cvars_ini_file, charsmax(cvars_ini_file), "%s/%s", configsDir, "cvars.ini");
|
|
loadCvarSettings(cvars_ini_file);
|
|
|
|
g_coloredMenus = colored_menus();
|
|
}
|
|
|
|
#if defined PRECACHE_SPEECHINI
|
|
public plugin_precache( )
|
|
{
|
|
new configsDir[64], config[64];
|
|
get_configsdir(configsDir, charsmax(configsDir));
|
|
formatex( config, charsmax(config), "%s/%s", configsDir, "speech.ini" );
|
|
|
|
new fp = fopen( config, "rt" ); // Read file as text
|
|
|
|
if ( ! fp ) // File doesn't exists
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
new szText[256];
|
|
new line = 0;
|
|
new szName[32], szSound[128], sndExt[5];
|
|
new field1[32], field2[64], field3[64];
|
|
new fieldNums = 0;
|
|
new const voxIdent[] = "vox", fvoxIdent[] = "fvox", barneyIdent[] = "barney", hgruntIdent[] = "hgrunt";
|
|
|
|
while ( line < MAX_CMDS && ! feof( fp ) ) // Loop till MAX_CMDS or EOF
|
|
{
|
|
fgets( fp, szText, charsmax(szText) ); // Store line content
|
|
|
|
/* Strips newline */
|
|
new len = strlen( szText );
|
|
if ( len != 0 && szText[len-1] == '^n' ) // len != 0 because if the last line of the file is empty, there's no newline
|
|
{
|
|
szText[--len] = 0;
|
|
}
|
|
|
|
if ( len == 0 || szText[0] == ';' || szText[0] == '/' ) // Line is empty or a comment
|
|
{
|
|
continue;
|
|
}
|
|
|
|
parse( szText, szName, charsmax(szName), szSound, charsmax(szSound) );
|
|
fieldNums = parse( szSound, field1, charsmax(field1), field2, charsmax(field2), field3, charsmax(field3) );
|
|
if ( fieldNums == 2 && field1[0] == 's' ) // .wav (spk)
|
|
{
|
|
copy( szSound, charsmax(szSound), field2 );
|
|
copy( sndExt, charsmax(sndExt), ".wav" );
|
|
}
|
|
else if ( fieldNums == 3 && field1[0] == 'm' && ( field2[0] == 'p' || field2[0] == 'l' ) ) // .mp3 (mp3 play | mp3 loop)
|
|
{
|
|
copy( szSound, charsmax(szSound), field3 );
|
|
copy( sndExt, charsmax(sndExt), ".mp3" );
|
|
}
|
|
else // WTH is this sound, drop it.
|
|
continue;
|
|
|
|
replace_all( szSound, charsmax(szSound), "\'", "" ); // Strips all ugly (and sometimes useless) \'
|
|
|
|
if ( szSound[0] == '/' )
|
|
{
|
|
replace( szSound, charsmax(szSound), "/", "" ); // Strip leading slash
|
|
}
|
|
|
|
if ( sndExt[1] == 'm' || ( ! equali( szSound, voxIdent, charsmax(voxIdent) ) && ! equali( szSound, fvoxIdent, charsmax(fvoxIdent) ) && ! equali( szSound, barneyIdent, charsmax(barneyIdent) ) && ! equali( szSound, hgruntIdent, charsmax(hgruntIdent) ) ) )
|
|
{
|
|
// SzSound is a mp3, or a custom wav (not a vox, fvox, or default sound from HL pak)
|
|
if ( !equali( szSound[strlen(szSound)-4], sndExt ) )
|
|
{
|
|
add( szSound, charsmax(szSound), sndExt ); // Add filetype extension if it isn't already specified
|
|
}
|
|
if ( sndExt[1] == 'w' )
|
|
{
|
|
format( szSound, charsmax(szSound), "sound/%s", szSound ); // spk basedir is $moddir/sound, but mp3 play is $moddir, fix this for the file_exists check
|
|
}
|
|
if ( file_exists( szSound ) )
|
|
{
|
|
if ( sndExt[1] == 'm')
|
|
{
|
|
precache_generic( szSound ); // mp3
|
|
}
|
|
else
|
|
{
|
|
replace( szSound, charsmax(szSound), "sound/", "" ); // wav, strip the leading sound/ we added for our file_exists check
|
|
precache_sound( szSound );
|
|
}
|
|
}
|
|
}
|
|
line++;
|
|
}
|
|
fclose( fp ); // Close file
|
|
return line;
|
|
}
|
|
#endif
|
|
|
|
/* Commands menu */
|
|
|
|
public actionCmdMenu(id, key)
|
|
{
|
|
switch (key)
|
|
{
|
|
case 8:
|
|
{
|
|
displayCmdMenu(id, ++g_menuPosition[id]);
|
|
}
|
|
case 9:
|
|
{
|
|
displayCmdMenu(id, --g_menuPosition[id]);
|
|
}
|
|
default:
|
|
{
|
|
new option = g_menuSelect[id][g_menuPosition[id] * 8 + key];
|
|
new flags = g_cmdMisc[option][1];
|
|
|
|
if (flags & 1)
|
|
server_cmd("%s", g_cmdCmd[option])
|
|
else if (flags & 2)
|
|
client_cmd(id, "%s", g_cmdCmd[option])
|
|
else if (flags & 4)
|
|
client_cmd(0, "%s", g_cmdCmd[option])
|
|
|
|
if (flags & 8)
|
|
{
|
|
displayCmdMenu(id, g_menuPosition[id]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
displayCmdMenu(id, pos)
|
|
{
|
|
if (pos < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
new menuBody[512];
|
|
new b = 0;
|
|
new start = pos * 8;
|
|
|
|
if (start >= g_menuSelectNum[id])
|
|
{
|
|
start = pos = g_menuPosition[id] = 0;
|
|
}
|
|
|
|
new limit = (g_menuSelectNum[id] / 8 + ((g_menuSelectNum[id] % 8)));
|
|
new len = formatex(menuBody, charsmax(menuBody), g_coloredMenus ? "\y%L\R%d/%d^n\w^n" : "%L %d/%d^n^n", id, g_cmdMenuName[g_menuLayer[id]], pos + 1, (limit == 0) ? 1 : limit);
|
|
new end = start + 8;
|
|
new keys = MENU_KEY_0;
|
|
|
|
if (end > g_menuSelectNum[id])
|
|
{
|
|
end = g_menuSelectNum[id];
|
|
}
|
|
|
|
for (new a = start; a < end; ++a)
|
|
{
|
|
if (g_cmdCmd[g_menuSelect[id][a]][0] == '-')
|
|
{
|
|
if (g_coloredMenus)
|
|
{
|
|
len += formatex(menuBody[len], charsmax(menuBody) - len, "\d%s^n\w", g_cmdName[g_menuSelect[id][a]]);
|
|
}
|
|
else
|
|
{
|
|
len += formatex(menuBody[len], charsmax(menuBody) - len, "%s^n", g_cmdName[g_menuSelect[id][a]]);
|
|
}
|
|
++b;
|
|
}
|
|
else
|
|
{
|
|
keys |= (1<<b);
|
|
len += formatex(menuBody[len], charsmax(menuBody) - len, "%d. %s^n", ++b, g_cmdName[g_menuSelect[id][a]]);
|
|
}
|
|
}
|
|
|
|
if (end != g_menuSelectNum[id])
|
|
{
|
|
formatex(menuBody[len], charsmax(menuBody) - len, "^n9. %L...^n0. %L", id, "MORE", id, pos ? "BACK" : "EXIT");
|
|
keys |= MENU_KEY_9;
|
|
}
|
|
else
|
|
{
|
|
formatex(menuBody[len], charsmax(menuBody) - len, "^n0. %L", id, pos ? "BACK" : "EXIT");
|
|
}
|
|
|
|
new MenuName[64];
|
|
|
|
formatex(MenuName, charsmax(MenuName), "%L", "en", g_cmdMenuName[g_menuLayer[id]]);
|
|
show_menu(id, keys, menuBody, -1, MenuName);
|
|
}
|
|
|
|
public cmdCmdMenu(id, level, cid)
|
|
{
|
|
if (!cmd_access(id, level, cid, 1))
|
|
{
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
new szCmd[32];
|
|
read_argv(0, szCmd, charsmax(szCmd));
|
|
new lvl = 0;
|
|
|
|
while (lvl < MAX_CMDS_LAYERS)
|
|
{
|
|
if (equal(g_cmdMenuCmd[lvl], szCmd))
|
|
{
|
|
break;
|
|
}
|
|
++lvl;
|
|
}
|
|
|
|
g_menuLayer[id] = lvl;
|
|
g_menuSelectNum[id] = 0;
|
|
|
|
new a = lvl * MAX_CMDS;
|
|
new d, c = 0;
|
|
|
|
while (c < g_cmdNum[lvl])
|
|
{
|
|
d = a + c;
|
|
|
|
if (access(id, g_cmdMisc[d][0]))
|
|
{
|
|
g_menuSelect[id][g_menuSelectNum[id]++] = d;
|
|
}
|
|
|
|
++c;
|
|
}
|
|
|
|
displayCmdMenu(id, g_menuPosition[id] = 0);
|
|
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
loadCmdSettings(szFilename[], level)
|
|
{
|
|
if (!file_exists(szFilename))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
new text[256], szFlags[32], szAccess[32];
|
|
new a, pos = 0, c, d = level * MAX_CMDS;
|
|
|
|
while (g_cmdNum[level] < MAX_CMDS && read_file(szFilename, pos++, text, charsmax(text), a))
|
|
{
|
|
if (text[0] == ';')
|
|
{
|
|
continue;
|
|
}
|
|
c = d + g_cmdNum[level];
|
|
|
|
if (parse(text, g_cmdName[c], charsmax(g_cmdName[]), g_cmdCmd[c], charsmax(g_cmdCmd[]), szFlags, charsmax(szFlags), szAccess, charsmax(szAccess)) > 3)
|
|
{
|
|
while (replace(g_cmdCmd[c], charsmax(g_cmdCmd[]), "\'", "^""))
|
|
{
|
|
// do nothing
|
|
}
|
|
|
|
g_cmdMisc[c][1] = read_flags(szFlags);
|
|
g_cmdMisc[c][0] = read_flags(szAccess);
|
|
g_cmdNum[level]++;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Cvars menu */
|
|
|
|
public actionCvarMenu(id, key)
|
|
{
|
|
switch (key)
|
|
{
|
|
case 8:
|
|
{
|
|
displayCvarMenu(id, ++g_menuPosition[id]);
|
|
}
|
|
case 9:
|
|
{
|
|
displayCvarMenu(id, --g_menuPosition[id]);
|
|
}
|
|
default:
|
|
{
|
|
new option = g_menuSelect[id][g_menuPosition[id] * 8 + key];
|
|
new szValue[32];
|
|
|
|
get_cvar_string(g_cvarNames[option], szValue, charsmax(szValue));
|
|
|
|
new end = g_cvarMisc[option][2];
|
|
new start = g_cvarMisc[option][1];
|
|
|
|
for (new i = start; ; ++i)
|
|
{
|
|
if (i < end)
|
|
{
|
|
if (equal(szValue, g_cvarCmd[i]))
|
|
{
|
|
if (++i >= end)
|
|
{
|
|
i = start;
|
|
}
|
|
|
|
set_cvar_string(g_cvarNames[option], g_cvarCmd[i]);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
set_cvar_string(g_cvarNames[option], g_cvarCmd[start]);
|
|
break;
|
|
}
|
|
}
|
|
displayCvarMenu(id, g_menuPosition[id]);
|
|
}
|
|
}
|
|
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
displayCvarMenu(id, pos)
|
|
{
|
|
if (pos < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
new menuBody[512];
|
|
new b = 0;
|
|
new start = pos * 8;
|
|
|
|
if (start >= g_menuSelectNum[id])
|
|
{
|
|
start = pos = g_menuPosition[id] = 0;
|
|
}
|
|
|
|
new len = formatex(menuBody, charsmax(menuBody), g_coloredMenus ? "\yCvars Menu\R%d/%d^n\w^n" : "Cvars Menu %d/%d^n^n", pos + 1, (g_menuSelectNum[id] / 8 + ((g_menuSelectNum[id] % 8) ? 1 : 0)));
|
|
|
|
new end = start + 8;
|
|
new keys = MENU_KEY_0;
|
|
new szValue[64];
|
|
|
|
if (end > g_menuSelectNum[id])
|
|
{
|
|
end = g_menuSelectNum[id];
|
|
}
|
|
|
|
for (new a = start; a < end; ++a)
|
|
{
|
|
get_cvar_string(g_cvarNames[g_menuSelect[id][a]], szValue, charsmax(szValue));
|
|
keys |= (1<<b);
|
|
++b;
|
|
|
|
if (g_coloredMenus)
|
|
{
|
|
len += formatex(menuBody[len], charsmax(menuBody) - len, "%d. %s\R%s^n\w", b, g_cvarNames[g_menuSelect[id][a]], szValue);
|
|
}
|
|
else
|
|
{
|
|
len += formatex(menuBody[len], charsmax(menuBody) - len, "%d. %s %s^n", b, g_cvarNames[g_menuSelect[id][a]], szValue);
|
|
}
|
|
}
|
|
|
|
if (end != g_menuSelectNum[id])
|
|
{
|
|
formatex(menuBody[len], charsmax(menuBody) - len, "^n9. %L...^n0. %L", id, "MORE", id, pos ? "BACK" : "EXIT");
|
|
keys |= MENU_KEY_9;
|
|
}
|
|
else
|
|
{
|
|
formatex(menuBody[len], charsmax(menuBody) - len, "^n0. %L", id, pos ? "BACK" : "EXIT");
|
|
}
|
|
|
|
show_menu(id, keys, menuBody);
|
|
}
|
|
|
|
public cmdCvarMenu(id, level, cid)
|
|
{
|
|
if (!cmd_access(id, level, cid, 1))
|
|
{
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
g_menuSelectNum[id] = 0;
|
|
|
|
for (new a = 0; a < g_cvarNum; ++a)
|
|
{
|
|
if (access(id, g_cvarMisc[a][0]))
|
|
{
|
|
g_menuSelect[id][g_menuSelectNum[id]++] = a;
|
|
}
|
|
}
|
|
|
|
displayCvarMenu(id, g_menuPosition[id] = 0);
|
|
|
|
return PLUGIN_HANDLED;
|
|
}
|
|
|
|
loadCvarSettings(szFilename[])
|
|
{
|
|
if (!file_exists(szFilename))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
new text[256], szValues[12][32];
|
|
new inum, a, pos = 0;
|
|
new cvar_values = MAX_CVARS * 5;
|
|
|
|
// a b c d
|
|
while (g_cvarNum < MAX_CVARS && read_file(szFilename, pos++, text, charsmax(text), a))
|
|
{
|
|
if (text[0] == ';')
|
|
{
|
|
continue;
|
|
}
|
|
|
|
inum = parse(text, g_cvarNames[g_cvarNum], charsmax(g_cvarNames[]),
|
|
szValues[0], charsmax(szValues[]),
|
|
szValues[1], charsmax(szValues[]),
|
|
szValues[2], charsmax(szValues[]),
|
|
szValues[3], charsmax(szValues[]),
|
|
szValues[4], charsmax(szValues[]),
|
|
szValues[5], charsmax(szValues[]),
|
|
szValues[6], charsmax(szValues[]),
|
|
szValues[7], charsmax(szValues[]),
|
|
szValues[8], charsmax(szValues[]),
|
|
szValues[9], charsmax(szValues[]),
|
|
szValues[10], charsmax(szValues[]),
|
|
szValues[11], charsmax(szValues[]));
|
|
|
|
inum -= 2;
|
|
if (inum < 2)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
g_cvarMisc[g_cvarNum][1] = g_cvarCmdNum;
|
|
|
|
for (a = 0; a < inum && g_cvarCmdNum < cvar_values; ++a)
|
|
{
|
|
while (replace(szValues[a], charsmax(szValues[]), "\'", "^""))
|
|
{
|
|
// do nothing
|
|
}
|
|
|
|
copy(g_cvarCmd[g_cvarCmdNum], charsmax(g_cvarCmd[]), szValues[a]);
|
|
g_cvarCmdNum++;
|
|
}
|
|
|
|
g_cvarMisc[g_cvarNum][2] = g_cvarCmdNum;
|
|
g_cvarMisc[g_cvarNum][0] = read_flags(szValues[inum]);
|
|
g_cvarNum++;
|
|
}
|
|
|
|
return 1;
|
|
}
|