#pragma semicolon 1 #pragma newdecls required //#define DEBUG #define PLUGIN_VERSION "1.0" #include #include #include #include #include enum { Immune_None, Immune_TK = 1, Immune_RFF = 2 } bool lateLoaded, isFinaleEnding; // bool isPlayerTroll[MAXPLAYERS+1], isUnderAttack[MAXPLAYERS+1]; // ImmunityFlag immunityFlags[MAXPLAYERS+1]; // int iJoinTime[MAXPLAYERS+1]; // int iIdleStartTime[MAXPLAYERS+1]; // int iLastFFTime[MAXPLAYERS+1]; // int iJumpAttempts[MAXPLAYERS+1]; // float playerTotalDamageFF[MAXPLAYERS+1]; // float autoFFScaleFactor[MAXPLAYERS+1]; enum struct PlayerData { int joinTime; int idleStartTime; int lastFFTime; int jumpAttempts; int ffCount; int totalFFCount; float TKDamageBuffer; float totalDamageFF; float autoRFFScaleFactor; bool isTroll; bool underAttack; int immunityFlags; bool pendingAction; } PlayerData pData[MAXPLAYERS+1]; ConVar hForgivenessTime, hBanTime, hThreshold, hJoinTime, hTKAction, hSuicideAction, hSuicideLimit, hFFAutoScaleAmount, hFFAutoScaleForgivenessAmount, hFFAutoScaleMaxRatio, hFFAutoScaleIgnoreAdmins; char gamemode[64]; bool isEnabled = true; public Plugin myinfo = { name = "TK Stopper", author = "jackzmc", description = "", version = PLUGIN_VERSION, url = "https://github.com/Jackzmc/sourcemod-plugins" }; public APLRes AskPluginLoad2(Handle myself, bool late, char[] error, int err_max) { if(late) { lateLoaded = true; } return APLRes_Success; } public void OnPluginStart() { EngineVersion g_Game = GetEngineVersion(); if(g_Game != Engine_Left4Dead && g_Game != Engine_Left4Dead2) { SetFailState("This plugin is for L4D/L4D2 only."); } hForgivenessTime = CreateConVar("l4d2_tk_forgiveness_time", "15", "The minimum amount of time to pass (in seconds) where a player's previous accumulated teamkiller-detection FF is forgiven", FCVAR_NONE, true, 0.0); hBanTime = CreateConVar("l4d2_tk_bantime", "60", "How long in minutes should a player be banned for? 0 for permanently", FCVAR_NONE, true, 0.0); hThreshold = CreateConVar("l4d2_tk_ban_ff_threshold", "75.0", "How much damage does a player need to do before being instantly banned", FCVAR_NONE, true, 0.0); hJoinTime = CreateConVar("l4d2_tk_ban_join_time", "2", "Upto how many minutes should any new player's ff be disabled? Used for jump attempts detection. also", FCVAR_NONE, true, 0.0); hTKAction = CreateConVar("l4d2_tk_action", "3", "How should the TK be punished?\n0 = No action (No message), 1 = Kick, 2 = Instant Ban, 3 = Ban on disconnect", FCVAR_NONE, true, 0.0, true, 3.0); hSuicideAction = CreateConVar("l4d2_suicide_action", "3", "How should a suicider be punished?\n0 = No action (No message), 1 = Kick, 2 = Instant Ban, 3 = Ban on disconnect", FCVAR_NONE, true, 0.0, true, 3.0); hSuicideLimit = CreateConVar("l4d2_suicide_limit", "1", "How many attempts does a new joined player have until action is taken for suiciding?", FCVAR_NONE, true, 0.0); // Reverse FF Auto Scale hFFAutoScaleAmount = CreateConVar("l4d2_tk_auto_ff_rate", "0.02", "The rate at which auto reverse-ff is scaled by.", FCVAR_NONE, true, 0.0); hFFAutoScaleMaxRatio = CreateConVar("l4d2_tk_auto_ff_max_ratio", "5.0", "The maximum amount that the reverse ff can go. 0.0 for unlimited", FCVAR_NONE, true, 0.0); hFFAutoScaleForgivenessAmount = CreateConVar("l4d2_tk_auto_ff_forgive_rate", "0.05", "This amount times amount of minutes since last ff is removed from ff rate", FCVAR_NONE, true, 0.0); hFFAutoScaleIgnoreAdmins = CreateConVar("l4d2_tk_auto_ff_ignore_admins", "1", "Should automatic reverse ff ignore admins? 0 = Admins are subjected\n1 = Admins are excempt", FCVAR_NONE, true, 0.0, true, 1.0); ConVar hGamemode = FindConVar("mp_gamemode"); hGamemode.AddChangeHook(Event_GamemodeChange); Event_GamemodeChange(hGamemode, gamemode, gamemode); AutoExecConfig(true, "l4d2_tkstopper"); HookEvent("finale_vehicle_ready", Event_FinaleVehicleReady); HookEvent("player_team", Event_PlayerDisconnect); HookEvent("charger_carry_start", Event_ChargerCarry); HookEvent("charger_carry_end", Event_ChargerCarry); HookEvent("lunge_pounce", Event_HunterPounce); HookEvent("pounce_end", Event_HunterPounce); HookEvent("pounce_stopped", Event_HunterPounce); HookEvent("choke_start", Event_SmokerChoke); HookEvent("choke_end", Event_SmokerChoke); HookEvent("choke_stopped", Event_SmokerChoke); HookEvent("jockey_ride", Event_JockeyRide); HookEvent("jockey_ride_end", Event_JockeyRide); HookEvent("player_bot_replace", Event_PlayerToBot); HookEvent("bot_player_replace", Event_BotToPlayer); RegAdminCmd("sm_ignore", Command_IgnorePlayer, ADMFLAG_KICK, "Makes a player immune for any anti trolling detection or reverse-ff for a session"); RegAdminCmd("sm_tkinfo", Command_TKInfo, ADMFLAG_KICK, "Debug info for TKSTopper"); RegAdminCmd("sm_review", Command_TKInfo, ADMFLAG_KICK, "Review FF info for a player"); if(lateLoaded) { for(int i = 1; i <= MaxClients; i++) { if(IsClientConnected(i) && IsClientInGame(i) && GetClientTeam(i) == 2) { SDKHook(i, SDKHook_OnTakeDamage, Event_OnTakeDamage); } } } LoadTranslations("common.phrases"); } public void Event_GamemodeChange(ConVar cvar, const char[] oldValue, const char[] newValue) { cvar.GetString(gamemode, sizeof(gamemode)); if(StrEqual(gamemode, "coop")) { isEnabled = true; } else { isEnabled = false; } } /////////////////////////////////////////////////////////////////////////////// // Special Infected Events /////////////////////////////////////////////////////////////////////////////// public Action Event_ChargerCarry(Event event, const char[] name, bool dontBroadcast) { int victim = GetClientOfUserId(event.GetInt("victim")); if(victim) { pData[victim].underAttack = StrEqual(name, "charger_carry_start"); } return Plugin_Continue; } public Action Event_HunterPounce(Event event, const char[] name, bool dontBroadcast) { int victim = GetClientOfUserId(event.GetInt("victim")); if(victim) { pData[victim].underAttack = StrEqual(name, "lunge_pounce"); } return Plugin_Continue; } public Action Event_SmokerChoke(Event event, const char[] name, bool dontBroadcast) { int victim = GetClientOfUserId(event.GetInt("victim")); if(victim) { pData[victim].underAttack = StrEqual(name, "choke_start"); } return Plugin_Continue; } public Action Event_JockeyRide(Event event, const char[] name, bool dontBroadcast) { int victim = GetClientOfUserId(event.GetInt("victim")); if(victim) { pData[victim].underAttack = StrEqual(name, "jockey_ride"); } return Plugin_Continue; } /////////////////////////////////////////////////////////////////////////////// // IDLE /////////////////////////////////////////////////////////////////////////////// public Action Event_BotToPlayer(Event event, const char[] name, bool dontBroadcast) { int player = GetClientOfUserId(event.GetInt("player")); // ignore fake players (side product of creating bots) if (!IsValidClient(player) || (GetClientTeam(player) != 2 && GetClientTeam(player) != 3) || IsFakeClient(player)) return Plugin_Continue; // If a player has been idle for over 600s (10 min), reset to them "just joined" // Purpose: Some trolls idle till end and then attack @ escape, or "gain trust" if(GetTime() - pData[player].idleStartTime >= 600) { pData[player].joinTime = GetTime(); } return Plugin_Continue; } public Action Event_PlayerToBot(Event event, char[] name, bool dontBroadcast) { int player = GetClientOfUserId(event.GetInt("player")); pData[player].idleStartTime = GetTime(); return Plugin_Continue; } /////////////////////////////////////////////////////////////////////////////// // Misc events /////////////////////////////////////////////////////////////////////////////// public void Event_FinaleVehicleReady(Event event, const char[] name, bool dontBroadcast) { isFinaleEnding = true; for(int i = 1; i <= MaxClients; i++) { if(pData[i].isTroll && IsClientConnected(i) && IsClientInGame(i)) { PrintChatToAdmins("Note: %N is still marked as troll and will be banned after this game. Use \"/ignore tk\" to ignore them.", i); } } } public void OnMapEnd() { isFinaleEnding = false; } public void OnClientPutInServer(int client) { pData[client].joinTime = pData[client].idleStartTime = GetTime(); SDKHook(client, SDKHook_OnTakeDamage, Event_OnTakeDamage); } public void OnClientPostAdminCheck(int client) { if(GetUserAdmin(client) != INVALID_ADMIN_ID && hFFAutoScaleIgnoreAdmins.BoolValue) { pData[client].immunityFlags = Immune_TK | Immune_RFF; } } // Called on map changes, so only reset some variables: public void OnClientDisconnect(int client) { pData[client].TKDamageBuffer = 0.0; pData[client].jumpAttempts = 0; pData[client].underAttack = false; pData[client].ffCount = 0; } // Only clear things when they fully left on their own accord: public void Event_PlayerDisconnect(Event event, const char[] name, bool dontBroadcast) { if(!event.GetBool("disconnect") || !isEnabled) return; int client = GetClientOfUserId(event.GetInt("userid")); if(client > 0 && event.GetInt("team") <= 2) { if (pData[client].isTroll && !IsFakeClient(client)) { BanClient(client, hBanTime.IntValue, BANFLAG_AUTO | BANFLAG_AUTHID, "Excessive FF (Auto)", "Excessive Friendly Fire", "TKStopper"); pData[client].isTroll = false; } if(!IsFakeClient(client)) { float minutesSinceiLastFFTime = GetLastFFMinutes(client); float activeRate = GetActiveRate(client); PrintToConsoleAll("[TKStopper] FF Summary for %N:", client); PrintToConsoleAll("\t\t%.2f TK-FF buffer (%.2f total ff, %d freq.) | %.3f (buf %f) rFF rate | lastff %.1f min ago | %d suicide jumps", pData[client].TKDamageBuffer, pData[client].totalDamageFF, pData[client].totalFFCount, activeRate, pData[client].autoRFFScaleFactor, minutesSinceiLastFFTime, pData[client].jumpAttempts ); } pData[client].autoRFFScaleFactor = 0.0; pData[client].totalDamageFF = 0.0; pData[client].ffCount = 0; pData[client].immunityFlags = 0; pData[client].totalFFCount = 0; } } public Action Event_OnTakeDamage(int victim, int& attacker, int& inflictor, float& damage, int& damagetype, int& weapon, float damageForce[3], float damagePosition[3]) { if(isEnabled && damage > 0.0 && victim <= MaxClients && attacker <= MaxClients && attacker > 0 && victim > 0) { if(GetClientTeam(victim) != GetClientTeam(attacker) || attacker == victim) return Plugin_Continue; else if(damagetype & DMG_BURN && IsFakeClient(attacker) && GetClientTeam(attacker) == 2) { // Ignore damage from fire caused by bots (players who left after causing fire) damage = 0.0; return Plugin_Changed; } // Otherwise if attacker was ignored or is a bot, stop here and let vanilla handle it else if(pData[attacker].immunityFlags & Immune_TK || IsFakeClient(attacker)) return Plugin_Continue; bool isAdmin = GetUserAdmin(attacker) != INVALID_ADMIN_ID; //Allow friendly firing BOTS that aren't idle players: //if(IsFakeClient(victim) && !HasEntProp(attacker, Prop_Send, "m_humanSpectatorUserID") || GetEntProp(attacker, Prop_Send, "m_humanSpectatorUserID") == 0) return Plugin_Continue; // Stop all damage early if already marked as troll if(pData[attacker].isTroll) { SDKHooks_TakeDamage(attacker, attacker, attacker, pData[attacker].autoRFFScaleFactor * damage); return Plugin_Stop; } // Allow vanilla-damage if being attacked by special (example, charger carry) if(pData[victim].underAttack) return Plugin_Continue; // Is damage not caused by fire or pipebombs? bool isDamageDirect = damagetype & (DMG_BLAST|DMG_BURN|DMG_BLAST_SURFACE) == 0; int time = GetTime(); // If is a fall within first 2 minutes, do appropiate action if(!isAdmin && damagetype & DMG_FALL && attacker == victim && damage > 0.0 && time - pData[victim].joinTime <= hJoinTime.IntValue * 60) { pData[victim].jumpAttempts++; float pos[3]; GetNearestPlayerPosition(victim, pos); PrintToConsoleAdmins("%N within join time (%d min), attempted to fall"); if(pData[victim].jumpAttempts > hSuicideLimit.IntValue) { if(hSuicideAction.IntValue == 1) { LogMessage("[NOTICE] Kicking %N for suicide attempts", victim, hBanTime.IntValue); NotifyAllAdmins("[Notice] Kicking %N for suicide attempts", victim, hBanTime.IntValue); KickClient(victim, "Troll"); } else if(hSuicideAction.IntValue == 2) { LogMessage("[NOTICE] Banning %N for suicide attempts for %d minutes.", victim, hBanTime.IntValue); NotifyAllAdmins("[Notice] Banning %N for suicide attempts for %d minutes.", victim, hBanTime.IntValue); BanClient(victim, hBanTime.IntValue, BANFLAG_AUTO | BANFLAG_AUTHID, "Suicide fall attempts", "Troll", "TKStopper"); } else if(hSuicideAction.IntValue == 3) { LogMessage("[NOTICE] %N will be banned for suicide attempts for %d minutes. ", victim, hBanTime.IntValue); NotifyAllAdmins("[Notice] %N will be banned for suicide attempts for %d minutes. Use \"/ignore tk\" to make them immune.", victim, hBanTime.IntValue); pData[victim].isTroll = true; } } } if(attacker == victim) return Plugin_Continue; // Forgive player based on threshold, resetting accumlated damage if(time - pData[attacker].lastFFTime > hForgivenessTime.IntValue) { pData[attacker].TKDamageBuffer = 0.0; } pData[attacker].TKDamageBuffer += damage; pData[attacker].totalDamageFF += damage; pData[attacker].totalFFCount++; pData[attacker].ffCount++; // Auto reverse ff logic // If not immune to RFF, damage is direct, _or admin shit_ if(~pData[attacker].immunityFlags & Immune_RFF && isDamageDirect) { float minutesSinceiLastFFTime = (time - pData[attacker].lastFFTime) / 60.0; pData[attacker].autoRFFScaleFactor -= minutesSinceiLastFFTime * hFFAutoScaleForgivenessAmount.FloatValue; // Decrement any accumlated ff 'counts' int ffCountMinutes = RoundFloat(minutesSinceiLastFFTime / 10.0); pData[attacker].ffCount -= (ffCountMinutes * 2); if(pData[attacker].ffCount < 0) { pData[attacker].ffCount = 0; } // Decrement any forgiven ratio (computed on demand) if(pData[attacker].autoRFFScaleFactor < 0.0) { pData[attacker].autoRFFScaleFactor = 0.0; } // Then calculate a new reverse ff ratio pData[attacker].autoRFFScaleFactor += hFFAutoScaleAmount.FloatValue * damage * (pData[attacker].ffCount*0.25); if(pData[attacker].isTroll) { pData[attacker].autoRFFScaleFactor *= 2; } // Cap max damage only for non-trolls if(!pData[attacker].isTroll && hFFAutoScaleMaxRatio.FloatValue > 0.0 && pData[attacker].autoRFFScaleFactor > hFFAutoScaleMaxRatio.FloatValue) { pData[attacker].autoRFFScaleFactor = hFFAutoScaleMaxRatio.FloatValue; } } int prevFFTime = pData[attacker].lastFFTime; pData[attacker].lastFFTime = time; // Check for excessive friendly fire damage in short timespan // If not immune to TK, if over TK threshold, not when escaping, and direct damage if(~pData[attacker].immunityFlags & Immune_TK && pData[attacker].TKDamageBuffer > hThreshold.IntValue && !isFinaleEnding && isDamageDirect && !IsFakeClient(victim) // Don't increment on bot-ff for now ) { float diffJoinMin = (float(GetTime()) - float(pData[attacker].joinTime)) / 60.0; float lastFFMin = (float(GetTime()) - float(prevFFTime)) / 60.0; LogAction(-1, attacker, "Excessive FF (%.2f HP/%.2f total) (%.2f RFF Rate) (joined %.1fm ago) (%.1fmin last FF)", pData[attacker].TKDamageBuffer, pData[attacker].totalDamageFF, pData[attacker].autoRFFScaleFactor, diffJoinMin, lastFFMin ); if(!pData[attacker].pendingAction) { if(hTKAction.IntValue == 1) { LogMessage("[TKStopper] Kicking %N for excessive FF (%.2f HP)", attacker, pData[attacker].TKDamageBuffer); NotifyAllAdmins("[Notice] Kicking %N for excessive FF (%.2f HP)", attacker, pData[attacker].TKDamageBuffer); KickClient(attacker, "Excessive FF"); } else if(hTKAction.IntValue == 2) { LogMessage("[TKStopper] Banning %N for excessive FF (%.2f HP) for %d minutes.", attacker, pData[attacker].TKDamageBuffer, hBanTime.IntValue); NotifyAllAdmins("[Notice] Banning %N for excessive FF (%.2f HP) for %d minutes.", attacker, pData[attacker].TKDamageBuffer, hBanTime.IntValue); BanClient(attacker, hBanTime.IntValue, BANFLAG_AUTO | BANFLAG_AUTHID, "Excessive FF (Auto)", "Excessive Friendly Fire (Automatic)", "TKStopper"); } else if(hTKAction.IntValue == 3) { LogMessage("[TKStopper] %N will be banned for FF on disconnect (%.2f HP) for %d minutes. ", attacker, pData[attacker].TKDamageBuffer, hBanTime.IntValue); NotifyAllAdmins("[Notice] %N will be banned for FF on disconnect (%.2f HP) for %d minutes. Use \"/ignore tk\" to make them immune.", attacker, pData[attacker].TKDamageBuffer, hBanTime.IntValue); pData[attacker].isTroll = true; } pData[attacker].pendingAction = true; } damage = 0.0; return Plugin_Handled; } // Modify damages based on criteria if(pData[victim].jumpAttempts > 0 || L4D_IsInFirstCheckpoint(victim) || L4D_IsInLastCheckpoint(victim) || time - pData[attacker].joinTime <= hJoinTime.IntValue * 60 ) { /* If the amount of seconds since they joined is <= the minimum join time cvar (min) threshold or if the player is in a saferoom or if the player tried to suicide jump Then cancel all damage: */ damage = 0.0; return Plugin_Handled; }else if(isFinaleEnding) { // Keep admins immune if escape vehicle out, or if victim is a bot if(isAdmin || IsFakeClient(victim)) return Plugin_Continue; if(isDamageDirect) SDKHooks_TakeDamage(attacker, attacker, attacker, damage * 2.0); damage = 0.0; return Plugin_Changed; }else if(isDamageDirect && pData[attacker].autoRFFScaleFactor > 0.3) { // Ignore fire and propane damage, mistakes can happen // Apply their reverse ff damage, and have victim take a decreasing amount if(pData[attacker].isTroll) return Plugin_Stop; else if(pData[attacker].immunityFlags & Immune_RFF) return Plugin_Continue; SDKHooks_TakeDamage(attacker, attacker, attacker, pData[attacker].autoRFFScaleFactor * damage); if(pData[attacker].autoRFFScaleFactor > 1.0) damage /= pData[attacker].autoRFFScaleFactor; else damage /= 2.0; return Plugin_Changed; } } return Plugin_Continue; } /// COMMANDS public Action Command_TKInfo(int client, int args) { int time = GetTime(); if(!isEnabled) { ReplyToCommand(client, "Warn: Plugin is disabled in current gamemode (%s)", gamemode); } if(args > 0) { static char arg1[32]; GetCmdArg(1, arg1, sizeof(arg1)); static char target_name[MAX_TARGET_LENGTH]; int target_list[1], target_count; bool tn_is_ml; if ((target_count = ProcessTargetString( arg1, client, target_list, 1, COMMAND_FILTER_NO_MULTI, target_name, sizeof(target_name), tn_is_ml)) <= 0 || target_count == 0) { ReplyToTargetError(client, target_count); return Plugin_Handled; } int target = target_list[0]; ReplyToCommand(client, "FF Review for '%N':", target); if(pData[target].isTroll) { ReplyToCommand(client, "- will be banned on disconnect for TK -", target); } if(pData[target].immunityFlags == Immune_TK) { ReplyToCommand(client, "Immunity: Teamkiller Detection", target); } else if(pData[target].immunityFlags == Immune_RFF) { ReplyToCommand(client, "Immunity: Auto reverse-ff", target); } else if(view_as(pData[target].immunityFlags) > 0) { ReplyToCommand(client, "Immunity: Teamkiller Detection, Auto reverse-ff", target); } else { ReplyToCommand(client, "Immunity: (none, use /ignore [immunity] to toggle)", target); } float minutesSinceiLastFFTime = GetLastFFMinutes(target); float activeRate = GetActiveRate(target); ReplyToCommand(client, "FF Frequency: %d (active %d, %d forgotten)", pData[target].totalFFCount, pData[target].ffCount, (pData[target].totalFFCount - pData[target].ffCount)); ReplyToCommand(client, "Total FF Damage: %.1f HP (%.1f min ago last ff)", pData[target].totalDamageFF, minutesSinceiLastFFTime); if(~pData[target].immunityFlags & Immune_TK) ReplyToCommand(client, "Recent FF (TKDetectBuff): %.1f", pData[target].TKDamageBuffer); if(~pData[target].immunityFlags & Immune_RFF) ReplyToCommand(client, "Auto Reverse-FF: %.1fx return rate", activeRate); ReplyToCommand(client, "Attempted suicide jumps: %d", pData[target].jumpAttempts); } else { for(int i = 1; i <= MaxClients; i++) { if(IsClientConnected(i) && IsClientInGame(i) && !IsFakeClient(i)) { float minutesSinceiLastFFTime = (time - pData[i].lastFFTime) / 60.0; float activeRate = pData[i].autoRFFScaleFactor - (minutesSinceiLastFFTime * hFFAutoScaleForgivenessAmount.FloatValue); if(activeRate < 0.0) { activeRate = 0.0; } PrintToConsoleAll("%20N: %.1f TK-FF buf (%.2f total ff, %d freq., inter %d) | %.3f (buf %f) rFF rate | lastff %.1f min ago | %d suicide jumps", i, pData[i].TKDamageBuffer, pData[i].totalDamageFF, pData[i].totalFFCount, pData[i].ffCount, activeRate, pData[i].autoRFFScaleFactor, minutesSinceiLastFFTime, pData[i].jumpAttempts ); } } } return Plugin_Handled; } public Action Command_IgnorePlayer(int client, int args) { char arg1[32], arg2[16]; GetCmdArg(1, arg1, sizeof(arg1)); GetCmdArg(2, arg2, sizeof(arg2)); if(args < 2) { ReplyToCommand(client, "Usage: sm_ignore "); return Plugin_Handled; } int flags = 0; if(StrEqual(arg2, "tk") || StrEqual(arg2, "teamkill")) { flags = Immune_TK; } else if(StrEqual(arg2, "all") || StrEqual(arg2, "a")) { flags = Immune_TK | Immune_RFF; } else if(StrEqual(arg2, "reverseff") || StrEqual(arg2, "rff")) { flags = Immune_RFF; } else { ReplyToCommand(client, "Usage: sm_ignore "); return Plugin_Handled; } char target_name[MAX_TARGET_LENGTH]; int target_list[MAXPLAYERS+1], target_count; bool tn_is_ml; if ((target_count = ProcessTargetString( arg1, client, target_list, MaxClients, COMMAND_FILTER_ALIVE, target_name, sizeof(target_name), tn_is_ml)) <= 0 ) { ReplyToTargetError(client, target_count); return Plugin_Handled; } for (int i = 0; i < target_count; i++) { int target = target_list[i]; /*if (GetUserAdmin(target) != INVALID_ADMIN_ID) { ReplyToCommand(client, "%N is an admin and is already immune.", target); return Plugin_Handled; }*/ if (flags & Immune_TK) { if (pData[target].immunityFlags & Immune_TK) { LogAction(client, target, "\"%L\" re-enabled teamkiller detection for \"%L\"", client, target); ShowActivity2(client, "[FTT] ", "%N has re-enabled teamkiller detection for %N", client, target); } else { LogAction(client, target, "\"%L\" ignored teamkiller detection for \"%L\"", client, target); ShowActivity2(client, "[FTT] ", "%N has ignored teamkiller detection for %N", client, target); } pData[target].immunityFlags ^= Immune_TK; } if (flags & Immune_RFF) { if (pData[target].immunityFlags & Immune_RFF) { LogAction(client, target, "\"%L\" re-enabled auto reverse friendly-fire for \"%L\"", client, target); ShowActivity2(client, "[FTT] ", "%N has re-enabled auto reverse friendly-fire for %N", client, target); } else { LogAction(client, target, "\"%L\" disabled auto reverse friendly-fire for \"%L\"", client, target); ShowActivity2(client, "[FTT] ", "%N has disabled auto reverse friendly-fire for %N", client, target); pData[target].autoRFFScaleFactor = 0.0; } pData[target].immunityFlags ^= Immune_RFF; } pData[target].isTroll = false; } return Plugin_Handled; } /// STOCKS float GetLastFFMinutes(int client) { return (GetTime() - pData[client].lastFFTime) / 60.0; } float GetActiveRate(int client) { float activeRate = pData[client].autoRFFScaleFactor - (GetLastFFMinutes(client) * hFFAutoScaleForgivenessAmount.FloatValue); if(activeRate < 0.0) activeRate = 0.0; return activeRate; } stock bool GetNearestPlayerPosition(int client, float pos[3]) { static float targetPos[3], lowestDist; int lowestID = -1; GetClientAbsOrigin(client, targetPos); for(int i = 1; i <= MaxClients; i++) { if(IsClientConnected(i) && IsClientInGame(i) && IsPlayerAlive(i) && GetClientTeam(i) == 2 && i != client) { GetClientAbsOrigin(i, pos); float distance = GetVectorDistance(pos, targetPos); if(lowestID == -1 || distance < lowestDist) { lowestID = i; lowestDist = distance; } } } GetClientAbsOrigin(lowestID, pos); return lowestID > 0; } stock void PrintToConsoleAdmins(const char[] format, any ...) { char buffer[254]; VFormat(buffer, sizeof(buffer), format, 2); for(int i = 1; i <= MaxClients; i++) { if(IsClientConnected(i) && IsClientInGame(i)) { AdminId admin = GetUserAdmin(i); if(admin != INVALID_ADMIN_ID) { PrintToConsole(i, "%s", buffer); } } } PrintToServer("%s", buffer); }