zombiesurvival-evolved/gamemodes/zombiesurvival/entities/weapons/weapon_zs_base/animations.lua

520 lines
13 KiB
Lua
Raw Permalink Normal View History

2014-10-02 08:49:54 +08:00
--[[*******************************************************
SWEP Construction Kit base code
Created by Clavus
Available for public use, thread at:
facepunch.com/threads/1032378
2014-10-02 08:49:54 +08:00
DESCRIPTION:
This script is meant for experienced scripters
that KNOW WHAT THEY ARE DOING. Don't come to me
2014-10-02 08:49:54 +08:00
with basic Lua questions.
2014-10-02 08:49:54 +08:00
Just copy into your SWEP or SWEP base of choice
and merge with your own code.
2014-10-02 08:49:54 +08:00
The SWEP.VElements, SWEP.WElements and
SWEP.ViewModelBoneMods tables are all optional
and only have to be visible to the client.
********************************************************]]
function SWEP:Anim_Initialize()
-- other initialize code goes here
2014-10-02 08:49:54 +08:00
if CLIENT then
-- Create a new table for every weapon instance
2014-10-02 08:49:54 +08:00
self.VElements = table.FullCopy( self.VElements )
self.WElements = table.FullCopy( self.WElements )
self.ViewModelBoneMods = table.FullCopy( self.ViewModelBoneMods )
self:CreateModels(self.VElements) -- create viewmodels
self:CreateModels(self.WElements) -- create worldmodels
-- init view model bone build function
if IsValid(self:GetOwner()) then
local vm = self:GetOwner():GetViewModel()
2014-10-02 08:49:54 +08:00
if IsValid(vm) then
self:ResetBonePositions(vm)
-- Init viewmodel visibility
2014-10-02 08:49:54 +08:00
--[[if (self.ShowViewModel == nil or self.ShowViewModel) then
vm:SetColor(Color(255,255,255,255))
else
-- we set the alpha to 1 instead of 0 because else ViewModelDrawn stops being called
2014-10-02 08:49:54 +08:00
vm:SetColor(Color(255,255,255,1))
-- ^ stopped working in GMod 13 because you have to do Entity:SetRenderMode(1) for translucency to kick in
-- however for some reason the view model resets to render mode 0 every frame so we just apply a debug material to prevent it from drawing
vm:SetMaterial("Debug/hsv")
2014-10-02 08:49:54 +08:00
end]]
end
end
2014-10-02 08:49:54 +08:00
end
end
function SWEP:Anim_Holster()
if CLIENT and IsValid(self:GetOwner()) then
local vm = self:GetOwner():GetViewModel()
2014-10-02 08:49:54 +08:00
if IsValid(vm) then
self:ResetBonePositions(vm)
end
end
2014-10-02 08:49:54 +08:00
return true
end
function SWEP:Anim_OnRemove()
self:Anim_Holster()
end
if CLIENT then
SWEP.vRenderOrder = nil
function SWEP:Anim_ViewModelDrawn()
local vm = self:GetOwner():GetViewModel()
2014-10-02 08:49:54 +08:00
if !IsValid(vm) then return end
2014-10-02 08:49:54 +08:00
self:UpdateBonePositions(vm)
if (!self.VElements) then return end
2014-10-02 08:49:54 +08:00
if (!self.vRenderOrder) then
-- we build a render order because sprites need to be drawn after models
2014-10-02 08:49:54 +08:00
self.vRenderOrder = {}
for k, v in pairs( self.VElements ) do
if (v.type == "Model") then
table.insert(self.vRenderOrder, 1, k)
elseif (v.type == "Sprite" or v.type == "Quad") then
table.insert(self.vRenderOrder, k)
end
end
2014-10-02 08:49:54 +08:00
end
for k, name in ipairs( self.vRenderOrder ) do
2014-10-02 08:49:54 +08:00
local v = self.VElements[name]
if (!v) then self.vRenderOrder = nil break end
if (v.hide) then continue end
2014-10-02 08:49:54 +08:00
local model = v.modelEnt
local sprite = v.spriteMaterial
2014-10-02 08:49:54 +08:00
if (!v.bone) then continue end
2014-10-02 08:49:54 +08:00
local pos, ang = self:GetBoneOrientation( self.VElements, v, vm )
2014-10-02 08:49:54 +08:00
if (!pos) then continue end
2014-10-02 08:49:54 +08:00
if (v.type == "Model" and IsValid(model)) then
model:SetPos(pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z )
ang:RotateAroundAxis(ang:Up(), v.angle.y)
ang:RotateAroundAxis(ang:Right(), v.angle.p)
ang:RotateAroundAxis(ang:Forward(), v.angle.r)
model:SetAngles(ang)
--model:SetModelScale(v.size)
2014-10-02 08:49:54 +08:00
local matrix = Matrix()
matrix:Scale(v.size)
model:EnableMatrix( "RenderMultiply", matrix )
2014-10-02 08:49:54 +08:00
if (v.material == "") then
model:SetMaterial("")
elseif (model:GetMaterial() != v.material) then
model:SetMaterial( v.material )
end
2014-10-02 08:49:54 +08:00
if (v.skin and v.skin != model:GetSkin()) then
model:SetSkin(v.skin)
end
2014-10-02 08:49:54 +08:00
if (v.bodygroup) then
for k, v in ipairs( v.bodygroup ) do
2014-10-02 08:49:54 +08:00
if (model:GetBodygroup(k) != v) then
model:SetBodygroup(k, v)
end
end
end
2014-10-02 08:49:54 +08:00
if (v.surpresslightning) then
render.SuppressEngineLighting(true)
end
2014-10-02 08:49:54 +08:00
render.SetColorModulation(v.color.r/255, v.color.g/255, v.color.b/255)
render.SetBlend(v.color.a/255)
model:DrawModel()
render.SetBlend(1)
render.SetColorModulation(1, 1, 1)
2014-10-02 08:49:54 +08:00
if (v.surpresslightning) then
render.SuppressEngineLighting(false)
end
2014-10-02 08:49:54 +08:00
elseif (v.type == "Sprite" and sprite) then
2014-10-02 08:49:54 +08:00
local drawpos = pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z
render.SetMaterial(sprite)
render.DrawSprite(drawpos, v.size.x, v.size.y, v.color)
2014-10-02 08:49:54 +08:00
elseif (v.type == "Quad" and v.draw_func) then
2014-10-02 08:49:54 +08:00
local drawpos = pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z
ang:RotateAroundAxis(ang:Up(), v.angle.y)
ang:RotateAroundAxis(ang:Right(), v.angle.p)
ang:RotateAroundAxis(ang:Forward(), v.angle.r)
2014-10-02 08:49:54 +08:00
cam.Start3D2D(drawpos, ang, v.size)
v.draw_func( self )
cam.End3D2D()
end
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
end
SWEP.wRenderOrder = nil
function SWEP:Anim_DrawWorldModel()
2014-10-02 08:49:54 +08:00
if (self.ShowWorldModel == nil or self.ShowWorldModel) then
self:DrawModel()
end
2014-10-02 08:49:54 +08:00
if (!self.WElements) then return end
2014-10-02 08:49:54 +08:00
if (!self.wRenderOrder) then
self.wRenderOrder = {}
for k, v in pairs( self.WElements ) do
if (v.type == "Model") then
table.insert(self.wRenderOrder, 1, k)
elseif (v.type == "Sprite" or v.type == "Quad") then
table.insert(self.wRenderOrder, k)
end
end
end
if (IsValid(self:GetOwner())) then
bone_ent = self:GetOwner()
2014-10-02 08:49:54 +08:00
else
-- when the weapon is dropped
2014-10-02 08:49:54 +08:00
bone_ent = self
end
2014-10-02 08:49:54 +08:00
for k, name in pairs( self.wRenderOrder ) do
2014-10-02 08:49:54 +08:00
local v = self.WElements[name]
if (!v) then self.wRenderOrder = nil break end
if (v.hide) then continue end
2014-10-02 08:49:54 +08:00
local pos, ang
2014-10-02 08:49:54 +08:00
if (v.bone) then
pos, ang = self:GetBoneOrientation( self.WElements, v, bone_ent )
else
pos, ang = self:GetBoneOrientation( self.WElements, v, bone_ent, "ValveBiped.Bip01_R_Hand" )
end
2014-10-02 08:49:54 +08:00
if (!pos) then continue end
2014-10-02 08:49:54 +08:00
local model = v.modelEnt
local sprite = v.spriteMaterial
2014-10-02 08:49:54 +08:00
if (v.type == "Model" and IsValid(model)) then
model:SetPos(pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z )
ang:RotateAroundAxis(ang:Up(), v.angle.y)
ang:RotateAroundAxis(ang:Right(), v.angle.p)
ang:RotateAroundAxis(ang:Forward(), v.angle.r)
model:SetAngles(ang)
--model:SetModelScale(v.size)
2014-10-02 08:49:54 +08:00
local matrix = Matrix()
matrix:Scale(v.size)
model:EnableMatrix( "RenderMultiply", matrix )
2014-10-02 08:49:54 +08:00
if (v.material == "") then
model:SetMaterial("")
elseif (model:GetMaterial() != v.material) then
model:SetMaterial( v.material )
end
2014-10-02 08:49:54 +08:00
if (v.skin and v.skin != model:GetSkin()) then
model:SetSkin(v.skin)
end
2014-10-02 08:49:54 +08:00
if (v.bodygroup) then
for k, v in ipairs( v.bodygroup ) do
2014-10-02 08:49:54 +08:00
if (model:GetBodygroup(k) != v) then
model:SetBodygroup(k, v)
end
end
end
2014-10-02 08:49:54 +08:00
if (v.surpresslightning) then
render.SuppressEngineLighting(true)
end
2014-10-02 08:49:54 +08:00
render.SetColorModulation(v.color.r/255, v.color.g/255, v.color.b/255)
render.SetBlend(v.color.a/255)
model:DrawModel()
render.SetBlend(1)
render.SetColorModulation(1, 1, 1)
2014-10-02 08:49:54 +08:00
if (v.surpresslightning) then
render.SuppressEngineLighting(false)
end
2014-10-02 08:49:54 +08:00
elseif (v.type == "Sprite" and sprite) then
2014-10-02 08:49:54 +08:00
local drawpos = pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z
render.SetMaterial(sprite)
render.DrawSprite(drawpos, v.size.x, v.size.y, v.color)
2014-10-02 08:49:54 +08:00
elseif (v.type == "Quad" and v.draw_func) then
2014-10-02 08:49:54 +08:00
local drawpos = pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z
ang:RotateAroundAxis(ang:Up(), v.angle.y)
ang:RotateAroundAxis(ang:Right(), v.angle.p)
ang:RotateAroundAxis(ang:Forward(), v.angle.r)
2014-10-02 08:49:54 +08:00
cam.Start3D2D(drawpos, ang, v.size)
v.draw_func( self )
cam.End3D2D()
end
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
end
function SWEP:GetBoneOrientation( basetab, tab, ent, bone_override )
2014-10-02 08:49:54 +08:00
local bone, pos, ang
if (tab.rel and tab.rel != "") then
2014-10-02 08:49:54 +08:00
local v = basetab[tab.rel]
2014-10-02 08:49:54 +08:00
if (!v) then return end
-- Technically, if there exists an element with the same name as a bone
-- you can get in an infinite loop. Let's just hope nobody's that stupid.
2014-10-02 08:49:54 +08:00
pos, ang = self:GetBoneOrientation( basetab, v, ent )
2014-10-02 08:49:54 +08:00
if (!pos) then return end
2014-10-02 08:49:54 +08:00
pos = pos + ang:Forward() * v.pos.x + ang:Right() * v.pos.y + ang:Up() * v.pos.z
ang:RotateAroundAxis(ang:Up(), v.angle.y)
ang:RotateAroundAxis(ang:Right(), v.angle.p)
ang:RotateAroundAxis(ang:Forward(), v.angle.r)
2014-10-02 08:49:54 +08:00
else
2014-10-02 08:49:54 +08:00
bone = ent:LookupBone(bone_override or tab.bone)
if (!bone) then return end
2014-10-02 08:49:54 +08:00
pos, ang = Vector(0,0,0), Angle(0,0,0)
local m = ent:GetBoneMatrix(bone)
if (m) then
pos, ang = m:GetTranslation(), m:GetAngles()
end
if (IsValid(self:GetOwner()) and self:GetOwner():IsPlayer() and
ent == self:GetOwner():GetViewModel() and self.ViewModelFlip) then
ang.r = -ang.r -- Fixes mirrored models
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
return pos, ang
end
function SWEP:CreateModels( tab )
if (!tab) then return end
-- Create the clientside models here because Garry says we can't do it in the render hook
2014-10-02 08:49:54 +08:00
for k, v in pairs( tab ) do
if (v.type == "Model" and v.model and v.model != "" and (!IsValid(v.modelEnt) or v.createdModel != v.model) and
2014-10-02 08:49:54 +08:00
string.find(v.model, ".mdl") and file.Exists (v.model, "GAME") ) then
2014-10-02 08:49:54 +08:00
v.modelEnt = ClientsideModel(v.model, RENDER_GROUP_VIEW_MODEL_OPAQUE)
if (IsValid(v.modelEnt)) then
v.modelEnt:SetPos(self:GetPos())
v.modelEnt:SetAngles(self:GetAngles())
v.modelEnt:SetParent(self)
v.modelEnt:SetNoDraw(true)
v.createdModel = v.model
else
v.modelEnt = nil
end
elseif (v.type == "Sprite" and v.sprite and v.sprite != "" and (!v.spriteMaterial or v.createdSprite != v.sprite)
2014-10-02 08:49:54 +08:00
and file.Exists ("materials/"..v.sprite..".vmt", "GAME")) then
2014-10-02 08:49:54 +08:00
local name = v.sprite.."-"
local params = { ["$basetexture"] = v.sprite }
-- make sure we create a unique name based on the selected options
2014-10-02 08:49:54 +08:00
local tocheck = { "nocull", "additive", "vertexalpha", "vertexcolor", "ignorez" }
for i, j in pairs( tocheck ) do
if (v[j]) then
params["$"..j] = 1
name = name.."1"
else
name = name.."0"
end
end
v.createdSprite = v.sprite
v.spriteMaterial = CreateMaterial(name,"UnlitGeneric",params)
2014-10-02 08:49:54 +08:00
end
end
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
local allbones
local hasGarryFixedBoneScalingYet = true
2014-10-02 08:49:54 +08:00
function SWEP:UpdateBonePositions(vm)
2014-10-02 08:49:54 +08:00
if self.ViewModelBoneMods then
2014-10-02 08:49:54 +08:00
if (!vm:GetBoneCount()) then return end
-- !! WORKAROUND !! --
-- We need to check all model names :/
2014-10-02 08:49:54 +08:00
local loopthrough = self.ViewModelBoneMods
if (!hasGarryFixedBoneScalingYet) then
allbones = {}
for i=0, vm:GetBoneCount() do
local bonename = vm:GetBoneName(i)
if (self.ViewModelBoneMods[bonename]) then
2014-10-02 08:49:54 +08:00
allbones[bonename] = self.ViewModelBoneMods[bonename]
else
allbones[bonename] = {
2014-10-02 08:49:54 +08:00
scale = Vector(1,1,1),
pos = Vector(0,0,0),
angle = Angle(0,0,0)
}
end
end
2014-10-02 08:49:54 +08:00
loopthrough = allbones
end
-- !! ----------- !! --
2014-10-02 08:49:54 +08:00
for k, v in pairs( loopthrough ) do
local bone = vm:LookupBone(k)
if (!bone) then continue end
-- !! WORKAROUND !! --
2014-10-02 08:49:54 +08:00
local s = Vector(v.scale.x,v.scale.y,v.scale.z)
local p = Vector(v.pos.x,v.pos.y,v.pos.z)
local ms = Vector(1,1,1)
if (!hasGarryFixedBoneScalingYet) then
local cur = vm:GetBoneParent(bone)
while(cur >= 0) do
local pscale = loopthrough[vm:GetBoneName(cur)].scale
ms = ms * pscale
cur = vm:GetBoneParent(cur)
end
end
2014-10-02 08:49:54 +08:00
s = s * ms
-- !! ----------- !! --
2014-10-02 08:49:54 +08:00
if vm:GetManipulateBoneScale(bone) != s then
vm:ManipulateBoneScale( bone, s )
end
if vm:GetManipulateBoneAngles(bone) != v.angle then
vm:ManipulateBoneAngles( bone, v.angle )
end
if vm:GetManipulateBonePosition(bone) != p then
vm:ManipulateBonePosition( bone, p )
end
end
else
self:ResetBonePositions(vm)
end
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
function SWEP:ResetBonePositions(vm)
-- New code
vm:SetColor(color_white)
vm:SetMaterial("")
--------
if (!vm:GetBoneCount()) then return end
for i=0, vm:GetBoneCount() do
vm:ManipulateBoneScale( i, Vector(1, 1, 1) )
vm:ManipulateBoneAngles( i, Angle(0, 0, 0) )
vm:ManipulateBonePosition( i, Vector(0, 0, 0) )
end
2014-10-02 08:49:54 +08:00
end
--[[*************************
2014-10-02 08:49:54 +08:00
Global utility code
*************************]]
2014-10-02 08:49:54 +08:00
-- Fully copies the table, meaning all tables inside this table are copied too and so on (normal table.Copy copies only their reference).
-- Does not copy entities of course, only copies their reference.
-- WARNING: do not use on tables that contain themselves somewhere down the line or you'll get an infinite loop
2014-10-02 08:49:54 +08:00
function table.FullCopy( tab )
if (!tab) then return nil end
2014-10-02 08:49:54 +08:00
local res = {}
for k, v in pairs( tab ) do
if (type(v) == "table") then
res[k] = table.FullCopy(v) -- recursion ho!
2014-10-02 08:49:54 +08:00
elseif (type(v) == "Vector") then
res[k] = Vector(v.x, v.y, v.z)
elseif (type(v) == "Angle") then
res[k] = Angle(v.p, v.y, v.r)
else
res[k] = v
end
end
2014-10-02 08:49:54 +08:00
return res
2014-10-02 08:49:54 +08:00
end
2014-10-02 08:49:54 +08:00
end
----------------
-- Extra code --
----------------
function SWEP:RemoveModels()
if (self.VElements) then
for k, v in pairs( self.VElements ) do
if (IsValid( v.modelEnt )) then
v.modelEnt:Remove()
v.modelEnt = nil
end
end
end
if (self.WElements) then
for k, v in pairs( self.WElements ) do
if (IsValid( v.modelEnt )) then
v.modelEnt:Remove()
v.modelEnt = nil
end
end
end
end