sm-plugins/GlowColors/scripting/GlowColors.sp
2018-06-20 14:59:13 +02:00

536 lines
14 KiB
SourcePawn

#pragma semicolon 1
#pragma newdecls required
#include <sourcemod>
#include <sdktools>
#include <sdkhooks>
#include <clientprefs>
#include <regex>
#include <dhooks>
#undef REQUIRE_PLUGIN
#include <zombiereloaded>
#define REQUIRE_PLUGIN
public Plugin myinfo =
{
name = "GlowColors",
author = "BotoX",
description = "Change your clients colors.",
version = "1.2",
url = ""
}
// bool CBaseEntity::AcceptInput( const char *szInputName, CBaseEntity *pActivator, CBaseEntity *pCaller, variant_t Value, int outputID )
Handle g_hAcceptInput;
Menu g_GlowColorsMenu;
Handle g_hClientCookie = INVALID_HANDLE;
ConVar g_Cvar_MinBrightness;
Regex g_Regex_RGB;
Regex g_Regex_HEX;
int g_aGlowColor[MAXPLAYERS + 1][3];
float g_aRainbowFrequency[MAXPLAYERS + 1];
public void OnPluginStart()
{
Handle hGameConf = LoadGameConfigFile("sdktools.games");
if(hGameConf == INVALID_HANDLE)
{
SetFailState("Couldn't load sdktools game config!");
return;
}
int Offset = GameConfGetOffset(hGameConf, "AcceptInput");
g_hAcceptInput = DHookCreate(Offset, HookType_Entity, ReturnType_Bool, ThisPointer_CBaseEntity, AcceptInput);
DHookAddParam(g_hAcceptInput, HookParamType_CharPtr);
DHookAddParam(g_hAcceptInput, HookParamType_CBaseEntity);
DHookAddParam(g_hAcceptInput, HookParamType_CBaseEntity);
DHookAddParam(g_hAcceptInput, HookParamType_Object, 20, DHookPass_ByVal|DHookPass_ODTOR|DHookPass_OCTOR|DHookPass_OASSIGNOP); //varaint_t is a union of 12 (float[3]) plus two int type params 12 + 8 = 20
DHookAddParam(g_hAcceptInput, HookParamType_Int);
CloseHandle(hGameConf);
g_hClientCookie = RegClientCookie("glowcolor", "", CookieAccess_Protected);
g_Regex_RGB = CompileRegex("^(([01]?[0-9]?[0-9]|2[0-4][0-9]|25[0-5])\\s+){2}([01]?[0-9]?[0-9]|2[0-4][0-9]|25[0-5])$");
g_Regex_HEX = CompileRegex("^(#?)([A-Fa-f0-9]{6})$");
RegAdminCmd("sm_glowcolors", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_glowcolors <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_glowcolours", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_glowcolours <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_glowcolor", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_glowcolor <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_glowcolour", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_glowcolour <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_colors", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_colors <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_colours", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_colours <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_color", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_color <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_colour", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_colour <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_glow", Command_GlowColors, ADMFLAG_CUSTOM1, "Change your players glowcolor. sm_glow <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>");
RegAdminCmd("sm_rainbow", Command_Rainbow, ADMFLAG_CUSTOM1, "Enable rainbow glowcolors. sm_rainbow [frequency]");
HookEvent("player_spawn", Event_ApplyGlowColor, EventHookMode_Post);
HookEvent("player_team", Event_ApplyGlowColor, EventHookMode_Post);
g_Cvar_MinBrightness = CreateConVar("sm_glowcolor_minbrightness", "100", "Lowest brightness value for glowcolor.", 0, true, 0.0, true, 255.0);
AutoExecConfig(true, "plugin.GlowColors");
LoadConfig();
for(int client = 1; client <= MaxClients; client++)
{
if(IsClientInGame(client))
{
OnClientPutInServer(client);
if(!IsFakeClient(client) && AreClientCookiesCached(client))
{
OnClientCookiesCached(client);
ApplyGlowColor(client);
}
}
}
}
public void OnPluginEnd()
{
for(int client = 1; client <= MaxClients; client++)
{
if(IsClientInGame(client) && !IsFakeClient(client) && AreClientCookiesCached(client))
{
OnClientDisconnect(client);
ApplyGlowColor(client);
}
}
delete g_GlowColorsMenu;
CloseHandle(g_hClientCookie);
}
void LoadConfig()
{
char sConfigFile[PLATFORM_MAX_PATH];
BuildPath(Path_SM, sConfigFile, sizeof(sConfigFile), "configs/GlowColors.cfg");
if(!FileExists(sConfigFile))
{
SetFailState("Could not find config: \"%s\"", sConfigFile);
}
KeyValues Config = new KeyValues("GlowColors");
if(!Config.ImportFromFile(sConfigFile))
{
delete Config;
SetFailState("ImportFromFile() failed!");
}
if(!Config.GotoFirstSubKey(false))
{
delete Config;
SetFailState("GotoFirstSubKey() failed!");
}
g_GlowColorsMenu = new Menu(MenuHandler_GlowColorsMenu, MenuAction_Select);
g_GlowColorsMenu.SetTitle("GlowColors");
g_GlowColorsMenu.ExitButton = true;
g_GlowColorsMenu.AddItem("255 255 255", "None");
char sKey[32];
char sValue[16];
do
{
Config.GetSectionName(sKey, sizeof(sKey));
Config.GetString(NULL_STRING, sValue, sizeof(sValue));
g_GlowColorsMenu.AddItem(sValue, sKey);
}
while(Config.GotoNextKey(false));
}
public void OnClientPutInServer(int client)
{
g_aGlowColor[client][0] = 255;
g_aGlowColor[client][1] = 255;
g_aGlowColor[client][2] = 255;
g_aRainbowFrequency[client] = 0.0;
DHookEntity(g_hAcceptInput, false, client);
}
public void OnClientCookiesCached(int client)
{
if(IsClientAuthorized(client))
ReadClientCookies(client);
}
public void OnClientPostAdminCheck(int client)
{
if(AreClientCookiesCached(client))
ReadClientCookies(client);
}
void ReadClientCookies(int client)
{
char sCookie[16];
if(CheckCommandAccess(client, "sm_glowcolors", ADMFLAG_CUSTOM1))
GetClientCookie(client, g_hClientCookie, sCookie, sizeof(sCookie));
if(StrEqual(sCookie, ""))
{
g_aGlowColor[client][0] = 255;
g_aGlowColor[client][1] = 255;
g_aGlowColor[client][2] = 255;
}
else
ColorStringToArray(sCookie, g_aGlowColor[client]);
}
public void OnClientDisconnect(int client)
{
if(CheckCommandAccess(client, "sm_glowcolors", ADMFLAG_CUSTOM1))
{
if(g_aGlowColor[client][0] == 255 &&
g_aGlowColor[client][1] == 255 &&
g_aGlowColor[client][2] == 255)
{
SetClientCookie(client, g_hClientCookie, "");
}
else
{
char sCookie[16];
FormatEx(sCookie, sizeof(sCookie), "%d %d %d",
g_aGlowColor[client][0],
g_aGlowColor[client][1],
g_aGlowColor[client][2]);
SetClientCookie(client, g_hClientCookie, sCookie);
}
}
g_aGlowColor[client][0] = 255;
g_aGlowColor[client][1] = 255;
g_aGlowColor[client][2] = 255;
if(g_aRainbowFrequency[client])
SDKUnhook(client, SDKHook_PostThinkPost, OnPostThinkPost);
g_aRainbowFrequency[client] = 0.0;
}
public void OnPostThinkPost(int client)
{
float i = GetGameTime();
float Frequency = g_aRainbowFrequency[client];
int Red = RoundFloat(Sine(Frequency * i + 0.0) * 127.0 + 128.0);
int Green = RoundFloat(Sine(Frequency * i + 2.0943951) * 127.0 + 128.0);
int Blue = RoundFloat(Sine(Frequency * i + 4.1887902) * 127.0 + 128.0);
ToolsSetEntityColor(client, Red, Green, Blue);
}
// bool CBaseEntity::AcceptInput( const char *szInputName, CBaseEntity *pActivator, CBaseEntity *pCaller, variant_t Value, int outputID )
public MRESReturn AcceptInput(int pThis, Handle hReturn, Handle hParams)
{
// Should not happen?
if(DHookIsNullParam(hParams, 2))
return MRES_Ignored;
int client = EntRefToEntIndex(DHookGetParam(hParams, 2));
if(client < 1 || client > MAXPLAYERS)
return MRES_Ignored;
char szInputName[32];
DHookGetParamString(hParams, 1, szInputName, sizeof(szInputName));
if(!StrEqual(szInputName, "addoutput", false))
return MRES_Ignored;
char sValue[128];
DHookGetParamObjectPtrString(hParams, 4, 0, ObjectValueType_String, sValue, sizeof(sValue));
int iValueLen = strlen(sValue);
int aArgs[4] = {0, ...};
int iArgs = 0;
bool bFound = false;
for(int i = 0; i < iValueLen; i++)
{
if(sValue[i] == ' ')
{
if(bFound)
{
sValue[i] = '\0';
bFound = false;
if(iArgs > sizeof(aArgs))
break;
}
continue;
}
if(!bFound)
{
aArgs[iArgs++] = i;
bFound = true;
}
}
if(strncmp(sValue[aArgs[0]], "rendercolor", 11, false) == 0)
{
int aColor[3];
aColor[0] = StringToInt(sValue[aArgs[1]]) & 0xFF;
aColor[1] = StringToInt(sValue[aArgs[2]]) & 0xFF;
aColor[2] = StringToInt(sValue[aArgs[3]]) & 0xFF;
if(aColor[0] == 255 && aColor[1] == 255 && aColor[2] == 255)
{
ApplyGlowColor(client);
DHookSetReturn(hReturn, true);
return MRES_Supercede;
}
}
else if(StrEqual(sValue[aArgs[0]], "rendermode", false))
{
RenderMode renderMode = view_as<RenderMode>(StringToInt(sValue[aArgs[1]]) & 0xFF);
if(renderMode == RENDER_NORMAL)
{
ApplyGlowColor(client);
return MRES_Ignored;
}
}
return MRES_Ignored;
}
public Action Command_GlowColors(int client, int args)
{
if(args < 1)
{
DisplayGlowColorMenu(client);
return Plugin_Handled;
}
int Color;
if(args == 1)
{
char sColorString[32];
GetCmdArgString(sColorString, sizeof(sColorString));
if(!IsValidHex(sColorString))
{
PrintToChat(client, "Invalid HEX color code supplied.");
return Plugin_Handled;
}
Color = StringToInt(sColorString, 16);
g_aGlowColor[client][0] = (Color >> 16) & 0xFF;
g_aGlowColor[client][1] = (Color >> 8) & 0xFF;
g_aGlowColor[client][2] = (Color >> 0) & 0xFF;
}
else if(args == 3)
{
char sColorString[32];
GetCmdArgString(sColorString, sizeof(sColorString));
if(!IsValidRGBNum(sColorString))
{
PrintToChat(client, "Invalid RGB color code supplied.");
return Plugin_Handled;
}
ColorStringToArray(sColorString, g_aGlowColor[client]);
Color = (g_aGlowColor[client][0] << 16) +
(g_aGlowColor[client][1] << 8) +
(g_aGlowColor[client][2] << 0);
}
else
{
char sCommand[32];
GetCmdArg(0, sCommand, sizeof(sCommand));
PrintToChat(client, "[SM] Usage: %s <RRGGBB HEX | 0-255 0-255 0-255 RGB CODE>", sCommand);
return Plugin_Handled;
}
if(!ApplyGlowColor(client))
return Plugin_Handled;
if(GetCmdReplySource() == SM_REPLY_TO_CHAT)
PrintToChat(client, "\x01[SM] Set color to: \x07%06X%06X\x01", Color, Color);
return Plugin_Handled;
}
public Action Command_Rainbow(int client, int args)
{
float Frequency = 1.0;
if(args >= 1)
{
char sArg[32];
GetCmdArg(1, sArg, sizeof(sArg));
Frequency = StringToFloat(sArg);
if(Frequency > 10.0)
Frequency = 10.0;
}
if(!Frequency || (args < 1 && g_aRainbowFrequency[client]))
{
if(g_aRainbowFrequency[client])
SDKUnhook(client, SDKHook_PostThinkPost, OnPostThinkPost);
g_aRainbowFrequency[client] = 0.0;
PrintToChat(client, "[SM] Disabled rainbow glowcolors.");
ApplyGlowColor(client);
}
else
{
if(!g_aRainbowFrequency[client])
SDKHook(client, SDKHook_PostThinkPost, OnPostThinkPost);
g_aRainbowFrequency[client] = Frequency;
PrintToChat(client, "[SM] Enabled rainbow glowcolors. (Frequency = %f)", Frequency);
}
return Plugin_Handled;
}
void DisplayGlowColorMenu(int client)
{
g_GlowColorsMenu.Display(client, MENU_TIME_FOREVER);
}
public int MenuHandler_GlowColorsMenu(Menu menu, MenuAction action, int param1, int param2)
{
switch(action)
{
case MenuAction_Select:
{
char aItem[16];
menu.GetItem(param2, aItem, sizeof(aItem));
ColorStringToArray(aItem, g_aGlowColor[param1]);
int Color = (g_aGlowColor[param1][0] << 16) +
(g_aGlowColor[param1][1] << 8) +
(g_aGlowColor[param1][2] << 0);
ApplyGlowColor(param1);
PrintToChat(param1, "\x01[SM] Set color to: \x07%06X%06X\x01", Color, Color);
}
}
}
public void Event_ApplyGlowColor(Event event, const char[] name, bool dontBroadcast)
{
int client = GetClientOfUserId(GetEventInt(event, "userid"));
if(!client)
return;
CreateTimer(0.1, Timer_ApplyGlowcolor, client, TIMER_FLAG_NO_MAPCHANGE);
}
public void ZR_OnClientInfected(int client, int attacker, bool motherInfect, bool respawnOverride, bool respawn)
{
ApplyGlowColor(client);
}
public void ZR_OnClientHumanPost(int client, bool respawn, bool protect)
{
ApplyGlowColor(client);
}
public Action Timer_ApplyGlowcolor(Handle timer, int client)
{
ApplyGlowColor(client);
return Plugin_Stop;
}
bool ApplyGlowColor(int client)
{
if(!IsClientInGame(client))
return false;
bool Ret = true;
int Brightness = ColorBrightness(g_aGlowColor[client][0], g_aGlowColor[client][1], g_aGlowColor[client][2]);
if(Brightness < g_Cvar_MinBrightness.IntValue)
{
PrintToChat(client, "Your glowcolor is too dark! (brightness = %d/255, allowed values are >= %d)",
Brightness, g_Cvar_MinBrightness.IntValue);
g_aGlowColor[client][0] = 255;
g_aGlowColor[client][1] = 255;
g_aGlowColor[client][2] = 255;
Ret = false;
}
if(IsPlayerAlive(client))
ToolsSetEntityColor(client, g_aGlowColor[client][0], g_aGlowColor[client][1], g_aGlowColor[client][2]);
return Ret;
}
stock void ToolsGetEntityColor(int entity, int aColor[4])
{
static bool s_GotConfig = false;
static char s_sProp[32];
if(!s_GotConfig)
{
Handle GameConf = LoadGameConfigFile("core.games");
bool Exists = GameConfGetKeyValue(GameConf, "m_clrRender", s_sProp, sizeof(s_sProp));
CloseHandle(GameConf);
if(!Exists)
strcopy(s_sProp, sizeof(s_sProp), "m_clrRender");
s_GotConfig = true;
}
int Offset = GetEntSendPropOffs(entity, s_sProp);
for(int i = 0; i < 4; i++)
aColor[i] = GetEntData(entity, Offset + i, 1);
}
stock void ToolsSetEntityColor(int client, int Red, int Green, int Blue)
{
int aColor[4];
ToolsGetEntityColor(client, aColor);
SetEntityRenderColor(client, Red, Green, Blue, aColor[3]);
}
stock void ColorStringToArray(const char[] sColorString, int aColor[3])
{
char asColors[4][4];
ExplodeString(sColorString, " ", asColors, sizeof(asColors), sizeof(asColors[]));
aColor[0] = StringToInt(asColors[0]) & 0xFF;
aColor[1] = StringToInt(asColors[1]) & 0xFF;
aColor[2] = StringToInt(asColors[2]) & 0xFF;
}
stock bool IsValidRGBNum(char[] sString)
{
if(g_Regex_RGB.Match(sString) > 0)
return true;
return false;
}
stock bool IsValidHex(char[] sString)
{
if(g_Regex_HEX.Match(sString) > 0)
return true;
return false;
}
stock int ColorBrightness(int Red, int Green, int Blue)
{
// http://www.nbdtech.com/Blog/archive/2008/04/27/Calculating-the-Perceived-Brightness-of-a-Color.aspx
return RoundToFloor(SquareRoot(
Red * Red * 0.241 +
Green * Green + 0.691 +
Blue * Blue + 0.068));
}