dobrograd-13-06-2022/garrysmod/addons/feature-wire/lua/entities/base_wire_entity.lua

446 lines
13 KiB
Lua
Raw Normal View History

2023-11-16 15:01:19 +05:00
AddCSLuaFile()
DEFINE_BASECLASS( "base_gmodentity" )
ENT.Type = "anim"
ENT.PrintName = "Wire Unnamed Ent"
ENT.Purpose = "Base for all wired SEnts"
ENT.RenderGroup = RENDERGROUP_OPAQUE
ENT.Spawnable = false
ENT.AdminOnly = false
ENT.IsWire = true
if CLIENT then
local wire_drawoutline = CreateClientConVar("wire_drawoutline", 1, true, false)
function ENT:Initialize()
self.NextRBUpdate = CurTime() + 0.25
end
function ENT:Draw()
self:DoNormalDraw()
Wire_Render(self)
if self.GetBeamLength and (not self.GetShowBeam or self:GetShowBeam()) then
-- Every SENT that has GetBeamLength should draw a tracer. Some of them have the GetShowBeam boolean
Wire_DrawTracerBeam( self, 1, self.GetBeamHighlight and self:GetBeamHighlight() or false )
end
end
local WorldTip = { dietime = 0 }
function ENT:AddWorldTip( txt )
WorldTip.dietime = SysTime() + RealFrameTime() * 4
WorldTip.ent = self
end
local edgesize = 18
-- makes sure the overlay doesn't go out of the screen & provides several useful sizes and positions for the DrawBody function
function ENT:GetWorldTipPositions( w, h, w_body, h_body, w_footer, h_footer )
local pos = LocalPlayer():GetEyeTrace().HitPos
local spos = LocalPlayer():GetShootPos()
if pos == spos then -- if the position is right in your face, get a better position
pos = spos + LocalPlayer():GetAimVector() * 5
end
pos = pos:ToScreen()
pos.x = math.Round(pos.x)
pos.y = math.Round(pos.y)
w = math.min( w, ScrW() - 64 )
h = math.min( h, ScrH() - 64 )
local maxx = pos.x - 32
local maxy = pos.y - 32
local minx = maxx - w
local miny = maxy - h
if minx < 32 then
maxx = 32 + w
minx = 32
end
if miny < 32 then
maxy = 32 + h
miny = 32
end
local centerx = (maxx+minx)/2
local centery = (maxy+miny)/2
return { min = {x = minx,y = miny},
max = {x = maxx,y = maxy},
center = {x = centerx, y = centery},
size = {w = w, h = h},
bodysize = {w = w_body, h = h_body },
footersize = {w = w_footer, h = h_footer},
edgesize = edgesize
}
end
-- This is overridable by other wire entities which want to customize the overlay, but generally you shouldn't override it
function ENT:DrawWorldTipOutline( pos )
draw.NoTexture()
surface.SetDrawColor(Color(25,25,25,200))
local poly = {
{x = pos.min.x + edgesize, y = pos.min.y, u = 0, v = 0 },
{x = pos.max.x, y = pos.min.y, u = 0, v = 0 },
{x = pos.max.x, y = pos.max.y - edgesize + 0.5, u = 0, v = 0 },
{x = pos.max.x - edgesize + 0.5, y = pos.max.y, u = 0, v = 0 },
{x = pos.min.x, y = pos.max.y, u = 0, v = 0 },
{x = pos.min.x, y = pos.min.y + edgesize, u = 0, v = 0 },
}
render.CullMode(MATERIAL_CULLMODE_CCW)
surface.DrawPoly( poly )
surface.SetDrawColor(Color(0,0,0,255))
for i=1,#poly-1 do
surface.DrawLine( poly[i].x, poly[i].y, poly[i+1].x, poly[i+1].y )
end
surface.DrawLine( poly[#poly].x, poly[#poly].y, poly[1].x, poly[1].y )
end
local function getWireName( ent )
local name = ent:GetNWString("WireName")
if not name or name == "" then return ent.PrintName else return name end
end
-- This is overridable by other wire entities which want to customize the overlay
function ENT:GetWorldTipBodySize()
local txt = self:GetOverlayData().txt
if txt == nil or txt == "" then return 0,0 end
return surface.GetTextSize( txt )
end
-- This is overridable by other wire entities which want to customize the overlay
function ENT:DrawWorldTipBody( pos )
local data = self:GetOverlayData()
draw.DrawText( data.txt, "GModWorldtip", pos.center.x, pos.min.y + edgesize/2, Color(255,255,255,255), TEXT_ALIGN_CENTER )
end
-- This is overridable by other wire entities which want to customize the overlay
function ENT:DrawWorldTip()
local data = self:GetOverlayData()
if not data then return end
surface.SetFont( "GModWorldtip" )
local txt = data.txt
local class = getWireName( self ) .. " [" .. self:EntIndex() .. "]"
local name = "(" .. self:GetPlayerName() .. ")"
local w_body, h_body = self:GetWorldTipBodySize()
local w_class, h_class = surface.GetTextSize( class )
local w_name, h_name = surface.GetTextSize( name )
local w_total = txt ~= "" and w_body or 0
local h_total = txt ~= "" and h_body or 0
local w_footer, h_footer = 0, 0
local info_requires_multiline = false
if w_total < w_class + w_name - edgesize then
info_requires_multiline = true
w_footer = math.max(w_total,w_class,w_name)
h_footer = h_class + h_name + edgesize + 8
w_total = w_footer
h_total = h_total + h_footer
else
w_footer = math.max(w_total,w_class + 8 + w_name)
h_footer = math.max(h_class,h_name) + edgesize + 8
w_total = w_footer
h_total = h_total + h_footer
end
if h_body == 0 then h_total = h_total - h_body - edgesize end
local pos = self:GetWorldTipPositions( w_total + edgesize*2,h_total + edgesize,
w_body,h_body,
w_footer,h_footer )
self:DrawWorldTipOutline( pos )
local offset = pos.min.y
if h_body > 0 then
self:DrawWorldTipBody( pos )
offset = offset + h_body + edgesize
surface.SetDrawColor( Color(0,0,0,255) )
surface.DrawLine( pos.min.x, offset, pos.max.x, offset )
end
if info_requires_multiline then
draw.DrawText( class, "GModWorldtip", pos.center.x, offset + 8, Color(255,255,255,255), TEXT_ALIGN_CENTER )
draw.DrawText( name, "GModWorldtip", pos.center.x, offset + h_class + 16, Color(255,255,255,255), TEXT_ALIGN_CENTER )
else
draw.DrawText( class, "GModWorldtip", pos.min.x + edgesize, offset + 16, Color(255,255,255,255) )
draw.DrawText( name, "GModWorldtip", pos.min.x + pos.size.w - w_name - edgesize, offset + 16, Color(255,255,255,255) )
end
end
hook.Add("HUDPaint","wire_draw_world_tips",function()
if SysTime() > WorldTip.dietime then return end
local ent = WorldTip.ent
if not IsValid(ent) then return end
ent:DrawWorldTip()
end)
-- Custom better version of this base_gmodentity function
function ENT:BeingLookedAtByLocalPlayer()
local trace = LocalPlayer():GetEyeTrace()
if trace.Entity ~= self then return false end
if trace.HitPos:Distance(LocalPlayer():GetShootPos()) > 200 then return false end
return true
end
local drawInfo = false
hook.Add('Think', 'wire.drawInfo', function()
local ply = LocalPlayer()
if not IsValid(ply) then return end
local wep = ply:GetActiveWeapon()
drawInfo = IsValid(wep) and wep:GetClass() == 'gmod_tool'
end)
function ENT:DoNormalDraw(nohalo, notip)
if not drawInfo then
return self:DrawModel()
end
local looked_at = self:BeingLookedAtByLocalPlayer()
if not nohalo and wire_drawoutline:GetBool() and looked_at then
if self.RenderGroup == RENDERGROUP_OPAQUE then
self.OldRenderGroup = self.RenderGroup
self.RenderGroup = RENDERGROUP_TRANSLUCENT
end
self:DrawEntityOutline()
self:DrawModel()
else
if self.OldRenderGroup then
self.RenderGroup = self.OldRenderGroup
self.OldRenderGroup = nil
end
self:DrawModel()
end
if not notip and looked_at then
self:AddWorldTip()
end
end
-- function ENT:Think()
-- if (CurTime() >= (self.NextRBUpdate or 0)) then
-- -- We periodically update the render bounds every 10 seconds - the
-- -- reasons why are mostly anecdotal, but in some circumstances
-- -- entities might 'forget' their renderbounds. Nobody really knows
-- -- if this is still needed or not.
-- self.NextRBUpdate = CurTime() + 10
-- Wire_UpdateRenderBounds(self)
-- end
-- end
local halos = {}
local halos_inv = {}
function ENT:DrawEntityOutline()
if halos_inv[self] then return end
halos[#halos+1] = self
halos_inv[self] = true
end
hook.Add("PreDrawHalos", "Wiremod_overlay_halos", function()
if #halos == 0 then return end
halo.Add(halos, Color(100,100,255), 3, 3, 1, true, true)
halos = {}
halos_inv = {}
end)
--------------------------------------------------------------------------------
-- Overlay getting
--------------------------------------------------------------------------------
-- Basic legacy GetOverlayText, is no longer used here but we leave it here in case other addons rely on it.
function ENT:GetOverlayText()
local name = self:GetNWString("WireName")
if name == "" then name = self.PrintName end
local header = "- " .. name .. " -"
local data = self:GetOverlayData()
if data and data.txt then
return header .. "\n" .. data.txt
else
return header
end
end
--------------------------------------------------------------------------------
-- Overlay receiving
--------------------------------------------------------------------------------
net.Receive( "wire_overlay_data", function( len )
local ent = net.ReadEntity()
if IsValid( ent ) then
ent.OverlayData = net.ReadTable()
end
end )
end
--------------------------------------------------------------------------------
-- Overlay setting
--------------------------------------------------------------------------------
-- We want more fine-grained control over everything related to overlays,
-- so we have a custom system here
-- It allows us to optionally send values rather than entire strings, which saves networking
-- It also allows us to only update overlays when someone is looking at the entity.
function ENT:SetOverlayText( txt )
if not self.OverlayData then
self.OverlayData = {}
end
if txt and #txt > 12000 then
txt = string.sub(txt,1,12000) -- I have tested this and 12000 chars is enough to cover the entire screen at 1920x1080. You're unlikely to need more
end
self.OverlayData.txt = txt
if not self.OverlayData_UpdateTime then self.OverlayData_UpdateTime = {} end
self.OverlayData_UpdateTime.time = CurTime()
end
function ENT:SetOverlayData( data )
self.OverlayData = data
if self.OverlayData.txt and #self.OverlayData.txt > 12000 then
self.OverlayData.txt = string.sub(self.OverlayData.txt,1,12000)
end
if not self.OverlayData_UpdateTime then self.OverlayData_UpdateTime = {} end
self.OverlayData_UpdateTime.time = CurTime()
end
function ENT:GetOverlayData()
return self.OverlayData
end
if CLIENT then return end -- no more client
--------------------------------------------------------------------------------
-- Overlay syncing
--------------------------------------------------------------------------------
util.AddNetworkString( "wire_overlay_data" )
timer.Create("WireOverlayUpdate", 0.1, 0, function()
for _, ply in ipairs(player.GetAll()) do
local ent = ply:GetEyeTrace().Entity
if IsValid(ent) and ent.IsWire and
ent.OverlayData and
ent.OverlayData_UpdateTime and
ent.OverlayData_UpdateTime.time > (ent.OverlayData_UpdateTime[ply] or 0) then
ent.OverlayData_UpdateTime[ply] = CurTime()
net.Start( "wire_overlay_data" )
net.WriteEntity( ent )
net.WriteTable( ent.OverlayData )
net.Send( ply )
end
end
end)
--------------------------------------------------------------------------------
-- Other functions
--------------------------------------------------------------------------------
function ENT:Initialize()
BaseClass.Initialize(self)
self:PhysicsInit(SOLID_VPHYSICS)
self:SetMoveType(MOVETYPE_VPHYSICS)
self:SetSolid(SOLID_VPHYSICS)
self.WireDebugName = self.WireDebugName or (self.PrintName and self.PrintName:sub(6)) or self:GetClass():gsub("gmod_wire", "")
end
function ENT:OnRemove()
WireLib.Remove(self)
end
function ENT:OnRestore()
WireLib.Restored(self)
end
function ENT:BuildDupeInfo()
return WireLib.BuildDupeInfo(self)
end
function ENT:ApplyDupeInfo(ply, ent, info, GetEntByID)
WireLib.ApplyDupeInfo(ply, ent, info, GetEntByID)
end
function ENT:PreEntityCopy()
duplicator.ClearEntityModifier(self, "WireDupeInfo")
-- build the DupeInfo table and save it as an entity mod
local DupeInfo = self:BuildDupeInfo()
if DupeInfo then
duplicator.StoreEntityModifier(self, "WireDupeInfo", DupeInfo)
end
end
function ENT:OnEntityCopyTableFinish(dupedata)
-- Called by Garry's duplicator, to modify the table that will be saved about an ent
-- Remove anything with non-string keys, or util.TableToJSON will crash the game
dupedata.OverlayData_UpdateTime = nil
end
local function EntityLookup(CreatedEntities)
return function(id, default)
if id == nil then return default end
if id == 0 then return game.GetWorld() end
local ent = CreatedEntities[id]
if IsValid(ent) then return ent else return default end
end
end
function ENT:OnDuplicated()
self.DuplicationInProgress = true
end
function ENT:PostEntityPaste(Player,Ent,CreatedEntities)
-- We manually apply the entity mod here rather than using a
-- duplicator.RegisterEntityModifier because we need access to the
-- CreatedEntities table.
if Ent.EntityMods and Ent.EntityMods.WireDupeInfo then
Ent:ApplyDupeInfo(Player, Ent, Ent.EntityMods.WireDupeInfo, EntityLookup(CreatedEntities))
end
self.DuplicationInProgress = nil
end
-- Helper function for entities that can be linked
ENT.LINK_STATUS_UNLINKED = 1
ENT.LINK_STATUS_LINKED = 2
ENT.LINK_STATUS_INACTIVE = 2 -- alias
ENT.LINK_STATUS_DEACTIVATED = 2 -- alias
ENT.LINK_STATUS_ACTIVE = 3
ENT.LINK_STATUS_ACTIVATED = 3 -- alias
function ENT:ColorByLinkStatus(status)
local a = self:GetColor().a
if status == self.LINK_STATUS_UNLINKED then
self:SetColor(Color(255,0,0,a))
elseif status == self.LINK_STATUS_LINKED then
self:SetColor(Color(255,165,0,a))
elseif status == self.LINK_STATUS_ACTIVE then
self:SetColor(Color(0,255,0,a))
else
self:SetColor(Color(255,255,255,a))
end
end