dobrograd-13-06-2022/garrysmod/addons/feature-damage/lua/autorun/server/enhanceddmg.lua

328 lines
12 KiB
Lua
Raw Normal View History

2023-11-16 15:01:19 +05:00
--[[
Code is a mess, gotta fix
Todo: Make the sounds a single function, decrapify the model check. Also stop having the same piece of code multiple times, thats a bad practice
]]--
AddCSLuaFile()
hook.Add('Initialize', 'dbg.dmg', function()
timer.Create('damage.drown', 1, 0, function()
octolib.func.throttle(player.GetAll(), 10, 0.1, function(ply)
if not IsValid(ply) then return end
if ply:WaterLevel() == 3 then
local curScore = ply.drowningScore or 0
if curScore >= 10 then
local dmginfo = DamageInfo()
dmginfo:SetDamage(10)
dmginfo:SetDamageType(DMG_DROWN)
dmginfo:SetAttacker(game.GetWorld())
dmginfo:SetInflictor(game.GetWorld())
ply:TakeDamageInfo(dmginfo)
else
ply.drowningScore = curScore + 1
end
else
ply.drowningScore = nil
end
end)
end)
--This is terrible but whatevs
local function BreakLeg(ply,duration)
if !GetConVar('enhanceddamage_legbreak'):GetBool() then print('TEST') return end
if !ply.legshot then
ply.legshot = true
ply:MoveModifier('dmg', {
walkmul = 0.5,
norun = true,
nojump = true,
})
timer.Create('breakLeg_' .. ply:SteamID(), duration, 1, function() ply:MoveModifier('dmg', nil) end)
end
end
local function FallDamage(ply,speed)
if ply:IsGhost() or ply:Team() == TEAM_ADMIN then return 0 end
local damage = speed / 7.5
if (damage > ply:Health() / 2 and damage < ply:Health()) then
BreakLeg(ply,10)
end
ply.lastDMGT = DMG_FALL
return damage
end
local GM = GAMEMODE or GM
function DarkRP.damageHands(ply, chance)
if not ply:IsPlayer() or ply:Team() == TEAM_ADMIN then return false end
if math.random(100) > chance then return end
local weapon = ply:GetActiveWeapon()
if not IsValid(weapon) then return end
if not GM.Config.DisallowDrop[weapon:GetClass()] then
if not ply:jobHasWeapon(weapon:GetClass()) then
if not weapon.NoHandDamageDrop then
local ent = ply:dropDRPWeapon(weapon)
if IsValid(ent) and weapon.IsLethal then
ent.isEvidence = true
end
end
else
ply:SelectWeapon('dbg_hands')
end
end
ply.noPickups = true
timer.Create('resetNoPickups' .. ply:SteamID(), 30, 1, function() if IsValid(ply) then ply.noPickups = nil end end)
end
hook.Add('octoinv.canPickup', 'dbg-damage', function(ply, ent, item)
if ply.noPickups then return false, L.hurts_hand end
end)
hook.Add('octoinv.canUse', 'dbg-damage', function(cont, item, ply)
if ply.noPickups then return false, L.hurts_hand end
end)
hook.Add('PlayerSwitchWeapon', 'dbg-damage', function(ply)
if ply.noPickups then return true, L.hurts_hand end
end)
local hitgroupNames = {
['HITGROUP_HAND'] = 'руку',
[HITGROUP_HEAD] = 'голову',
['HITGROUP_NUTS'] = 'голову',
[HITGROUP_LEFTLEG] = 'левую ногу',
[HITGROUP_RIGHTLEG] = 'правую ногу',
[HITGROUP_LEFTARM] = 'левую руку',
[HITGROUP_RIGHTARM] = 'правую руку',
[HITGROUP_STOMACH] = 'область живота',
[HITGROUP_CHEST] = 'область груди',
}
local function notifyDamage(ply, hitgroup)
local hitgroupName = hitgroupNames[hitgroup]
if hitgroupName then
ply:Notify('hint', 'Тебе попали в ' .. hitgroupName)
end
end
local function Damage(ply, hitgroup, dmginfo)
local dmgpos = dmginfo:GetDamagePosition()
local PelvisIndx = ply:LookupBone('ValveBiped.Bip01_Pelvis')
if (PelvisIndx == nil) then return dmginfo end --Maybe Hitgroup still works, need testing
local PelvisPos = ply:GetBonePosition( PelvisIndx )
local NutsDistance = dmgpos:DistToSqr(PelvisPos)
local LHandIndex = ply:LookupBone('ValveBiped.Bip01_L_Hand')
local LHandPos = ply:GetBonePosition( LHandIndex )
local LHandDistance = dmgpos:DistToSqr(LHandPos)
local RHandIndex = ply:LookupBone('ValveBiped.Bip01_R_Hand')
local RHandPos = ply:GetBonePosition(RHandIndex)
local RHandDistance = dmgpos:DistToSqr(RHandPos)
local LHandIndex = ply:LookupBone('ValveBiped.Bip01_L_Hand')
local LHandPos = ply:GetBonePosition( LHandIndex )
local LHandDistance = dmgpos:DistToSqr(LHandPos)
local RCalfIndex = ply:LookupBone('ValveBiped.Bip01_R_Calf')
local RCalfPos = ply:GetBonePosition(RCalfIndex)
local RCalfDistance = dmgpos:DistToSqr(RCalfPos)
local LCalfIndex = ply:LookupBone('ValveBiped.Bip01_L_Calf')
local LCalfPos = ply:GetBonePosition(LCalfIndex)
local LCalfDistance = dmgpos:DistToSqr(LCalfPos)
local HeadIndex = ply:LookupBone('ValveBiped.Bip01_Head1')
local HeadPos = ply:GetBonePosition(HeadIndex) + Vector(0,0,3)
local HeadDistance = dmgpos:DistToSqr(HeadPos)
if (LHandDistance < 100 || RHandDistance < 100 ) then
hitgroup = 'HITGROUP_HAND'
elseif HeadDistance < 80 then
hitgroup = HITGROUP_HEAD
elseif (NutsDistance <= 49 && NutsDistance >= 25) then
hitgroup = 'HITGROUP_NUTS'
elseif LCalfDistance < 350 then
hitgroup = HITGROUP_LEFTLEG
elseif RCalfDistance < 350 then
hitgroup = HITGROUP_RIGHTLEG
end
if (hitgroup == HITGROUP_HEAD) then
dmginfo:ScaleDamage(10)
elseif (hitgroup == HITGROUP_LEFTARM || hitgroup == HITGROUP_RIGHTARM) then
dmginfo:ScaleDamage(1)
DarkRP.damageHands(ply, 50)
elseif (hitgroup == HITGROUP_LEFTLEG || hitgroup == HITGROUP_RIGHTLEG) then
dmginfo:ScaleDamage(0.75)
if ply:IsPlayer() then BreakLeg(ply,5) end
elseif (hitgroup == HITGROUP_CHEST) then
dmginfo:ScaleDamage(3)
elseif (hitgroup == HITGROUP_STOMACH) then
dmginfo:ScaleDamage(1)
elseif (hitgroup == 'HITGROUP_NUTS') then
dmginfo:ScaleDamage(1.5)
if ply:IsPlayer() then BreakLeg(ply,5) end
elseif (hitgroup == 'HITGROUP_HAND') then
dmginfo:ScaleDamage(0.45)
DarkRP.damageHands(ply, 75)
end
notifyDamage(ply, hitgroup)
end
hook.Add('ScalePlayerDamage','EnhancedPlayerDamage',Damage)
hook.Add('GetFallDamage','EnhancedFallDamage',FallDamage)
local bleeding = {}
local allowHolster = {
weapon_flashlight = true,
gmod_camera = true,
}
timer.Create('dbg-damage.dying', 1, 0, function()
for k = #bleeding, 1, -1 do
local sid = bleeding[k]
local ply = player.GetBySteamID(sid)
if not IsValid(ply) then
timer.Remove('dbg-damage.dying' .. sid)
table.remove(bleeding, k)
elseif ply:Health() > 10 or not ply:Alive() or ply:IsGhost() then
ply:MoveModifier('bleeding', nil)
ply.bleeding = nil
timer.Remove('dbg-damage.dying' .. sid)
table.remove(bleeding, k)
end
end
end)
local function dying(ply, dmgInfo)
if not IsValid(ply) or not ply:IsPlayer() or ply:IsGhost() then return end
if ply:Team() == TEAM_ADMIN then return end
if ply.bleeding then return end
local left = ply:Health() - dmgInfo:GetDamage()
if left <= 0 then return end
if left <= 10 then
local w = ply:GetActiveWeapon()
if IsValid(w) and ply:HasWeapon(w:GetClass()) and not allowHolster[w:GetClass()] and hook.Call('canDropWeapon', GM, ply, w) then
ply:dropDRPWeapon(w)
end
ply:Notify('warning', 'Ты при смерти. Если тебе не окажут помощь, ты погибнешь')
ply.bleeding = true
bleeding[#bleeding + 1] = ply:SteamID()
ply:MoveModifier('bleeding', {
walkmul = 0.5,
norun = true,
nojump = true,
nostand = true,
})
timer.Create('dbg-damage.dying' .. ply:SteamID(), 18, 0, function()
if not ply:IsMale() then
ply:EmitSound(Sound('vo/npc/female01/moan0' .. math.random(1,5) .. '.wav'))
else
ply:EmitSound(Sound('vo/npc/male01/moan0' .. math.random(1,5) .. '.wav'))
end
if ply:Health() <= 1 then
local dmg = DamageInfo()
dmg:SetDamage(1)
ply.attackedBy = ply.lastAttacker
if ply.lastDMGT then
dmg:SetDamageType(ply.lastDMGT)
end
ply.weaponUsed = ply.lastWeapon
ply:TakeDamageInfo(dmg)
else ply:SetHealth(ply:Health() - 1) end
end)
end
end
local function cant(ply)
if ply.bleeding then return false, 'Ты при смерти' end
end
hook.Add('EntityTakeDamage', 'dbg-damage.dying', dying)
hook.Add('CanPlayerEnterVehicle', 'dbg-damage.dying', cant)
hook.Add('octoinv.canPickup', 'dbg-damage.dying', cant)
hook.Add('octoinv.canUse', 'dbg-damage.dying', cant)
hook.Add('dbg-hands.canPunch', 'dbg-damage.dying', cant)
hook.Add('dbg-hands.canCloseLockable', 'dbg-damage.dying', cant)
hook.Add('dbg-hands.canOpenLockable', 'dbg-damage.dying', cant)
hook.Add('dbg-hands.canDrag', 'dbg-damage.dying', cant)
hook.Add('PlayerDisconnected', 'dbg-damage.dying', function(ply)
if ply.bleeding then
local dmg = DamageInfo()
dmg:SetDamage(ply:GetMaxHealth())
ply.attackedBy = ply.lastAttacker
if ply.lastDMGT then
dmg:SetDamageType(ply.lastDMGT)
end
ply.weaponUsed = ply.lastWeapon
ply:TakeDamageInfo(dmg)
local sid = ply:SteamID()
timer.Remove('dbg-damage.dying' .. sid)
table.RemoveByValue(bleeding, sid)
octodeath.triggerDeath(ply)
end
end)
end)
netstream.Hook('dbg-armor.unwear', function(ply)
if not ply:Alive() then return end
local data = ply.armorItem
if not data then
ply:Notify('warning', 'У тебя нет надетого бронежилета')
return
end
if data.armor ~= ply:Armor() then
ply:Notify('warning', 'Твой бронежилет поврежден')
return
end
local inv = ply:GetInventory()
local cont = inv and inv:GetContainer('_hand')
if not cont then
ply:Notify('warning', 'Освободи руки, чтобы туда можно было положить бронежилет')
return
end
if cont:AddItem('armor', data) >= 1 then
ply:SetArmor(0)
ply.armorItem = nil
ply:SetLocalVar('armor', nil)
ply:EmitSound('npc/combine_soldier/gear3.wav', 55)
else
ply:Notify('warning', 'В руках недостаточно места')
end
end)
CreateConVar('enhanceddamage_enabled', 1, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable enhanced damage')
CreateConVar('enhanceddamage_headdamagescale', 2, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_armdamagescale', 0.50, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_legdamagescale',0.50, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_chestdamagescale', 1.25, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_stomachdamagescale',0.75, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_nutsdamagescale', 2, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_handdamagescale', 0.25, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_armdropchance',20, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'The weapon drop chance for ')
CreateConVar('enhanceddamage_handdropchance', 40, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Change the scale for this bodypart')
CreateConVar('enhanceddamage_enablesounds', 1, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable the sounds when hurt ')
CreateConVar('enhanceddamage_legbreak', 1, {FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable enhanced damage')
CreateConVar('enhanceddamage_npcweapondrop',1,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable weapon dropping for npcs (Really buggy)')
CreateConVar('enhanceddamage_falldamage',1,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable enhanced falldamage (Much more "realistic" and breaks your bones)')
CreateConVar('enhanceddamage_npcfalldamage',1,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable falldamage for NPC')
CreateConVar('enhanceddamage_drowningdamage',1,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Toggle drowning')
CreateConVar('enhanceddamage_ragdolls',0,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Enable enhanced ragdolls.')
CreateConVar('enhanceddamage_autoremoveragdolls',20,{FCVAR_SERVER_CAN_EXECUTE,FCVAR_NOTIFY,FCVAR_ARCHIVE},'Time before the ragdolls are remove (0 for never)')