sm-plugins/_WebShortcuts/scripting/WebShortcuts.sp
2019-07-25 00:14:52 +02:00

965 lines
26 KiB
SourcePawn

#pragma semicolon 1
#include <sourcemod>
#tryinclude <steamtools>
#define PLUGIN_VERSION "1.1"
#define PLUGIN_DESCRIPTION "Redux of Web Shortcuts and Dynamic MOTD Functionality"
public Plugin:myinfo =
{
name = "Web Shortcuts", /* https://www.youtube.com/watch?v=h6k5jwllFfA&hd=1 */
author = "Kyle Sanderson, Nicholas Hastings",
description = PLUGIN_DESCRIPTION,
version = PLUGIN_VERSION,
url = "http://SourceMod.net"
};
enum _:States {
Game_TF2 = (1<<0),
Game_L4D = (1<<1),
Big_MOTD = (1<<8)
};
new g_iGameMode;
enum _:FieldCheckFlags
{
Flag_Steam_ID = (1<<0),
Flag_User_ID = (1<<1),
Flag_Friend_ID = (1<<2),
Flag_Name = (1<<3),
Flag_IP = (1<<4),
Flag_Language = (1<<5),
Flag_Rate = (1<<6),
Flag_Server_IP = (1<<7),
Flag_Server_Port = (1<<8),
Flag_Server_Name = (1<<9),
Flag_Server_Custom = (1<<10),
Flag_L4D_GameMode = (1<<11),
Flag_Current_Map = (1<<12),
Flag_Next_Map = (1<<13),
Flag_GameDir = (1<<14),
Flag_CurPlayers = (1<<15),
#if defined _steamtools_included
Flag_MaxPlayers = (1<<16),
Flag_VACStatus = (1<<17),
Flag_Server_Pub_IP = (1<<18),
Flag_Steam_ConnStatus = (1<<19)
#else
Flag_MaxPlayers = (1<<16)
#endif /* _steamtools_included */
};
#define IsTeamFortress2() (g_iGameMode & Game_TF2)
#define IsLeftForDead() (g_iGameMode & Game_L4D)
#define GoLargeOrGoHome() (IsTeamFortress2() && (g_iGameMode & Big_MOTD))
/*#include "Duck"*/
new Handle:g_hIndexArray = INVALID_HANDLE;
new Handle:g_hFastLookupTrie = INVALID_HANDLE;
new Handle:g_hCurrentTrie = INVALID_HANDLE;
new String:g_sCurrentSection[128];
public OnPluginStart()
{
g_hIndexArray = CreateArray(); /* We'll only use this for cleanup to prevent handle leaks and what not.
Our friend below doesn't have iteration, so we have to do this... */
g_hFastLookupTrie = CreateTrie();
AddCommandListener(Client_Say, "say");
AddCommandListener(Client_Say, "say_team");
/* From Psychonic */
Duck_OnPluginStart();
new Handle:cvarVersion = CreateConVar("webshortcutsredux_version", PLUGIN_VERSION, PLUGIN_DESCRIPTION, FCVAR_PLUGIN|FCVAR_NOTIFY);
/* On a reload, this will be set to the old version. Let's update it. */
SetConVarString(cvarVersion, PLUGIN_VERSION);
}
public Action:Client_Say(iClient, const String:sCommand[], argc)
{
if (argc < 1 || !IsValidClient(iClient))
{
return Plugin_Continue; /* Well. While we can probably have blank hooks, I doubt anyone wants this. Lets not waste cycles. Let the game deal with this. */
}
decl String:sFirstArg[64]; /* If this is too small, let someone know. */
GetCmdArg(1, sFirstArg, sizeof(sFirstArg));
TrimString(sFirstArg);
new Handle:hStoredTrie = INVALID_HANDLE;
if (!GetTrieValue(g_hFastLookupTrie, sFirstArg, hStoredTrie) || hStoredTrie == INVALID_HANDLE) /* L -> R. Strings are R -> L, but that can change. */
{
return Plugin_Continue; /* Didn't find anything. Bug out! */
}
if (DealWithOurTrie(iClient, sFirstArg, hStoredTrie))
{
return Plugin_Handled; /* We want other hooks to be called, I guess. We just don't want it to go to the game. */
}
return Plugin_Continue; /* Well this is embarasing. We didn't actually hook this. Or atleast didn't intend to. */
}
public bool:DealWithOurTrie(iClient, const String:sHookedString[], Handle:hStoredTrie)
{
decl String:sUrl[256];
if (!GetTrieString(hStoredTrie, "Url", sUrl, sizeof(sUrl)))
{
LogError("Unable to find a Url for: \"%s\".", sHookedString);
return false;
}
new iUrlBits;
if (!GetTrieValue(hStoredTrie, "UrlBits", iUrlBits))
{
iUrlBits = 0; /* That's fine, there are no replacements! Less work for us. */
}
decl String:sTitle[256];
new iTitleBits;
if (!GetTrieString(hStoredTrie, "Title", sTitle, sizeof(sTitle)))
{
sTitle[0] = '\0'; /* We don't really need a title. Don't worry, it's cool. */
iTitleBits = 0;
}
else
{
if (!GetTrieValue(hStoredTrie, "TitleBits", iTitleBits))
{
iTitleBits = 0; /* That's fine, there are no replacements! Less work for us. */
}
}
Duck_DoReplacements(iClient, sUrl, iUrlBits, sTitle, iTitleBits); /* Arrays are passed by reference. Variables are copied. */
new bool:bBig;
new bool:bNotSilent = true;
GetTrieValue(hStoredTrie, "Silent", bNotSilent);
if (GoLargeOrGoHome())
{
GetTrieValue(hStoredTrie, "Big", bBig);
}
decl String:sMessage[256];
if (GetTrieString(hStoredTrie, "Msg", sMessage, sizeof(sMessage)))
{
new iMsgBits;
GetTrieValue(hStoredTrie, "MsgBits", iMsgBits);
if (iMsgBits != 0)
{
Duck_DoReplacements(iClient, sMessage, iMsgBits, sMessage, 0); /* Lame Hack for now */
}
PrintToChatAll("%s", sMessage);
}
DisplayMOTDWithOptions(iClient, sTitle, sUrl, bBig, bNotSilent, MOTDPANEL_TYPE_URL);
return true;
}
public ClearExistingData()
{
new Handle:hHandle = INVALID_HANDLE;
for (new i = (GetArraySize(g_hIndexArray) - 1); i >= 0; i--)
{
hHandle = GetArrayCell(g_hIndexArray, i);
if (hHandle == INVALID_HANDLE)
{
continue;
}
delete hHandle;
}
ClearArray(g_hIndexArray);
ClearTrie(g_hFastLookupTrie);
}
public OnConfigsExecuted()
{
ClearExistingData();
decl String:sPath[256];
BuildPath(Path_SM, sPath, sizeof(sPath), "configs/Webshortcuts.txt");
if (!FileExists(sPath))
{
return;
}
ProcessFile(sPath);
}
public ProcessFile(const String:sPathToFile[])
{
new Handle:hSMC = SMC_CreateParser();
SMC_SetReaders(hSMC, SMCNewSection, SMCReadKeyValues, SMCEndSection);
new iLine;
new SMCError:ReturnedError = SMC_ParseFile(hSMC, sPathToFile, iLine); /* Calls the below functions, then execution continues. */
if (ReturnedError != SMCError_Okay)
{
decl String:sError[256];
SMC_GetErrorString(ReturnedError, sError, sizeof(sError));
if (iLine > 0)
{
LogError("Could not parse file (Line: %d, File \"%s\"): %s.", iLine, sPathToFile, sError);
delete hSMC; /* Sneaky Handles. */
return;
}
LogError("Parser encountered error (File: \"%s\"): %s.", sPathToFile, sError);
}
delete hSMC;
}
public SMCResult:SMCNewSection(Handle:smc, const String:name[], bool:opt_quotes)
{
if (!opt_quotes)
{
LogError("Invalid Quoting used with Section: %s.", name);
}
strcopy(g_sCurrentSection, sizeof(g_sCurrentSection), name);
if (GetTrieValue(g_hFastLookupTrie, name, g_hCurrentTrie))
{
return SMCParse_Continue;
}
else /* That's cool. Sounds like an initial insertion. Just wanted to make sure! */
{
g_hCurrentTrie = CreateTrie();
PushArrayCell(g_hIndexArray, g_hCurrentTrie); /* Don't be leakin */
SetTrieValue(g_hFastLookupTrie, name, g_hCurrentTrie);
SetTrieString(g_hCurrentTrie, "Name", name);
}
return SMCParse_Continue;
}
public SMCResult:SMCReadKeyValues(Handle:smc, const String:key[], const String:value[], bool:key_quotes, bool:value_quotes)
{
if (!key_quotes)
{
LogError("Invalid Quoting used with Key: \"%s\".", key);
}
else if (!value_quotes)
{
LogError("Invalid Quoting used with Key: \"%s\" Value: \"%s\".", key, value);
}
else if (g_hCurrentTrie == INVALID_HANDLE)
{
return SMCParse_Continue;
}
switch (key[0])
{
case 'p','P':
{
if (!StrEqual(key, "Pointer", false))
{
return SMCParse_Continue;
}
new iFindValue;
iFindValue = FindValueInArray(g_hIndexArray, g_hCurrentTrie);
if (iFindValue > -1)
{
RemoveFromArray(g_hIndexArray, iFindValue);
}
if (g_sCurrentSection[0] != '\0')
{
RemoveFromTrie(g_hFastLookupTrie, g_sCurrentSection);
}
delete g_hCurrentTrie; /* We're about to invalidate below */
if (GetTrieValue(g_hFastLookupTrie, value, g_hCurrentTrie))
{
SetTrieValue(g_hFastLookupTrie, g_sCurrentSection, g_hCurrentTrie, true);
return SMCParse_Continue;
}
g_hCurrentTrie = CreateTrie(); /* Ruhro, the thing this points to doesn't actually exist. Should we error or what? Nah, lets try and recover. */
PushArrayCell(g_hIndexArray, g_hCurrentTrie); /* Don't be losin handles */
SetTrieValue(g_hFastLookupTrie, g_sCurrentSection, g_hCurrentTrie, true);
SetTrieString(g_hCurrentTrie, "Name", g_sCurrentSection, true);
}
case 'u','U':
{
if (!StrEqual(key, "Url", false))
{
return SMCParse_Continue;
}
SetTrieString(g_hCurrentTrie, "Url", value, true);
new iBits;
Duck_CalcBits(value, iBits); /* Passed by Ref */
SetTrieValue(g_hCurrentTrie, "UrlBits", iBits, true);
}
case 'T','t':
{
if (!StrEqual(key, "Title", false))
{
return SMCParse_Continue;
}
SetTrieString(g_hCurrentTrie, "Title", value, true);
new iBits;
Duck_CalcBits(value, iBits); /* Passed by Ref */
SetTrieValue(g_hCurrentTrie, "TitleBits", iBits, true);
}
case 'b','B':
{
if (!GoLargeOrGoHome() || !StrEqual(key, "Big", false)) /* Maybe they don't know they can't use it? Oh well. Protect the silly. */
{
return SMCParse_Continue;
}
SetTrieValue(g_hCurrentTrie, "Big", TranslateToBool(value), true);
}
case 'h','H':
{
if (!StrEqual(key, "Hook", false))
{
return SMCParse_Continue;
}
SetTrieValue(g_hFastLookupTrie, value, g_hCurrentTrie, true);
}
case 's', 'S':
{
if (!StrEqual(key, "Silent", false))
{
return SMCParse_Continue;
}
SetTrieValue(g_hCurrentTrie, "Silent", !TranslateToBool(value), true);
}
case 'M', 'm':
{
if (!StrEqual(key, "Msg", false))
{
return SMCParse_Continue;
}
SetTrieString(g_hCurrentTrie, "Msg", value, true);
new iBits;
Duck_CalcBits(value, iBits); /* Passed by Ref */
SetTrieValue(g_hCurrentTrie, "MsgBits", iBits, true);
}
}
return SMCParse_Continue;
}
public SMCResult:SMCEndSection(Handle:smc)
{
g_hCurrentTrie = INVALID_HANDLE;
g_sCurrentSection[0] = '\0';
}
public bool:TranslateToBool(const String:sSource[])
{
switch(sSource[0])
{
case '0', 'n', 'N', 'f', 'F':
{
return false;
}
case '1', 'y', 'Y', 't', 'T', 's', 'S':
{
return true;
}
}
return false; /* Assume False */
}
public DisplayMOTDWithOptions(iClient, const String:sTitle[], const String:sUrl[], bool:bBig, bool:bNotSilent, iType)
{
new Handle:hKv = CreateKeyValues("motd");
if (bBig)
{
KvSetNum(hKv, "customsvr", 1);
}
KvSetNum(hKv, "type", iType);
if (sTitle[0] != '\0')
{
KvSetString(hKv, "title", sTitle);
}
if (sUrl[0] != '\0')
{
KvSetString(hKv, "msg", sUrl);
}
ShowVGUIPanel(iClient, "info", hKv, bNotSilent);
delete hKv;
}
static stock bool:IsValidClient(iClient)
{
return (0 < iClient <= MaxClients && IsClientInGame(iClient));
}
/* Psychonics Realm */
#define FIELD_CHECK(%1,%2);\
if (StrContains(source, %1) != -1) { field |= %2; }
#define TOKEN_STEAM_ID "{STEAM_ID}"
#define TOKEN_USER_ID "{USER_ID}"
#define TOKEN_FRIEND_ID "{FRIEND_ID}"
#define TOKEN_NAME "{NAME}"
#define TOKEN_IP "{IP}"
#define TOKEN_LANGUAGE "{LANGUAGE}"
#define TOKEN_RATE "{RATE}"
#define TOKEN_SERVER_IP "{SERVER_IP}"
#define TOKEN_SERVER_PORT "{SERVER_PORT}"
#define TOKEN_SERVER_NAME "{SERVER_NAME}"
#define TOKEN_SERVER_CUSTOM "{SERVER_CUSTOM}"
#define TOKEN_L4D_GAMEMODE "{L4D_GAMEMODE}"
#define TOKEN_CURRENT_MAP "{CURRENT_MAP}"
#define TOKEN_NEXT_MAP "{NEXT_MAP}"
#define TOKEN_GAMEDIR "{GAMEDIR}"
#define TOKEN_CURPLAYERS "{CURPLAYERS}"
#define TOKEN_MAXPLAYERS "{MAXPLAYERS}"
#if defined _steamtools_included
#define TOKEN_VACSTATUS "{VAC_STATUS}"
#define TOKEN_SERVER_PUB_IP "{SERVER_PUB_IP}"
#define TOKEN_STEAM_CONNSTATUS "{STEAM_CONNSTATUS}"
new g_bSteamTools;
#endif /* _steamtools_included */
/* Cached values */
new String:g_szServerIp[16];
new String:g_szServerPort[6];
/* These can all be larger but whole buffer holds < 128 */
new String:g_szServerName[128];
new String:g_szServerCustom[128];
new String:g_szL4DGameMode[128];
new String:g_szCurrentMap[128];
new String:g_szGameDir[64];
/*new Handle:g_hCmdQueue[MAXPLAYERS+1];*/
#if defined _steamtools_included
public Steam_FullyLoaded()
{
g_bSteamTools = true;
}
public OnLibraryRemoved(const String:sLibrary[])
{
if (!StrEqual(sLibrary, "SteamTools", false))
{
return;
}
g_bSteamTools = false;
}
#endif
public Duck_OnPluginStart()
{
decl String:sGameDir[64];
GetGameFolderName(sGameDir, sizeof(sGameDir));
if (!strncmp(sGameDir, "tf", 2, false) || !strncmp(sGameDir, "tf_beta", 7, false))
{
g_iGameMode |= Game_TF2;
g_iGameMode |= Big_MOTD;
}
/* On a reload, these will already be registered and could be set to non-default */
if (IsTeamFortress2())
{
/* AddCommandListener(Duck_TF2OnClose, "closed_htmlpage"); */
}
LongIPToString(GetConVarInt(FindConVar("hostip")), g_szServerIp);
GetConVarString(FindConVar("hostport"), g_szServerPort, sizeof(g_szServerPort));
new Handle:hostname = FindConVar("hostname");
decl String:szHostname[256];
GetConVarString(hostname, szHostname, sizeof(szHostname));
Duck_UrlEncodeString(g_szServerName, sizeof(g_szServerName), szHostname);
HookConVarChange(hostname, OnCvarHostnameChange);
decl String:szCustom[256];
new Handle:hCVARCustom = CreateConVar("WebShortcuts_Custom", "", "Custom String for this server.");
GetConVarString(hCVARCustom, szCustom, sizeof(szCustom));
Duck_UrlEncodeString(g_szServerCustom, sizeof(g_szServerCustom), szCustom);
HookConVarChange(hCVARCustom, OnCvarCustomChange);
new iSDKVersion = GuessSDKVersion();
if (iSDKVersion == SOURCE_SDK_LEFT4DEAD || iSDKVersion == SOURCE_SDK_LEFT4DEAD2)
{
g_iGameMode |= Game_L4D;
new Handle:hGameMode = FindConVar("mp_gamemode");
decl String:szGamemode[256];
GetConVarString(hGameMode, szGamemode, sizeof(szGamemode));
Duck_UrlEncodeString(g_szL4DGameMode, sizeof(g_szL4DGameMode), szGamemode);
HookConVarChange(hGameMode, OnCvarGamemodeChange);
}
Duck_UrlEncodeString(g_szGameDir, sizeof(g_szGameDir), sGameDir);
}
public OnMapStart()
{
decl String:sTempMap[sizeof(g_szCurrentMap)];
GetCurrentMap(sTempMap, sizeof(sTempMap));
Duck_UrlEncodeString(g_szCurrentMap, sizeof(g_szCurrentMap), sTempMap);
}
stock Duck_DoReplacements(iClient, String:sUrl[256], iUrlBits, String:sTitle[256], iTitleBits) /* Huge thanks to Psychonic */
{
if (iUrlBits & Flag_Steam_ID || iTitleBits & Flag_Steam_ID)
{
decl String:sSteamId[64];
if (GetClientAuthString(iClient, sSteamId, sizeof(sSteamId)))
{
ReplaceString(sSteamId, sizeof(sSteamId), ":", "%3a");
if (iTitleBits & Flag_Steam_ID)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_STEAM_ID, sSteamId);
if (iUrlBits & Flag_Steam_ID)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_STEAM_ID, sSteamId);
}
else
{
if (iTitleBits & Flag_Steam_ID)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_STEAM_ID, "");
if (iUrlBits & Flag_Steam_ID)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_STEAM_ID, "");
}
}
if (iUrlBits & Flag_User_ID || iTitleBits & Flag_User_ID)
{
decl String:sUserId[16];
IntToString(GetClientUserId(iClient), sUserId, sizeof(sUserId));
if (iTitleBits & Flag_User_ID)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_USER_ID, sUserId);
if (iUrlBits & Flag_User_ID)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_USER_ID, sUserId);
}
if (iUrlBits & Flag_Friend_ID || iTitleBits & Flag_Friend_ID)
{
decl String:sFriendId[64];
if (GetClientFriendID(iClient, sFriendId, sizeof(sFriendId)))
{
if (iTitleBits & Flag_Friend_ID)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_FRIEND_ID, sFriendId);
if (iUrlBits & Flag_Friend_ID)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_FRIEND_ID, sFriendId);
}
else
{
if (iTitleBits & Flag_Friend_ID)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_FRIEND_ID, "");
if (iUrlBits & Flag_Friend_ID)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_FRIEND_ID, "");
}
}
if (iUrlBits & Flag_Name || iTitleBits & Flag_Name)
{
decl String:sName[MAX_NAME_LENGTH];
if (GetClientName(iClient, sName, sizeof(sName)))
{
decl String:sEncName[sizeof(sName)*3];
Duck_UrlEncodeString(sEncName, sizeof(sEncName), sName);
if (iTitleBits & Flag_Name)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_NAME, sEncName);
if (iUrlBits & Flag_Name)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_NAME, sEncName);
}
else
{
if (iTitleBits & Flag_Name)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_NAME, "");
if (iUrlBits & Flag_Name)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_NAME, "");
}
}
if (iUrlBits & Flag_IP || iTitleBits & Flag_IP)
{
decl String:sClientIp[32];
if (GetClientIP(iClient, sClientIp, sizeof(sClientIp)))
{
if (iTitleBits & Flag_IP)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_IP, sClientIp);
if (iUrlBits & Flag_IP)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_IP, sClientIp);
}
else
{
if (iTitleBits & Flag_IP)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_IP, "");
if (iUrlBits & Flag_IP)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_IP, "");
}
}
if (iUrlBits & Flag_Language || iTitleBits & Flag_Language)
{
decl String:sLanguage[32];
if (GetClientInfo(iClient, "cl_language", sLanguage, sizeof(sLanguage)))
{
decl String:sEncLanguage[sizeof(sLanguage)*3];
Duck_UrlEncodeString(sEncLanguage, sizeof(sEncLanguage), sLanguage);
if (iTitleBits & Flag_Language)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_LANGUAGE, sEncLanguage);
if (iUrlBits & Flag_Language)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_LANGUAGE, sEncLanguage);
}
else
{
if (iTitleBits & Flag_Language)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_LANGUAGE, "");
if (iUrlBits & Flag_Language)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_LANGUAGE, "");
}
}
if (iUrlBits & Flag_Rate || iTitleBits & Flag_Rate)
{
decl String:sRate[16];
if (GetClientInfo(iClient, "rate", sRate, sizeof(sRate)))
{
/* due to iClient's rate being silly, this won't necessarily be all digits */
decl String:sEncRate[sizeof(sRate)*3];
Duck_UrlEncodeString(sEncRate, sizeof(sEncRate), sRate);
if (iTitleBits & Flag_Rate)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_RATE, sEncRate);
if (iUrlBits & Flag_Rate)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_RATE, sEncRate);
}
else
{
if (iTitleBits & Flag_Rate)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_RATE, "");
if (iUrlBits & Flag_Rate)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_RATE, "");
}
}
if (iTitleBits & Flag_Server_IP)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_IP, g_szServerIp);
if (iUrlBits & Flag_Server_IP)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_IP, g_szServerIp);
if (iTitleBits & Flag_Server_Port)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_PORT, g_szServerPort);
if (iUrlBits & Flag_Server_Port)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_PORT, g_szServerPort);
if (iTitleBits & Flag_Server_Name)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_NAME, g_szServerName);
if (iUrlBits & Flag_Server_Name)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_NAME, g_szServerName);
if (iTitleBits & Flag_Server_Custom)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_CUSTOM, g_szServerCustom);
if (iUrlBits & Flag_Server_Custom)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_CUSTOM, g_szServerCustom);
if (IsLeftForDead() && ((iUrlBits & Flag_L4D_GameMode) || (iTitleBits & Flag_L4D_GameMode)))
{
if (iTitleBits & Flag_L4D_GameMode)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_L4D_GAMEMODE, g_szL4DGameMode);
if (iUrlBits & Flag_L4D_GameMode)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_L4D_GAMEMODE, g_szL4DGameMode);
}
if (iTitleBits & Flag_Current_Map)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_CURRENT_MAP, g_szCurrentMap);
if (iUrlBits & Flag_Current_Map)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_CURRENT_MAP, g_szCurrentMap);
if (iUrlBits & Flag_Next_Map || iTitleBits & Flag_Next_Map)
{
decl String:szNextMap[PLATFORM_MAX_PATH];
if (GetNextMap(szNextMap, sizeof(szNextMap)))
{
if (iTitleBits & Flag_Next_Map)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_NEXT_MAP, szNextMap);
if (iUrlBits & Flag_Next_Map)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_NEXT_MAP, szNextMap);
}
else
{
if (iTitleBits & Flag_Next_Map)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_NEXT_MAP, "");
if (iUrlBits & Flag_Next_Map)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_NEXT_MAP, "");
}
}
if (iTitleBits & Flag_GameDir)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_GAMEDIR, g_szGameDir);
if (iUrlBits & Flag_GameDir)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_GAMEDIR, g_szGameDir);
if (iUrlBits & Flag_CurPlayers || iTitleBits & Flag_CurPlayers)
{
decl String:sCurPlayers[10];
IntToString(GetClientCount(false), sCurPlayers, sizeof(sCurPlayers));
if (iTitleBits & Flag_CurPlayers)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_CURPLAYERS, sCurPlayers);
if (iUrlBits & Flag_CurPlayers)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_CURPLAYERS, sCurPlayers);
}
if (iUrlBits & Flag_MaxPlayers || iTitleBits & Flag_MaxPlayers)
{
decl String:maxplayers[10];
IntToString(MaxClients, maxplayers, sizeof(maxplayers));
if (iTitleBits & Flag_MaxPlayers)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_MAXPLAYERS, maxplayers);
if (iUrlBits & Flag_MaxPlayers)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_MAXPLAYERS, maxplayers);
}
#if defined _steamtools_included
if (iUrlBits & Flag_VACStatus || iTitleBits & Flag_VACStatus)
{
if (g_bSteamTools && Steam_IsVACEnabled())
{
if (iTitleBits & Flag_VACStatus)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_VACSTATUS, "1");
if (iUrlBits & Flag_VACStatus)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_VACSTATUS, "1");
}
else
{
if (iTitleBits & Flag_VACStatus)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_VACSTATUS, "0");
if (iUrlBits & Flag_VACStatus)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_VACSTATUS, "0");
}
}
if (iUrlBits & Flag_Server_Pub_IP || iTitleBits & Flag_Server_Pub_IP)
{
if (g_bSteamTools)
{
decl ip[4];
decl String:sIPString[16];
Steam_GetPublicIP(ip);
FormatEx(sIPString, sizeof(sIPString), "%d.%d.%d.%d", ip[0], ip[1], ip[2], ip[3]);
if (iTitleBits & Flag_Server_Pub_IP)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_PUB_IP, sIPString);
if (iUrlBits & Flag_Server_Pub_IP)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_PUB_IP, sIPString);
}
else
{
if (iTitleBits & Flag_Server_Pub_IP)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_SERVER_PUB_IP, "");
if (iUrlBits & Flag_Server_Pub_IP)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_SERVER_PUB_IP, "");
}
}
if (iUrlBits & Flag_Steam_ConnStatus || iTitleBits & Flag_Steam_ConnStatus)
{
if (g_bSteamTools && Steam_IsConnected())
{
if (iTitleBits & Flag_Steam_ConnStatus)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_STEAM_CONNSTATUS, "1");
if (iUrlBits & Flag_Steam_ConnStatus)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_STEAM_CONNSTATUS, "1");
}
else
{
if (iTitleBits & Flag_Steam_ConnStatus)
ReplaceString(sTitle, sizeof(sTitle), TOKEN_STEAM_CONNSTATUS, "0");
if (iUrlBits & Flag_Steam_ConnStatus)
ReplaceString(sUrl, sizeof(sUrl), TOKEN_STEAM_CONNSTATUS, "0");
}
}
#endif /* _steamtools_included */
}
stock bool:GetClientFriendID(client, String:sFriendID[], size)
{
#if defined _steamtools_included
Steam_GetCSteamIDForClient(client, sFriendID, size);
#else
decl String:sSteamID[64];
if (!GetClientAuthString(client, sSteamID, sizeof(sSteamID)))
{
sFriendID[0] = '\0'; /* Sanitize incase the return isn't checked. */
return false;
}
TrimString(sSteamID); /* Just incase... */
if (StrEqual(sSteamID, "STEAM_ID_LAN", false))
{
sFriendID[0] = '\0';
return false;
}
decl String:toks[3][16];
ExplodeString(sSteamID, ":", toks, sizeof(toks), sizeof(toks[]));
new iServer = StringToInt(toks[1]);
new iAuthID = StringToInt(toks[2]);
new iFriendID = (iAuthID*2) + 60265728 + iServer;
if (iFriendID >= 100000000)
{
decl String:temp[12], String:carry[12];
FormatEx(temp, sizeof(temp), "%d", iFriendID);
FormatEx(carry, 2, "%s", temp);
new icarry = StringToInt(carry[0]);
new upper = 765611979 + icarry;
FormatEx(temp, sizeof(temp), "%d", iFriendID);
FormatEx(sFriendID, size, "%d%s", upper, temp[1]);
}
else
{
Format(sFriendID, size, "765611979%d", iFriendID);
}
#endif
return true;
}
Duck_CalcBits(const String:source[], &field)
{
field = 0;
FIELD_CHECK(TOKEN_STEAM_ID, Flag_Steam_ID);
FIELD_CHECK(TOKEN_USER_ID, Flag_User_ID);
FIELD_CHECK(TOKEN_FRIEND_ID, Flag_Friend_ID);
FIELD_CHECK(TOKEN_NAME, Flag_Name);
FIELD_CHECK(TOKEN_IP, Flag_IP);
FIELD_CHECK(TOKEN_LANGUAGE, Flag_Language);
FIELD_CHECK(TOKEN_RATE, Flag_Rate);
FIELD_CHECK(TOKEN_SERVER_IP, Flag_Server_IP);
FIELD_CHECK(TOKEN_SERVER_PORT, Flag_Server_Port);
FIELD_CHECK(TOKEN_SERVER_NAME, Flag_Server_Name);
FIELD_CHECK(TOKEN_SERVER_CUSTOM, Flag_Server_Custom);
if (IsLeftForDead())
{
FIELD_CHECK(TOKEN_L4D_GAMEMODE, Flag_L4D_GameMode);
}
FIELD_CHECK(TOKEN_CURRENT_MAP, Flag_Current_Map);
FIELD_CHECK(TOKEN_NEXT_MAP, Flag_Next_Map);
FIELD_CHECK(TOKEN_GAMEDIR, Flag_GameDir);
FIELD_CHECK(TOKEN_CURPLAYERS, Flag_CurPlayers);
FIELD_CHECK(TOKEN_MAXPLAYERS, Flag_MaxPlayers);
#if defined _steamtools_included
FIELD_CHECK(TOKEN_VACSTATUS, Flag_VACStatus);
FIELD_CHECK(TOKEN_SERVER_PUB_IP, Flag_Server_Pub_IP);
FIELD_CHECK(TOKEN_STEAM_CONNSTATUS, Flag_Steam_ConnStatus);
#endif
}
/* Courtesy of Mr. Asher Baker */
stock LongIPToString(ip, String:szBuffer[16])
{
FormatEx(szBuffer, sizeof(szBuffer), "%i.%i.%i.%i", (((ip & 0xFF000000) >> 24) & 0xFF), (((ip & 0x00FF0000) >> 16) & 0xFF), (((ip & 0x0000FF00) >> 8) & 0xFF), (((ip & 0x000000FF) >> 0) & 0xFF));
}
/* loosely based off of PHP's urlencode */
stock Duck_UrlEncodeString(String:output[], size, const String:input[])
{
new icnt = 0;
new ocnt = 0;
for(;;)
{
if (ocnt == size)
{
output[ocnt-1] = '\0';
return;
}
new c = input[icnt];
if (c == '\0')
{
output[ocnt] = '\0';
return;
}
// Use '+' instead of '%20'.
// Still follows spec and takes up less of our limited buffer.
if (c == ' ')
{
output[ocnt++] = '+';
}
else if ((c < '0' && c != '-' && c != '.') ||
(c < 'A' && c > '9') ||
(c > 'Z' && c < 'a' && c != '_') ||
(c > 'z' && c != '~'))
{
output[ocnt++] = '%';
Format(output[ocnt], size-strlen(output[ocnt]), "%x", c);
ocnt += 2;
}
else
{
output[ocnt++] = c;
}
icnt++;
}
}
public OnCvarHostnameChange(Handle:convar, const String:oldValue[], const String:newValue[])
{
Duck_UrlEncodeString(g_szServerName, sizeof(g_szServerName), newValue);
}
public OnCvarGamemodeChange(Handle:convar, const String:oldValue[], const String:newValue[])
{
Duck_UrlEncodeString(g_szL4DGameMode, sizeof(g_szL4DGameMode), newValue);
}
public OnCvarCustomChange(Handle:convar, const String:oldValue[], const String:newValue[])
{
Duck_UrlEncodeString(g_szServerCustom, sizeof(g_szServerCustom), newValue);
}