332 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			SourcePawn
		
	
	
	
	
	
			
		
		
	
	
			332 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			SourcePawn
		
	
	
	
	
	
| #pragma semicolon 1
 | |
| #pragma newdecls required
 | |
| 
 | |
| #include <sourcemod>
 | |
| #include <sdkhooks>
 | |
| #include <sdktools>
 | |
| #include <dhooks>
 | |
| #include <cstrike>
 | |
| #include <zombiereloaded>
 | |
| 
 | |
| public Plugin myinfo =
 | |
| {
 | |
| 	name 			= "DamageProxy",
 | |
| 	author 			= "BotoX",
 | |
| 	description 	= "",
 | |
| 	version 		= "0.0",
 | |
| 	url 			= ""
 | |
| };
 | |
| 
 | |
| bool g_bLateLoad = false;
 | |
| 
 | |
| Handle g_hFireBulletDetour;
 | |
| int g_hVelocityModifier;
 | |
| int g_hActiveWeapon;
 | |
| 
 | |
| int g_LastAttacker = 0;
 | |
| int g_LastVictim = 0;
 | |
| 
 | |
| char g_iPhysboxToClient[2048];
 | |
| 
 | |
| int g_iSpecialKnife[MAXPLAYERS + 1];
 | |
| bool g_bNoSlowdown[MAXPLAYERS + 1];
 | |
| bool g_bRestoreHP[MAXPLAYERS + 1];
 | |
| bool g_bFullKnife[MAXPLAYERS + 1];
 | |
| 
 | |
| KeyValues g_Config;
 | |
| 
 | |
| public void OnPluginStart()
 | |
| {
 | |
| 	Handle hGameData = LoadGameConfigFile("DamageProxy.games");
 | |
| 	if(!hGameData)
 | |
| 		SetFailState("Failed to load DamageProxy gamedata.");
 | |
| 
 | |
| 	g_hFireBulletDetour = DHookCreateFromConf(hGameData, "CCSPlayer__FireBullet");
 | |
| 	if(!g_hFireBulletDetour)
 | |
| 		SetFailState("Failed to setup detour for CCSPlayer__FireBullet");
 | |
| 
 | |
| 	delete hGameData;
 | |
| 
 | |
| 	if(!DHookEnableDetour(g_hFireBulletDetour, false, Detour_OnFireBullet))
 | |
| 		SetFailState("Failed to detour CCSPlayer__FireBullet.");
 | |
| 
 | |
| 	g_hVelocityModifier = FindSendPropInfo("CCSPlayer", "m_flVelocityModifier");
 | |
| 	if(g_hVelocityModifier == -1)
 | |
| 		SetFailState("Couldn't find CCSPlayer::m_flVelocityModifier");
 | |
| 
 | |
| 	g_hActiveWeapon = FindSendPropInfo("CBaseCombatCharacter", "m_hActiveWeapon");
 | |
| 	if(g_hActiveWeapon == -1)
 | |
| 		SetFailState("Couldn't find CBaseCombatCharacter::m_hActiveWeapon");
 | |
| }
 | |
| 
 | |
| public APLRes AskPluginLoad2(Handle myself, bool late, char[] error, int err_max)
 | |
| {
 | |
| 	g_bLateLoad = late;
 | |
| 	return APLRes_Success;
 | |
| }
 | |
| 
 | |
| public void OnMapStart()
 | |
| {
 | |
| 	bool bLate = g_bLateLoad;
 | |
| 	g_bLateLoad = false;
 | |
| 
 | |
| 	if(g_Config)
 | |
| 		delete g_Config;
 | |
| 
 | |
| 	char sMapName[PLATFORM_MAX_PATH];
 | |
| 	GetCurrentMap(sMapName, sizeof(sMapName));
 | |
| 
 | |
| 	char sConfigFile[PLATFORM_MAX_PATH];
 | |
| 	BuildPath(Path_SM, sConfigFile, sizeof(sConfigFile), "configs/damageproxy/%s.cfg", sMapName);
 | |
| 	if(!FileExists(sConfigFile))
 | |
| 	{
 | |
| 		LogMessage("Could not find mapconfig: \"%s\"", sConfigFile);
 | |
| 		return;
 | |
| 	}
 | |
| 	LogMessage("Found mapconfig: \"%s\"", sConfigFile);
 | |
| 
 | |
| 	g_Config = new KeyValues("items");
 | |
| 	if(!g_Config.ImportFromFile(sConfigFile))
 | |
| 	{
 | |
| 		delete g_Config;
 | |
| 		LogError("ImportFromFile() failed!");
 | |
| 		return;
 | |
| 	}
 | |
| 	g_Config.Rewind();
 | |
| 
 | |
| 	if(!bLate)
 | |
| 		return;
 | |
| 
 | |
| 	/* Late Load */
 | |
| 	for(int client = 1; client <= MaxClients; client++)
 | |
| 	{
 | |
| 		if(!IsClientInGame(client))
 | |
| 			continue;
 | |
| 
 | |
| 		OnClientPutInServer(client);
 | |
| 
 | |
| 		int iKnife = GetPlayerWeaponSlot(client, CS_SLOT_KNIFE);
 | |
| 		if(iKnife > 0)
 | |
| 			OnWeaponEquipped(client, iKnife);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| public void OnClientPutInServer(int client)
 | |
| {
 | |
| 	g_iSpecialKnife[client] = 0;
 | |
| 	g_bNoSlowdown[client] = false;
 | |
| 	g_bRestoreHP[client] = false;
 | |
| 	g_bFullKnife[client] = false;
 | |
| 
 | |
| 	if(g_Config)
 | |
| 	{
 | |
| 		SDKHook(client, SDKHook_WeaponEquipPost, OnWeaponEquipped);
 | |
| 		SDKHook(client, SDKHook_WeaponDropPost, OnWeaponDropped);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| public void OnWeaponEquipped(int client, int entity)
 | |
| {
 | |
| 	if(!g_Config)
 | |
| 		return;
 | |
| 
 | |
| 	if(!IsValidEntity(entity))
 | |
| 		return;
 | |
| 
 | |
| 	char sClassname[64];
 | |
| 	GetEntityClassname(entity, sClassname, sizeof(sClassname));
 | |
| 	if(strcmp(sClassname, "weapon_knife", false) != 0)
 | |
| 		return;
 | |
| 
 | |
| 	int iHammerID = GetEntProp(entity, Prop_Data, "m_iHammerID");
 | |
| 	if(!iHammerID)
 | |
| 		return;
 | |
| 
 | |
| 	char sHammerID[16] = "z";
 | |
| 	if(ZR_IsClientHuman(client))
 | |
| 		sHammerID[0] = 'h';
 | |
| 	IntToString(iHammerID, sHammerID[1], sizeof(sHammerID) - 1);
 | |
| 
 | |
| 	g_Config.Rewind();
 | |
| 	if(!g_Config.JumpToKey(sHammerID))
 | |
| 		return;
 | |
| 
 | |
| 	float fKnockbackScale = g_Config.GetFloat("KnockbackScale", 1.0);
 | |
| 	ZR_SetClientKnockbackScale(client, fKnockbackScale);
 | |
| 
 | |
| 	float fKnockbackMaxForce = g_Config.GetFloat("KnockbackMaxForce", 0.0);
 | |
| 	if(fKnockbackMaxForce >= 0.0)
 | |
| 		ZR_SetClientKnockbackMaxForce(client, fKnockbackMaxForce);
 | |
| 
 | |
| 	float fKnockbackMaxVel = g_Config.GetFloat("KnockbackMaxVel", -1.0);
 | |
| 	if(fKnockbackMaxVel >= 0.0)
 | |
| 		ZR_SetClientKnockbackMaxVelocity(client, fKnockbackMaxVel);
 | |
| 
 | |
| 	g_bNoSlowdown[client] = view_as<bool>(g_Config.GetNum("NoSlowDown", 0));
 | |
| 
 | |
| 	g_bRestoreHP[client] = view_as<bool>(g_Config.GetNum("RestoreHP", 1));
 | |
| 
 | |
| 	g_bFullKnife[client] = view_as<bool>(g_Config.GetNum("FullKnife", 0));
 | |
| 
 | |
| 	g_iSpecialKnife[client] = entity;
 | |
| 
 | |
| 	CheckChildren(client, entity);
 | |
| }
 | |
| 
 | |
| public void OnWeaponDropped(int client, int entity)
 | |
| {
 | |
| 	if(entity == g_iSpecialKnife[client])
 | |
| 	{
 | |
| 		ZR_SetClientKnockbackScale(client, 1.0);
 | |
| 		ZR_SetClientKnockbackMaxForce(client, 0.0);
 | |
| 		ZR_SetClientKnockbackMaxVelocity(client, -1.0);
 | |
| 		g_iSpecialKnife[client] = 0;
 | |
| 		g_bNoSlowdown[client] = false;
 | |
| 		g_bRestoreHP[client] = false;
 | |
| 		g_bFullKnife[client] = false;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| public void OnEntityDestroyed(int entity)
 | |
| {
 | |
| 	if(entity >= 0 && entity < sizeof(g_iPhysboxToClient))
 | |
| 		g_iPhysboxToClient[entity] = 0;
 | |
| }
 | |
| 
 | |
| void CheckChildren(int client, int parent)
 | |
| {
 | |
| 	bool bKillPush = view_as<bool>(g_Config.GetNum("KillPush", 1));
 | |
| 
 | |
| 	int entity = INVALID_ENT_REFERENCE;
 | |
| 	while((entity = FindEntityByClassname(entity, "*")) != INVALID_ENT_REFERENCE)
 | |
| 	{
 | |
| 		if(GetEntPropEnt(entity, Prop_Data, "m_pParent") != parent)
 | |
| 			continue;
 | |
| 
 | |
| 		char sClassname[64];
 | |
| 		GetEntityClassname(entity, sClassname, sizeof(sClassname));
 | |
| 
 | |
| 		char sTargetname[64];
 | |
| 		GetEntPropString(entity, Prop_Data, "m_iName", sTargetname, sizeof(sTargetname));
 | |
| 
 | |
| 		int iTemplateLoc = FindCharInString(sTargetname, '&', true);
 | |
| 		if(iTemplateLoc != -1)
 | |
| 			sTargetname[iTemplateLoc] = 0;
 | |
| 
 | |
| 		// -1 = Never kill
 | |
| 		// 0 = Default
 | |
| 		// 1 = Always kill
 | |
| 		int iKill = 0;
 | |
| 		if(sTargetname[0])
 | |
| 			iKill = g_Config.GetNum(sTargetname, 0);
 | |
| 
 | |
| 		PrintToConsoleAll("%d child: \"%s\" %d (%s) kill:%d", client, sClassname, entity, sTargetname, iKill);
 | |
| 
 | |
| 		if(!strncmp(sClassname, "func_physbox", 12, false) || !strcmp(sClassname, "func_breakable", false))
 | |
| 		{
 | |
| 			g_iPhysboxToClient[entity] = client;
 | |
| 			SDKHook(entity, SDKHook_OnTakeDamage, OnTakeDamage);
 | |
| 
 | |
| 			PrintToConsoleAll("Hooking \"%s\" %d (%s) OnTakeDamage", sClassname, entity, sTargetname);
 | |
| 		}
 | |
| 		else if((bKillPush && !strcmp(sClassname, "trigger_push", false)) || iKill > 0)
 | |
| 		{
 | |
| 			if((bKillPush || iKill > 0) && iKill >= 0)
 | |
| 			{
 | |
| 				AcceptEntityInput(entity, "Kill");
 | |
| 				PrintToConsoleAll("Killing \"%s\" %d (%s)", sClassname, entity, sTargetname);
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			CheckChildren(client, entity);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| public Action OnPlayerRunCmd(int client, int &buttons, int &impulse, float vel[3], float angles[3], int &weapon, int &subtype, int &cmdnum, int &tickcount, int &seed, int mouse[2])
 | |
| {
 | |
| 	// because knifes don't call FireBullet we reset it here too
 | |
| 	g_LastAttacker = 0;
 | |
| 	g_LastVictim = 0;
 | |
| }
 | |
| 
 | |
| public MRESReturn Detour_OnFireBullet(int pThis, Handle hReturn, Handle hParams)
 | |
| {
 | |
| 	// need to reset it per fired bullet else only one of the shotgun pellets would be able to hit the same player
 | |
| 	g_LastAttacker = 0;
 | |
| 	g_LastVictim = 0;
 | |
| 	return MRES_Handled;
 | |
| }
 | |
| 
 | |
| public Action OnTakeDamage(int victim, int &attacker, int &inflictor, float &damage, int &damagetype, int &weapon, float damageForce[3], float damagePosition[3], int damagecustom)
 | |
| {
 | |
| 	if(attacker <= 0 || attacker > MAXPLAYERS)
 | |
| 		return Plugin_Continue;
 | |
| 
 | |
| 	int client = g_iPhysboxToClient[victim];
 | |
| 	if(client <= 0)
 | |
| 	{
 | |
| 		PrintToConsoleAll("!!! %d client = %d no physbox !!!", victim, client);
 | |
| 		return Plugin_Continue;
 | |
| 	}
 | |
| 
 | |
| 	int knife = g_iSpecialKnife[client];
 | |
| 	if(knife <= 0)
 | |
| 	{
 | |
| 		PrintToConsoleAll("!!! %d client = %d no knife !!!", victim, client);
 | |
| 		return Plugin_Continue;
 | |
| 	}
 | |
| 
 | |
| 	// fix multiple penetration
 | |
| 	if(g_LastAttacker == attacker && g_LastVictim == client)
 | |
| 		return Plugin_Handled;
 | |
| 
 | |
| 	g_LastAttacker = attacker;
 | |
| 	g_LastVictim = client;
 | |
| 
 | |
| 	int flags = ZR_KNOCKBACK_CUSTOM | ZR_KNOCKBACK_SCALE | ZR_KNOCKBACK_LIMITFORCE | ZR_KNOCKBACK_LIMITVEL;
 | |
| 
 | |
| 	if(g_bFullKnife[client])
 | |
| 	{
 | |
| 		char sWeaponClassname[64];
 | |
| 		if(inflictor == attacker)
 | |
| 		{
 | |
| 			int iWeapon = GetClientActiveWeapon(attacker);
 | |
| 			if(iWeapon > 0)
 | |
| 				GetEdictClassname(iWeapon, sWeaponClassname, sizeof(sWeaponClassname));
 | |
| 		}
 | |
| 		else
 | |
| 			GetEdictClassname(inflictor, sWeaponClassname, sizeof(sWeaponClassname));
 | |
| 
 | |
| 		if(StrEqual(sWeaponClassname, "weapon_knife"))
 | |
| 			flags &= ~(ZR_KNOCKBACK_SCALE | ZR_KNOCKBACK_LIMITFORCE | ZR_KNOCKBACK_LIMITVEL);
 | |
| 	}
 | |
| 
 | |
| 	float flVelocityModifier;
 | |
| 	if(g_bNoSlowdown[client])
 | |
| 		flVelocityModifier = GetEntDataFloat(client, g_hVelocityModifier);
 | |
| 
 | |
| 	int iClientHealth;
 | |
| 	if(g_bRestoreHP[client])
 | |
| 		iClientHealth = GetClientHealth(client);
 | |
| 
 | |
| 	// Don't damage kevlar.
 | |
| 	damagetype |= DMG_RADIATION;
 | |
| 
 | |
| 	SDKHooks_TakeDamage(client, inflictor, attacker, damage, damagetype, weapon, damageForce, damagePosition, flags);
 | |
| 
 | |
| 	if(g_bNoSlowdown[client])
 | |
| 		SetEntDataFloat(client, g_hVelocityModifier, flVelocityModifier, true);
 | |
| 
 | |
| 	if(g_bRestoreHP[client])
 | |
| 		SetEntityHealth(client, iClientHealth);
 | |
| 
 | |
| 	return Plugin_Continue;
 | |
| }
 | |
| 
 | |
| stock int GetClientActiveWeapon(int client)
 | |
| {
 | |
| 	return GetEntDataEnt2(client, g_hActiveWeapon);
 | |
| }
 |