Added amb#34 - Plugin cvar/command menus.

Still need to add ML support.
This commit is contained in:
Steve Dudenhoeffer 2007-04-18 15:52:03 +00:00
parent 1ed1d0005c
commit 17d8197e72

964
plugins/pluginmenu.sma Normal file
View File

@ -0,0 +1,964 @@
/* AMX Mod X
* Plugin Cvar and Command Menu
*
* by the AMX Mod X Development Team
*
* This file is part of AMX Mod X.
*
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* In addition, as a special exception, the author gives permission to
* link the code of this program with the Half-Life Game Engine ("HL
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
* L.L.C ("Valve"). You must obey the GNU General Public License in all
* respects for all of the code used other than the HL Engine and MODs
* from Valve. If you modify this file, you may extend this exception
* to your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from your
* version.
*/
#include <amxmodx>
#include <amxmisc>
// Temporary define for having users test on their core.
// This will ensure that the build will work on 1.76.
#define BUILD_FOR_176
new DisabledCallback;
new EnabledCallback;
// pcvar that the client is currently modifying
new CurrentCvar[33];
// Name of the cvar being modified
new CurrentCvarName[33][32];
// Plugin ID that the client is modifying
new CurrentPlid[33];
// Page that the client is currently on
new CurrentPage[33];
// Menu function ID that the client is in
new CurrentMenuFunction[33] = { -1,... };
new CurrentCommand[33][32];
new ExplicitPlugin[33];
public plugin_init()
{
register_plugin("Plugin Menu",AMXX_VERSION_STR,"AMXX Dev Team");
register_dictionary("common.txt");
register_dictionary("pausecfg.txt"); // Needed for PAUSE_COULDNT_FIND
register_clcmd("amx_plugincvarmenu", "CvarMenuCommand", ADMIN_CVAR, " - displays the plugin cvar menu");
register_clcmd("amx_plugincmdmenu", "CommandMenuCommand", ADMIN_MENU, " - displays the plugin command menu");
register_clcmd("amx_changecvar","CommandChangeCvar");
register_clcmd("amx_executecmd","CommandExecuteCommand");
// Register global menu callbacks.
DisabledCallback=menu_makecallback("AlwaysDisableCallback");
EnabledCallback=menu_makecallback("AlwaysEnableCallback");
}
// Add these menus to the amxmodmenu
public plugin_cfg()
{
new PluginName[64];
get_plugin(-1,"",0,PluginName,sizeof(PluginName)-1);
AddMenuItem("Plugin Cvars", "amx_plugincvarmenu", ADMIN_CVAR, PluginName);
AddMenuItem("Plugin Commands", "amx_plugincmdmenu", ADMIN_MENU, PluginName);
}
// Reset all fields for each client as they connect.
public client_connect(id)
{
CurrentCvar[id]=0;
CurrentPlid[id]=0;
CurrentMenuFunction[id]=-1;
CurrentCvarName[id][0]=0;
CurrentCommand[id][0]=0;
ExplicitPlugin[id]=-1;
}
/**
* Creates a plugin list menu.
*
* @param MenuText The text to display as the title.
* @param Handler The function to call when an item is selected.
* @param Command The function to pass to the handler. It will be passed as "PLID Command".
* @param Callback Function to call for each plugin to be listed. Displays a number next to it (how many cvars, etc.)
*/
stock DisplayPluginMenu(id,const MenuText[], const Handler[], const Command[], const Callback[])
{
new Menu=menu_create(MenuText,Handler);
new PluginState[32];
new PluginName[64];
new func=get_func_id(Callback);
new tally;
new PluginCmd[64];
new MenuText[64];
for (new i=0, max=get_pluginsnum();
i<max;
i++)
{
if (callfunc_begin_i(func,-1)==1)
{
callfunc_push_int(i); // push the plid
if ((tally=callfunc_end())>0)
{
get_plugin(i,"",0,PluginName,sizeof(PluginName)-1,"",0,"",0,PluginState,sizeof(PluginState)-1);
// Command syntax is: "# Function", # being plugin ID, function being public function to call.
formatex(PluginCmd,sizeof(PluginCmd)-1,"%d %s",i,Command);
formatex(MenuText,sizeof(MenuText)-1,"%s - %d",PluginName,tally);
// If the plugin is running, add this as an activated menu item.
if (strcmp(PluginState,"running",true)==0 ||
strcmp(PluginState,"debug", true)==0)
{
menu_additem(Menu,MenuText,PluginCmd,EnabledCallback);
}
else
{
menu_additem(Menu,MenuText,"",_,DisabledCallback);
}
}
}
}
#if !defined BUILD_FOR_176
menu_setprop(Menu,MPROP_NUMBER_COLOR,"\y");
#endif
menu_setprop(Menu,MPROP_EXIT,MEXIT_ALL);
menu_display(id,Menu,0);
}
/**
* Byrefs the plugin id of a target plugin (passed by argv(1)), but only if it's valid.
*
* @param id id of the display messages to upon failure.
* @param plid Variable to byref the plugin id.
* @return True on successful lookup, false on failure.
*/
stock bool:GetPlidForValidPlugins(id, &plid)
{
// If arguments have been passed, then we were given
// a specific plugin to examine.
if (read_argc()>1)
{
// Yes, we were provided a plugin.
new TargetPlugin[64];
read_argv(1,TargetPlugin,sizeof(TargetPlugin)-1);
new BufferName[64];
new BufferFile[64];
new BufferState[64];
// Scan for the plugin ID.
for (new i=0, max=get_pluginsnum();
i<max;
i++)
{
get_plugin(i,BufferFile,sizeof(BufferFile)-1,BufferName,sizeof(BufferName)-1,"",0,"",0,BufferState,sizeof(BufferState)-1);
if (strcmp(BufferFile,TargetPlugin,true) != 0||
strcmp(BufferName,TargetPlugin,true) != 0)
{
// We have a match.
// Check the status of the plugin. If it's anything other than "running" or "debug" fail.
if (strcmp(BufferState,"running") != 0 &&
strcmp(BufferState,"debug") != 0)
{
// TODO: ML This
console_print(id,"Plugin ^"%s^" is not running.",BufferFile);
// Return a failed state.
return false;
}
plid=i;
break;
}
}
// If the plugin was not found, then tell them there was an error.
if (plid==-1)
{
console_print(id, "%L", id, "PAUSE_COULDNT_FIND", TargetPlugin);
// return a failure state
return false;
}
}
return true;
}
/**
* Returns the number of cvars available for a plugin by plid. (Callback for the plugin menu.)
*
* @return number of cvars in the plugin.
*/
public GetNumberOfCvarsForPlid(plid)
{
new count=0;
new CvarPlid;
for (new i=0, max=get_plugins_cvarsnum();
i<max;
i++)
{
get_plugins_cvar(i, "", 0,_, CvarPlid, _);
if (CvarPlid==plid)
{
count++;
}
}
return count;
}
/**
* Returns the number of commands available for a plugin by plid. (Callback for the plugin menu.)
*
* @return Number of valid commands in the plugin.
*/
public GetNumberOfCmdsForPlid(plid)
{
new count=0;
for (new i=0, max=get_concmdsnum(-1,-1);
i<max;
i++)
{
if (get_concmd_plid(i,-1,-1)==plid)
{
count++;
}
}
return count;
}
/**
* Whether or not the client has access to modify this cvar.
*
* @param id The admin id.
* @param Cvar The name of the cvar to be checked.
* @return True if the client has access, false otherwise.
*/
stock bool:CanIModifyCvar(id, const Cvar[])
{
new UserFlags=get_user_flags(id);
// If the user has rcon access don't bother checking anything.
if (UserFlags & ADMIN_RCON)
{
return true;
}
// If the cvar is "sv_password" (somehow), then check access.
if (equali(Cvar,"sv_password") && UserFlags & ADMIN_PASSWORD)
{
return true;
}
// Check to see if the cvar is flagged as protected.
if (get_cvar_flags(Cvar) & FCVAR_PROTECTED)
{
// non-rcon user trying to modify a protected cvar.
return false;
}
// All known checks done, they can change this cvar if they
// were able to open the menu.
return true;
}
/**
* Simple function to ensure that a menu item is always disabled.
*
* All parameters are dummy, nothing is used.
*/
public AlwaysDisableCallback(playerid, menuid, itemid)
{
return ITEM_DISABLED;
}
/**
* Simple function to ensure that a menu item is always enabled.
*
* All parameters are dummy, nothing is used.
*/
public AlwaysEnableCallback(playerid, menuid, itemid)
{
return ITEM_ENABLED;
}
/**
* Handler for the plugin menu.
*
* @param id The client selecting an item.
* @param menu The menu handle.
* @param item The item number that was selected.
*/
public PluginMenuSelection(id, menu, item)
{
if (item==MENU_EXIT)
{
menu_destroy(menu);
}
if (item<0)
{
return PLUGIN_HANDLED;
}
new Command[64];
new Dummy[1];
// All of the commands set for each item is the public
// function that we want to call after the item is selected.
// The parameters are: function(idPlayer,itemnumber)
// Note the menu is destroyed BEFORE the command
// gets executed.
// The command retrieved is in the format: "PLID Command"
menu_item_getinfo(menu, item, Dummy[0], Command, sizeof(Command)-1,Dummy,0,Dummy[0]);
new plid=str_to_num(Command);
new Function[32];
for (new i=0;i<sizeof(Command)-1;i++)
{
if (Command[i]==' ')
{
// we're at the break. move up one space.
i++;
copy(Function,sizeof(Function)-1,Command[i]);
break;
}
}
menu_destroy(menu);
new funcid=get_func_id(Function);
if (funcid != -1 && callfunc_begin_i(funcid)==1)
{
CurrentPage[id]=0;
CurrentPlid[id]=plid;
CurrentMenuFunction[id]=funcid;
callfunc_push_int(id);
callfunc_push_int(plid);
callfunc_push_int(0);
callfunc_end();
}
return PLUGIN_HANDLED;
}
/**
* The command to change a cvar has been called.
*
* @param id The client who is changing the cvar.
*/
public CommandChangeCvar(id)
{
// All access checks are done before this command is called.
// So if the client has no pcvar pointer in his array slot
// then just ignore the command.
if (CurrentCvar[id]==0)
{
return PLUGIN_CONTINUE;
}
new Args[256];
read_args(Args,sizeof(Args)-1);
remove_quotes(Args);
if (equali(Args,"!cancel",7))
{
// The client didn't want to change this cvar.
client_print(id,print_chat,"[AMXX] Cvar not changed.");
}
else
{
// Changed to set_cvar_* for 1.76 tests
new pointer=CurrentCvar[id];
#if defined BUILD_FOR_176
set_cvar_string(CurrentCvarName[id],Args);
#else
set_pcvar_string(CurrentCvar[id],Args);
#endif
client_print(id,print_chat,"[AMXX] Cvar ^"%s^" changed to ^"%s^"",CurrentCvarName[id],Args);
// Copy of admincmd's global output.
new Name[32];
new AuthID[40];
get_user_name(id,Name,sizeof(Name)-1);
get_user_authid(id,AuthID,sizeof(AuthID)-1);
// TODO: Additional amx_show_activity settings
log_amx("Cmd: ^"%s<%d><%s><>^" set cvar (name ^"%s^") (value ^"%s^")", Name, get_user_userid(id), AuthID, CurrentCvarName[id], Args);
new activity = get_cvar_num("amx_show_activity");
if (activity != 0)
{
new players[32], pnum, admin[64], cvar_val[64], len;
get_players(players, pnum, "c");
for (new i = 0; i < pnum; i++)
{
len = format(admin, 255, "%L", players[i], "ADMIN");
if (activity == 1)
{
len += copy(admin[len], 255-len, ":");
}
else
{
len += format(admin[len], 255-len, " %s:", Name);
}
if (get_pcvar_flags(pointer) & FCVAR_PROTECTED ||
equali(CurrentCvarName[id],"rcon_password") ||
equali(CurrentCvarName[id],"amx_sql_",8))
{
format(cvar_val, 63, "*** %L ***", players[i], "PROTECTED");
}
else
{
copy(cvar_val, 63, Args);
}
client_print(players[i], print_chat, "%L", players[i], "SET_CVAR_TO", admin, CurrentCvarName[id], cvar_val);
}
}
console_print(id, "[AMXX] %L", id, "CVAR_CHANGED", CurrentCvarName[id], Args);
}
// Now redraw the menu for the client
if (CurrentMenuFunction[id]!=-1 && callfunc_begin_i(CurrentMenuFunction[id])==1)
{
callfunc_push_int(id);
callfunc_push_int(CurrentPlid[id]);
callfunc_push_int(CurrentPage[id]);
callfunc_end();
}
return PLUGIN_HANDLED;
}
/**
* Process a selection from the cvar menu.
*
* @param id The client who chose an item.
* @param menu The menu handle.
* @param item The item that has been selected.
*/
public CvarMenuSelection(id, menu, item)
{
if (item==MENU_EXIT)
{
menu_destroy(menu);
if (ExplicitPlugin[id]==-1)
{
client_cmd(id,"amx_plugincvarmenu");
}
return PLUGIN_HANDLED;
}
else if (item==MENU_BACK)
{
--CurrentPage[id];
client_print(id,print_chat,"MENU_BACK");
return PLUGIN_HANDLED;
}
else if (item==MENU_MORE)
{
++CurrentPage[id];
client_print(id,print_chat,"MENU_MORE");
return PLUGIN_HANDLED;
}
else
{
new CvarName[64];
new Command[32];
new Dummy[1];
// pcvar pointer is stored in command, extract the name of the cvar from the name field.
menu_item_getinfo(menu, item, Dummy[0], Command, sizeof(Command)-1,CvarName,sizeof(CvarName)-1,Dummy[0]);
CurrentCvar[id]=str_to_num(Command);
if (CurrentCvar[id]==0) // This should never happen, but just incase..
{
client_print(id,print_chat,"[AMXX] There was an error extracting the cvar pointer. (Name=^"%s^")",CvarName);
return PLUGIN_HANDLED;
}
// TODO: ML this
// Scan up "CvarName" and stop at the first space
for (new i=0;i<sizeof(CvarName)-1;i++)
{
if (CvarName[i]==' ')
{
CvarName[i]='^0';
break;
}
}
copy(CurrentCvarName[id],sizeof(CurrentCvarName[])-1,CvarName);
client_print(id,print_chat,"[AMXX] Type in the new value for %s, or !cancel to cancel.",CvarName);
client_cmd(id,"messagemode amx_changecvar");
menu_destroy(menu);
return PLUGIN_HANDLED;
}
return 0;
}
/**
* Displays the cvar menu to a client.
*
* @param id id of the client.
* @param plid Plugin ID to display cvars from.
* @param page Page of the menu to start at.
*/
public DisplayCvarMenu(id, plid, page)
{
new PluginName[32];
new MenuTitle[64];
get_plugin(plid,"",0,PluginName,sizeof(PluginName)-1,"",0,"",0,"",0);
formatex(MenuTitle,sizeof(MenuTitle)-1,"%s Cvars:",PluginName);
new Menu=menu_create(MenuTitle,"CvarMenuSelection");
new Cvar[64];
new CvarPlid;
new CvarText[64];
new CvarData[32];
new CvarPtr;
for (new i=0, max=get_plugins_cvarsnum();
i<max;
i++)
{
get_plugins_cvar(i, Cvar, sizeof(Cvar),_, CvarPlid, CvarPtr);
if (CvarPlid==plid)
{
if (CanIModifyCvar(id,Cvar))
{
get_pcvar_string(CvarPtr,CvarData,sizeof(CvarData)-1);
formatex(CvarText,sizeof(CvarText)-1,"%s - %s",Cvar,CvarData);
// Now store the pcvar data in Cvar
num_to_str(CvarPtr,Cvar,sizeof(Cvar)-1);
menu_additem(Menu,CvarText,Cvar,_,EnabledCallback);
}
else
{
menu_additem(Menu,Cvar,"",_,DisabledCallback);
}
}
}
menu_setprop(Menu,MPROP_EXIT,MEXIT_ALL);
#if !defined BUILD_FOR_176
menu_setprop(Menu,MPROP_NUMBER_COLOR,"\y");
#endif
menu_display(id,Menu,page);
}
/**
* Process the "amx_plugincvarmenu" command.
*
* @param id id of the client that is calling the command.
* @param level Access level required by the command.
* @param cid Command ID.
*/
public CvarMenuCommand(id, level, cid)
{
if (!cmd_access(id,level,cid,0))
{
return PLUGIN_HANDLED;
}
// This is which plugin to display. -1 means display all plugins in a list.
new plid=-1;
if (GetPlidForValidPlugins(id,plid)!=true)
{
// If GetPlidForValidPlugins returns false then it failed to find the plugin.
return PLUGIN_HANDLED;
}
// Check if we were passed a specific plugin to display or not.
if (plid==-1)
{
ExplicitPlugin[id]=-1;
// We need to display a list of the plugins, instead of a specific plugin.
DisplayPluginMenu(id,"Plugin Cvar Menu:", "PluginMenuSelection","DisplayCvarMenu","GetNumberOfCvarsForPlid");
}
else
{
ExplicitPlugin[id]=plid;
CurrentPlid[id]=plid;
CurrentPage[id]=0;
DisplayCvarMenu(id,plid,0);
}
return PLUGIN_HANDLED;
}
/**
* Handler for the menu that displays a single command ("Execute with no params", etc).
*
* @param id Id of the client.
* @param menu Menu handle.
* @param item Item that was selected.
*/
public SpecificCommandHandler(id,menu,item)
{
// Exit was called, return to the previous menu.
if (item<0)
{
if (CurrentMenuFunction[id]!=-1 && callfunc_begin_i(CurrentMenuFunction[id])==1)
{
callfunc_push_int(id);
callfunc_push_int(CurrentPlid[id]);
callfunc_push_int(CurrentPage[id]);
callfunc_end();
}
menu_destroy(menu);
return PLUGIN_HANDLED;
}
new Dummy[1];
if (item==0) // "With params"
{
menu_item_getinfo(menu, item, Dummy[0], CurrentCommand[id], sizeof(CurrentCommand[])-1,"",0,Dummy[0]);
if (CurrentCommand[id][0]==0) // This should never happen, but just incase..
{
client_print(id,print_chat,"[AMXX] There was an error extracting the command name.");
return PLUGIN_HANDLED;
}
// TODO: ML this
client_print(id,print_chat,"[AMXX] Type in the parameters for %s, or !cancel to cancel.",CurrentCommand[id]);
client_cmd(id,"messagemode amx_executecmd");
menu_destroy(menu);
return PLUGIN_HANDLED; // Don't return to original menu immediately!
}
else if (item==1) // "No params"
{
menu_item_getinfo(menu, item, Dummy[0], CurrentCommand[id], sizeof(CurrentCommand[])-1,"",0,Dummy[0]);
if (CurrentCommand[id][0]==0) // This should never happen, but just incase..
{
client_print(id,print_chat,"[AMXX] There was an error extracting the command name.");
return PLUGIN_HANDLED;
}
// TODO: ML this
// Now redraw the menu for the client BEFORE the command is executed, incase
// that menu brings up a menu of its own.
if (CurrentMenuFunction[id]!=-1 && callfunc_begin_i(CurrentMenuFunction[id])==1)
{
callfunc_push_int(id);
callfunc_push_int(CurrentPlid[id]);
callfunc_push_int(CurrentPage[id]);
callfunc_end();
}
menu_destroy(menu);
client_cmd(id,"%s",CurrentCommand[id]);
client_print(id,print_chat,"[AMXX] Command ^"%s^" executed with no parameters",CurrentCommand[id]);
return PLUGIN_HANDLED;
}
// We should never get here, but just incase..
menu_destroy(menu);
return PLUGIN_HANDLED;
}
/**
* Generates and displays a menu to the client for a specific command.
*
* @param id The client to display the menu to.
* @param cid The command id to display.
*/
stock DisplaySpecificCommand(id,cid)
{
new CommandName[64];
new CommandDesc[128];
new CommandTitle[256];
new CommandAccess;
new Menu;
get_concmd(cid,CommandName,sizeof(CommandName)-1,CommandAccess, CommandDesc,sizeof(CommandDesc)-1, -1, -1);
if (CommandDesc[0]!='^0')
{
formatex(CommandTitle,sizeof(CommandTitle)-1,"%s^n%s",CommandName,CommandDesc);
Menu=menu_create(CommandTitle,"SpecificCommandHandler");
}
else
{
Menu=menu_create(CommandName,"SpecificCommandHandler");
}
menu_additem(Menu,"Execute with parameters.",CommandName,_,EnabledCallback);
menu_additem(Menu,"Execute with no parameters.",CommandName,_,EnabledCallback);
#if !defined BUILD_FOR_176
menu_setprop(Menu,MPROP_NUMBER_COLOR,"\y");
#endif
menu_display(id,Menu,0);
}
/**
* Handles the executed command (via "amx_executecmd").
*
* @param id The id of the client who executed this.
*/
public CommandExecuteCommand(id)
{
// If they had no command stored, then just ignore it entirely.
if (CurrentCommand[id][0]=='^0')
{
return PLUGIN_CONTINUE;
}
new Args[256];
read_args(Args,sizeof(Args)-1);
remove_quotes(Args);
if (equali(Args,"!cancel",7))
{
// The client didn't want to execute this command.
client_print(id,print_chat,"[AMXX] Command not executed.");
// Now redraw the menu for the client
if (CurrentMenuFunction[id]!=-1 && callfunc_begin_i(CurrentMenuFunction[id])==1)
{
callfunc_push_int(id);
callfunc_push_int(CurrentPlid[id]);
callfunc_push_int(CurrentPage[id]);
callfunc_end();
}
}
else
{
// TODO: ML
client_print(id,print_chat,"[AMXX] Command ^"%s^" executed with ^"%s^"",CurrentCommand[id],Args);
// Now redraw the menu for the client
if (CurrentMenuFunction[id]!=-1 && callfunc_begin_i(CurrentMenuFunction[id])==1)
{
callfunc_push_int(id);
callfunc_push_int(CurrentPlid[id]);
callfunc_push_int(CurrentPage[id]);
callfunc_end();
}
// Execute the command on the client.
client_cmd(id,"%s %s",CurrentCommand[id],Args);
}
return PLUGIN_HANDLED;
}
/**
* Handle a specific selection from the command menu.
*
* @param id id of the client who made the selection.
* @param menu The menu handle.
* @param item The item that was selected.
*/
public CommandMenuSelection(id, menu, item)
{
if (item==MENU_EXIT)
{
menu_destroy(menu);
// If the player did not explicitly specify a plugin, return them to the
// plugin selection menu.
if (ExplicitPlugin[id]==-1)
{
client_cmd(id,"amx_plugincmdmenu");
}
return PLUGIN_HANDLED;
}
else if (item==MENU_BACK)
{
--CurrentPage[id];
client_print(id,print_chat,"MENU_BACK");
return PLUGIN_HANDLED;
}
else if (item==MENU_MORE)
{
++CurrentPage[id];
client_print(id,print_chat,"MENU_MORE");
return PLUGIN_HANDLED;
}
else
{
new Command[32];
new Dummy[1];
// pcvar pointer is stored in command, extract the name of the cvar from the name field.
menu_item_getinfo(menu, item, Dummy[0], Command, sizeof(Command)-1,"",0,Dummy[0]);
menu_destroy(menu);
DisplaySpecificCommand(id,str_to_num(Command));
return PLUGIN_HANDLED;
}
return 0;
}
/**
* This blocks "say" and "say_team" commands.
* Other commands that shouldn't be displayed (eg: amxauth<stuff>) should be filtered out already.
*
* @param Command The command that is being checked.
*/
stock bool:IsDisplayableCmd(const Command[])
{
// Block "say" and "say_team"
if (equal(Command,"say",3))
{
return false;
}
return true;
}
/**
* Displays a command list for the specified plugin.
*
* @param id Id of the client that's being displayed to.
* @param plid Plugin ID of the plugin that is to be displayed.
* @param page The page to start at.
*/
public DisplayCmdMenu(id, plid, page)
{
new PluginName[32];
new MenuTitle[64];
get_plugin(plid,"",0,PluginName,sizeof(PluginName)-1,"",0,"",0,"",0);
formatex(MenuTitle,sizeof(MenuTitle)-1,"%s Commands:",PluginName);
new Menu=menu_create(MenuTitle,"CommandMenuSelection");
new Command[64];
new CidString[32];
new CommandAccess;
new userflags=get_user_flags(id);
new bool:isadmin=bool:is_user_admin(id);
for (new i=0, max=get_concmdsnum(-1,-1);
i<max;
i++)
{
if (get_concmd_plid(i,-1,-1)==plid)
{
get_concmd(i,Command,sizeof(Command)-1,CommandAccess, "",0, -1, -1);
if (IsDisplayableCmd(Command))
{
if ( userflags & CommandAccess ||
(CommandAccess==ADMIN_ADMIN && isadmin) ||
CommandAccess==ADMIN_USER ||
CommandAccess==ADMIN_ALL)
{
num_to_str(i,CidString,sizeof(CidString)-1);
menu_additem(Menu,Command,CidString,0,EnabledCallback);
}
else
{
menu_additem(Menu,Command,"",0,DisabledCallback);
}
}
}
}
#if !defined BUILD_FOR_176
menu_setprop(Menu,MPROP_NUMBER_COLOR,"\y");
#endif
menu_display(id,Menu,page);
}
/**
* Handles the "amx_plugincmdmenu" command.
*
* @param id Id of the client that's being checked.
* @param level Access level of the command.
* @param cid Command ID of the command that was executed.
*/
public CommandMenuCommand(id, level, cid)
{
if (!cmd_access(id,level,cid,0))
{
return PLUGIN_HANDLED;
}
// This is which plugin to display. -1 means display all plugins in a list.
new plid=-1;
if (GetPlidForValidPlugins(id,plid)!=true)
{
// If GetPlidForValidPlugins returns false then it failed to find the plugin.
return PLUGIN_HANDLED;
}
// Check if we were passed a specific plugin to display or not.
if (plid==-1)
{
// We need to display a list of the plugins, instead of a specific plugin.
ExplicitPlugin[id]=-1;
DisplayPluginMenu(id,"Plugin Command Menu:", "PluginMenuSelection","DisplayCmdMenu","GetNumberOfCmdsForPlid");
}
else
{
ExplicitPlugin[id]=plid;
CurrentPlid[id]=plid;
CurrentPage[id]=0;
DisplayCmdMenu(id,plid,0);
}
return PLUGIN_HANDLED;
}