wiremod-svn-archive/wire/lua/weapons/gmod_tool/stools/adv_duplicator.lua

992 lines
28 KiB
Lua
Raw Normal View History

2007-02-26 05:35:07 +00:00
//Advanced Duplicator by TAD2020
TOOL.Category = "Construction"
TOOL.Name = "Advanced Duplicator"
TOOL.Command = nil
TOOL.ConfigName = ""
if ( CLIENT ) then
language.Add( "Tool_adv_duplicator_name", "Advanced Duplicator" )
language.Add( "Tool_adv_duplicator_desc", "Duplicate an entity, or group of entities" )
2007-02-26 05:35:07 +00:00
language.Add( "Tool_adv_duplicator_0", "Right click to copy, Reload places Paster" )
language.Add( "Tool_duplicator_1", "Now left click to paste, Reload places Paster" )
end
2007-02-26 05:35:07 +00:00
TOOL.ClientConVar[ "simple" ] = 0
TOOL.ClientConVar[ "save_filename" ] = ""
TOOL.ClientConVar[ "load_filename" ] = ""
TOOL.ClientConVar[ "load_filename_cl" ] = ""
TOOL.ClientConVar[ "file_desc" ] = ""
TOOL.ClientConVar[ "delay" ] = 0
TOOL.ClientConVar[ "undo_delay" ] = 0
TOOL.ClientConVar[ "range" ] = "1500"
TOOL.ClientConVar[ "show_beam" ] = "1"
cleanup.Register( "duplicates" )
// Paste a copy
function TOOL:LeftClick( trace )
if ( CLIENT ) then return true end
Ents, Constraints = duplicator.Paste( self:GetOwner(), trace.HitPos )
if (!Ents) then return false end
for _, ent in pairs( Ents ) do self:GetOwner():AddCleanup( "duplicates", ent ) end
for _, ent in pairs( Constraints ) do self:GetOwner():AddCleanup( "duplicates", ent ) end
if (Ents) then return true end
//expermental test stuff
/*if (self.stage != 0) then return false end
local pln = self:GetOwner():UniqueID()
if (!duplicator[pln]) then return false end
self.Ents,self.Constraints,self.DupeInfo = nil,nil,nil
self.constIDtable, self.entIDtable, self.CreatedConstraints, self.CreatedEnts = {}, {}, {}, {}
//self.HeadEntity = nil
Msg("\n=======================--STOOLPasteStart--=======================\n")
self.Ents = table.Copy(duplicator[pln].Ents)
self.Constraints = table.Copy(duplicator[pln].Constraints)
self.DupeInfo = table.Copy(duplicator[pln].DupeInfo)
self.offset = trace.HitPos
if (self.Ents) then
self:HideGhost(true)
undo.Create("Duplicator")
self.thinker = 2
self.stage = 1
return true
else
return false
end*/
end
// Put The stuff we're pointing at in the 'clipboard'
function TOOL:RightClick( trace )
if (!trace.Entity) then return false end
if (!trace.Entity:IsValid()) then return false end
if (trace.Entity:IsPlayer()) then return false end
local StartPos = trace.Entity:GetPos()
local offset = {
start = StartPos,
endpos = StartPos + Vector( 0,0,-1024),
mask = MASK_NPCWORLDSTATIC
}
duplicator.offset = util.TraceLine( offset ).HitPos
self:ReleaseGhostEntity()
if ( CLIENT ) then return true end
// Call the duplicator module to get all the stuff constrained to the trace.ent
local Ents, Constraints = duplicator.Copy( self:GetOwner(), trace.Entity, duplicator.offset )
local simple = self:GetClientNumber( "simple" ) == 1
// this should tide us over until the whole thing gets re-written
if simple then
for EntID,Ent in pairs(Ents) do
if Ent != trace.Entity then
duplicator[self:GetOwner():UniqueID()].Ents[ EntID ] = nil
end
end
duplicator[self:GetOwner():UniqueID()].Constraints = {}
Ents, Constraints = { [ trace.Entity:EntIndex() ] = trace.Entity }, {}
end
local angle = self:GetOwner():GetAngles()
angle.pitch = 0
angle.roll = 0
local HoldAngle = angle - trace.Entity:GetAngles()
local pl = self:GetOwner()
local pln = pl:UniqueID()
duplicator[pln].HeadEntID = trace.Entity:EntIndex()
duplicator[pln].HoldAngle = HoldAngle
duplicator.LoadedFile = false
if !SinglePlayer() then
self:SendGhostToClient()
else
self:StartGhostEntities()
end
return true
end
//make a paster ent
function TOOL:Reload( trace )
if (CLIENT) then return true end
2007-02-26 05:35:07 +00:00
local pl = self:GetOwner()
local pln = self:GetOwner():UniqueID()
2007-02-26 05:35:07 +00:00
if (!duplicator[pln].Ents) then return false end
local paster = ents.Create( "gmod_adv_dupe_paster" )
if (!paster:IsValid()) then return false end
2007-02-26 05:35:07 +00:00
//paster:SetPos( pl:GetShootPos() + pl:GetAimVector() * 32 )
//paster:SetAngles( pl:GetAimVector() )
paster:SetPos( trace.HitPos )
local Ang = trace.HitNormal:Angle()
Ang.pitch = Ang.pitch + 90
paster:SetAngles( Ang )
paster:Spawn()
paster:SetPlayer( pl )
2007-02-26 05:35:07 +00:00
local min = paster:OBBMins()
paster:SetPos( trace.HitPos - trace.HitNormal * min.z )
local phys = paster:GetPhysicsObject()
if (phys:IsValid()) then phys:EnableMotion(false) end
//paster:SetNotSolid(true)
local ttable = {pl = pl}
table.Merge(paster:GetTable(), ttable )
undo.Create("Paster")
undo.AddEntity( paster )
undo.SetPlayer( pl )
undo.Finish()
2007-02-26 05:35:07 +00:00
local delay = self:GetClientNumber( "delay" )
local undo_delay = self:GetClientNumber( "undo_delay" )
local range = self:GetClientNumber("range")
local show_beam = (self:GetClientNumber("show_beam") == 1)
paster:Setup(table.Copy(duplicator[pln].Ents), table.Copy(duplicator[pln].Constraints), table.Copy(duplicator[pln].DupeInfo), table.Copy(duplicator[pln].DORInfo), duplicator[pln].HeadEntID, delay, undo_delay, range, show_beam)
return true
end
function TOOL.BuildCPanel( CPanel )
CPanel:AddControl( "Header", { Text = "#Tool_adv_duplicator_name", Description = "#Tool_adv_duplicator_desc" } )
CPanel:AddControl( "Checkbox", { Label = "#Simple Mode", Command = "adv_duplicator_simple" } )
end
function TOOL:Think()
if (CLIENT) then
local IsGhosts = (self.GhostEntities && #self.GhostEntities > 0)
if ( !IsGhosts && duplicator.GhostEnts && #duplicator.GhostEnts < 1 ) then return end
if ( !IsGhosts && duplicator.GhostEnts && #duplicator.GhostEnts > 0 ) then self:StartGhostEntities() end
end
self:UpdateGhostEntities()
//expermental test stuff
/*if (SERVER) then
if ((self.thinker or 0) < 1) then //skip a few frames
self.thinker = self.thinker +1
return true
end
self.thinker = 0
if (self.stage == 1) then
//Msg("starting ent\n")
for entID, EntTable in pairs(self.Ents) do
//Msg("doing ent "..entID.."\n")
local EntClass = EntTable.Class
// Check the antities class is registered with the duplicator
if EntClass and duplicator.GetEntType(EntClass) then
local Args = duplicator.PasteGetEntArgs( self:GetOwner(), EntTable, self.offset )
// make the Entity
Ent = duplicator.GetEntType(EntClass).Func(self:GetOwner(), unpack(Args))
if (Ent && Ent:IsValid()) then
Ent:GetPhysicsObject():EnableMotion(false)
Ent:SetNotSolid(true)
undo.AddEntity( Ent )
self.entIDtable[entID] = Ent
table.insert(self.CreatedEnts,Ent)
duplicator.PasteApplyEntMods( self:GetOwner(), Ent, EntTable )
end
elseif (EntClass) then
Msg("Duplicator Paste: Unknown class " .. EntClass .. "\n")
end
if ( entID == duplicator[self:GetOwner():UniqueID()].HeadEntID ) then
self.HeadEntity = Ent
end
if (table.Count(self.Ents) >= 1) then
self.Ents[entID] = nil
return true
end
end
Msg("going to stage 2\n")
self.stage = 2
return true
elseif (self.stage == 2) then
//Msg("starting const\n")
local Constraint = table.remove(self.Constraints)
if (Constraint) then
//Msg("doing const\n")
// Check If the constraint type has been registered with the duplicator
if Constraint.Type and duplicator.KnownConstraintType(Constraint.Type) then
local Args, DoConstraint = duplicator.PasteGetConstraintArgs( self:GetOwner(), Constraint, self.entIDtable, self.offset )
// make the constraint
if DoConstraint then
//Msg("making the const\n")
local const = duplicator.ConstraintTypeFunc( Constraint.Type, Args )
table.insert(self.CreatedConstraints,const)
undo.AddEntity( const )
if (Constraint.ConstID) then
self.constIDtable[Constraint.ConstID] = const
//Msg("Dupe add constraint ID: " .. Constraint.ConstID .. "\n")
end
end
end
return true
else
Msg("going to stage 3\n")
self.stage = 3
return true
end
elseif (self.stage == 3) then
undo.SetPlayer( self:GetOwner() )
undo.Finish()
//Msg("starting dupeinfo\n")
for id, infoTable in pairs(self.DupeInfo) do
local ent = self.entIDtable[id]
local isdupeinfo = false
if (ent) and (ent:IsValid()) and (infoTable) and (ent.ApplyDupeInfo) then
ent:ApplyDupeInfo(
self:GetOwner(), ent, infoTable,
function(id) return self.entIDtable[id] end,
function(id) return self.constIDtable[id] end
)
isdupeinfo = true
end
if (table.Count(self.DupeInfo) >= 1) then
self.DupeInfo[id] = nil
if (isdupeinfo) then return true end
end
end
Msg("going to stage 4\n")
self.stage = 4
return true
elseif (self.stage == 5) then
//this just looks wrong, so no
Msg("starting rotate\n")
duplicator.PasteRotate( self:GetOwner(), self.HeadEntity, self.CreatedEnts )
Msg("going to stage 5\n")
self.stage = 4
return true
elseif (self.stage == 4) then
//Msg("starting cleanup\n")
for _, ent in pairs( self.CreatedEnts ) do
self:GetOwner():AddCleanup( "duplicates", ent )
ent:GetPhysicsObject():EnableMotion(true)
ent:SetNotSolid(false)
end
for _, ent in pairs( self.CreatedConstraints ) do self:GetOwner():AddCleanup( "duplicates", ent ) end
self.stage = 0
Msg("====done====\n")
self:HideGhost(false)
end
end*/
end
/*---------------------------------------------------------
Make a ghost entity
---------------------------------------------------------*/
function TOOL:MakeGhostEntity_Duplicator( model, pos, angle )
if ( !model or model == "" ) then
//Msg("Model is NULL!\n")
model = "models/props_junk/watermelon01.mdl"
end
util.PrecacheModel( model )
// We do ghosting serverside in single player
// It's done clientside in multiplayer
if (SERVER && !SinglePlayer()) then return end
if (CLIENT && SinglePlayer()) then return end
local GhostEntity = ents.Create( "prop_physics" )
// If there's too many entities we might not spawn..
if ( !GhostEntity || GhostEntity == NULL ) then return end
if ( !util.IsValidProp( model ) ) then
model = "models/props_junk/watermelon01.mdl"
end
GhostEntity:SetModel( model )
GhostEntity:SetPos( pos )
GhostEntity:SetAngles( angle )
GhostEntity:Spawn()
GhostEntity:SetMoveType( MOVETYPE_NONE )
GhostEntity:SetSolid( SOLID_NONE );
GhostEntity:SetNotSolid( true );
GhostEntity:SetRenderMode( RENDERMODE_TRANSALPHA )
GhostEntity:SetColor( 255, 255, 255, 150 )
return GhostEntity
end
/*---------------------------------------------------------
Starts up the ghost entities
---------------------------------------------------------*/
function TOOL:StartGhostEntities()
// We do ghosting serverside in single player
// It's done clientside in multiplayer
if (SERVER && !SinglePlayer()) then return end
if (CLIENT && SinglePlayer()) then return end
if (CLIENT) then return end
local pl = self:GetOwner()
local pln = self:GetOwner():UniqueID()
if (!duplicator[pln]) or (!duplicator[pln].HeadEntID) then return end //gonna need data also seems to stop client ghost in mp
// Clear any existing ghosts
self:ReleaseGhostEntity()
// set us up some tables
self.GhostEntities = {}
self.GhostOffset = {}
//get the point where we are currently looking
local tr = utilx.GetPlayerTrace( pl )
local trace = util.TraceLine( tr )
if (!trace.Hit) then return end
duplicator.offset = trace.HitPos
//local Ents = pl:GetTable().Duplicator.Ents
local Ents = duplicator[pln].Ents
//load and make the ghost
for entID, EntTable in pairs(Ents) do
local entClass = EntTable.Class
if entClass then
//get the args for this class
local entArgs = duplicator.GetEntClassArgs(entClass)
if entArgs then //check that we know how to dupe this class
local Args = { model="models/props_junk/watermelon01.mdl", pos=Vector(0,0,0), ang=Vector(0,0,0) }
for n,Key in pairs(entArgs) do
if type(Key) != "table" then
local Arg = EntTable[Key]
key = string.lower(Key)
if key == "ang" or key == "angle" then
Args.ang = Arg or Vector(0,0,0)
elseif key == "pos" or key == "position" then
Args.pos = Arg + duplicator.offset or Vector(0,0,0)
elseif key == "mdl" or key == "model" or key == "smodel" then
Args.model = Arg
end
end
end
local GhostEntity = self:MakeGhostEntity_Duplicator( Args.model, Args.pos, Args.ang )
if ( GhostEntity ) then
self.GhostEntities[entID] = GhostEntity
self.GhostOffset[entID] = Args.pos - duplicator.offset
if ( entID == duplicator[pln].HeadEntID ) then
self.GhostHead = GhostEntity
self.GhostHeadOff = Args.pos - duplicator.offset
end
end
elseif (EntClass) then
Msg("Duplicator Paste: Unknown class " .. EntClass .. "\n")
end
end
end
if (!self.GhostHead) then
Msg("Duplicator Error, no head entity!")
self:ReleaseGhostEntity()
return
end
for k, ent in pairs ( self.GhostEntities ) do
if ( self.GhostHead != ent ) then
ent:SetParent( self.GhostHead )
end
end
end
function TOOL:UpdateGhostEntities()
if (!self.GhostEntities) then return end
local tr = utilx.GetPlayerTrace( self:GetOwner(), self:GetOwner():GetCursorAimVector() )
local trace = util.TraceLine( tr )
if (!trace.Hit) then return end
local ent = self.GhostHead
if (!ent) then return end
if (ent == NULL) then return end
ent:SetPos( trace.HitPos + self.GhostHeadOff )
/*local angle = self:GetOwner():GetAngles()
angle.pitch = 0
angle.roll = 0
if (!CLIENT) then
ent:SetAngles( angle - duplicator[self:GetOwner():UniqueID()].HoldAngle )
else
ent:SetAngles( angle - duplicator.HoldAngle )
end*/
end
function TOOL:SendGhostToClient(fileload)
if (CLIENT) then return end
if (SERVER && SinglePlayer()) then return end
local pl = self:GetOwner()
Msg("=====\nsentghosttoclient pl = ")
Msg(tostring(pl))
local pln = pl:UniqueID()
Msg("\nstart send ghost data\n")
if (!duplicator[pln].Ents) then return false end
local ents = duplicator[pln].Ents
local HeadEntID = duplicator[pln].HeadEntID
if (fileload) then
//get the point where we are currently looking
local tr = utilx.GetPlayerTrace( pl )
local trace = util.TraceLine( tr )
if (!trace.Hit) then return end
duplicator.offset = trace.HitPos
end
local entscount = table.Count(duplicator[pln].Ents)
if (entscount > 50) then entscount = 50 end
Msg("entscount: "..entscount.."\n")
self:SendGhostDataHeadToClient( HeadEntID, duplicator.offset, duplicator[pln].HoldAngle, entscount )
local sendnum = 1
local Args = {}
Msg("sending headent "..HeadEntID.."\n")
Args = self:GetGhostArgsForClient( HeadEntID, ents[HeadEntID] )
self:SendGhostDataUMsgToClient( HeadEntID, Args )
for entID, EntTable in pairs(ents) do
if (entID != HeadEntID) then
Args = self:GetGhostArgsForClient( entID, EntTable )
self:SendGhostDataUMsgToClient( entID, Args )
end
sendnum = sendnum + 1
if (sendnum > 50) then return end //only send 50
end
end
function TOOL:SendGhostDataHeadToClient( HeadEntID, offset, HoldAngle, entscount )
umsg.Start("ClientGhostDataHead", pl)
umsg.Short(HeadEntID)
umsg.Vector(offset)
umsg.Angle(HoldAngle)
umsg.Short(entscount)
umsg.End()
end
function TOOL:GetGhostArgsForClient( entID, EntTable )
if (CLIENT) then return end
local entClass = EntTable.Class
Msg("start send ent ( "..entID.." ) data, class: "..(entClass or 0).."\n")
if entClass then
//get the args for this class
local entArgs = duplicator.GetEntClassArgs(entClass)
if entArgs then //check that we know how to dupe this class
local Args = { model="models/props_junk/watermelon01.mdl", pos=Vector(0,0,0), ang=Vector(0,0,0) }
for n,Key in pairs(entArgs) do
if type(Key) != "table" then
local Arg = EntTable[Key]
key = string.lower(Key)
if key == "ang" or key == "angle" then
Args.ang = Arg or Vector(0,0,0)
elseif key == "pos" or key == "position" then
Args.pos = Arg or Vector(0,0,0)
elseif key == "mdl" or key == "model" or key == "smodel" then
Args.model = Model(Arg)
end
end
end
return Args
end
end
return false
end
function TOOL:SendGhostDataUMsgToClient( entID, Args )
if (CLIENT) then return end
if (Args) then
umsg.Start("ClientGhostData", pl)
umsg.Short(entID)
umsg.String(Args.model)
umsg.Vector(Args.pos)
umsg.Angle(Args.ang)
umsg.End()
else
umsg.Start("ClientGhostData", pl)
umsg.Short(0)
umsg.End()
end
end
//makes the ghost disappear so we can see what we are doing
function TOOL:HideGhost(hide)
if (CLIENT) then return end
if (SERVER && !SinglePlayer()) then return end
if (hide) then
for _, ghostent in pairs( self.GhostEntities ) do
ghostent:SetColor( 255, 255, 255, 0 )
end
else
for _, ghostent in pairs( self.GhostEntities ) do
ghostent:SetColor( 255, 255, 255, 150 )
end
end
end
function TOOL:UpdateList()
if (!self:GetOwner():IsValid()) then return false end
if (!self:GetOwner():IsPlayer()) then return false end
local dir = "adv_duplicator"
local ndir = dir.."/"..string.gsub(self:GetOwner():GetName(), ":", "_")
2007-02-26 05:35:07 +00:00
self:GetOwner():SendLua( "if ( !duplicator ) then duplicator={} end" )
self:GetOwner():SendLua( "duplicator.LoadList={}" )
if ( file.Exists(dir) && file.IsDir(dir) ) then
for key, val in pairs( file.Find( dir.."/*" ) ) do
if ( !file.IsDir( dir.."/"..val ) ) then
self:GetOwner():SendLua( "table.insert(duplicator.LoadList,\"".. val .."\")" )
end
end
end
if ( file.Exists(ndir) && file.IsDir(ndir) ) then
for key, val in pairs( file.Find( ndir.."/*" ) ) do
if ( !file.IsDir( ndir.."/"..val ) ) then
self:GetOwner():SendLua( "table.insert(duplicator.LoadList,\"".. val .."\")" )
end
end
end
// Force user to update list
2007-02-26 05:35:07 +00:00
self:GetOwner():SendLua( "AdvDuplicator_UpdateControlPanel(\""..ndir.."\")" )
end
function TOOL:Deploy()
if ( CLIENT ) then return end
self.stage = 0
self.thinker = 0
if (duplicator[self:GetOwner():UniqueID()]) then self:StartGhostEntities() end
self:UpdateList()
end
if SERVER then
//Serverside save of duplicated ents
local function AdvDupeSS_Save( pl, _, args )
if !pl:IsValid()
or !pl:IsPlayer()
//or !pl:GetTable().Duplicator
or !duplicator[pl:UniqueID()]
then return end
//save to file
duplicator.SaveToFile( pl, tostring(pl:GetInfo( "adv_duplicator_save_filename" )), tostring(pl:GetInfo( "adv_duplicator_file_desc" )) )
pl:GetWeapon( "gmod_tool" ):GetTable():GetToolObject():UpdateList()
end
concommand.Add( "adv_duplicator_save", AdvDupeSS_Save )
//Load duplicated ents from file
local function AdvDupeSS_Load( pl, command, args )
if !pl:IsValid()
or !pl:IsPlayer()
then return end
local filename = ""
if !args[1] //if a filename wasn't passed with a arg, then get the selection in the panel
then filename = pl:GetInfo( "adv_duplicator_load_filename" )
else filename = tostring(args[1]) end
duplicator.LoadFromFile( pl, filename )
if ( pl:GetActiveWeapon():GetClass() == "gmod_tool" ) then
//pl:SendLua( "LocalPlayer():GetActiveWeapon():GetTable():GetToolObject():StartGhostEntities()")
pl:GetActiveWeapon():GetTable():GetToolObject():StartGhostEntities()
pl:GetActiveWeapon():GetTable():GetToolObject():SendGhostToClient(true)
pl:SendLua( "LocalPlayer():GetActiveWeapon():GetTable():GetToolObject():UpdateGhostEntities()" )
end
end
concommand.Add( "adv_duplicator_load", AdvDupeSS_Load )
// Clientside save of duplicated ents
local function AdvDupeCL_Save( pl, command, args )
2007-02-26 05:35:07 +00:00
if !pl:IsValid()
or !pl:IsPlayer()
2007-02-26 05:35:07 +00:00
//or !pl:GetTable().Duplicator
or !duplicator[pl:UniqueID()]
then return end
2007-02-26 05:35:07 +00:00
//save to file
duplicator.SaveAndSendSaveToClient( pl, tostring(pl:GetInfo( "adv_duplicator_save_filename" )), tostring(pl:GetInfo( "adv_duplicator_file_desc" )) )
2007-02-26 05:35:07 +00:00
pl:GetWeapon( "gmod_tool" ):GetTable():GetToolObject():UpdateList()
2007-02-26 05:35:07 +00:00
end
concommand.Add( "adv_duplicator_save_cl", AdvDupeCL_Save )
//sends the selected file to the client
local function AdvDupeCl_Send( pl, command, args )
2007-02-26 05:35:07 +00:00
if !pl:IsValid()
or !pl:IsPlayer()
then return end
2007-02-26 05:35:07 +00:00
local filename = ""
if !args[1] //if a filename wasn't passed with a arg, then get the selection in the panel
then filename = pl:GetInfo( "adv_duplicator_load_filename" )
else filename = tostring(args[1]) end
2007-02-26 05:35:07 +00:00
duplicator.SendSaveToClient( pl, filename )
end
2007-02-26 05:35:07 +00:00
concommand.Add( "adv_duplicator_send_cl", AdvDupeCl_Send )
local function AdvDupeSS_UpdateLoadList( pl, command, args )
pl:GetWeapon( "gmod_tool" ):GetTable():GetToolObject():UpdateList()
end
concommand.Add( "adv_duplicator_updatelist", AdvDupeSS_UpdateLoadList )
local function AdvDupeSS_StartGhostEntities( pl, command, args )
pl:GetWeapon( "gmod_tool" ):GetTable():GetToolObject():StartGhostEntities()
end
concommand.Add( "adv_duplicator_startghost", AdvDupeSS_StartGhostEntities )
else // CLIENT
2007-02-26 05:35:07 +00:00
function AdvDuplicator_UpdateControlPanel(ndir)
local CPanel = GetControlPanel( "adv_duplicator" )
if (CPanel != nil) then
CPanel:ClearControls()
CPanel:AddHeader()
CPanel:AddDefaultControls()
local params = {}
params.Label = "#Duplicator_load"
params.Height = 180
params.Options = {}
if ( duplicator.LoadList ) then
for k,v in pairs( duplicator.LoadList ) do
params.Options[v] = {}
params.Options[v].adv_duplicator_load_filename = v
end
end
CPanel:AddControl( "ListBox", params )
2007-02-26 05:35:07 +00:00
if (SinglePlayer()) then
local params = {}
params.Text = "Load Selected File"
params.Command = "adv_duplicator_load"
CPanel:AddControl( "Button", params )
2007-02-26 05:35:07 +00:00
local params = {}
params.Text = "#Duplicator_save"
params.Command = "adv_duplicator_save"
CPanel:AddControl( "Button", params )
end
if (!SinglePlayer()) then
local params = {}
params.Text = "Load Selected File From Server"
params.Command = "adv_duplicator_load"
CPanel:AddControl( "Button", params )
2007-02-26 05:35:07 +00:00
local params = {}
params.Text = "Save To Server"
params.Command = "adv_duplicator_save"
CPanel:AddControl( "Button", params )
2007-02-26 05:35:07 +00:00
local params = {}
params.Text = "Save to Server Then Download"
params.Command = "adv_duplicator_save_cl"
CPanel:AddControl( "Button", params )
2007-02-26 05:35:07 +00:00
local params = {}
params.Text = "Download Selected File"
params.Command = "adv_duplicator_send_cl"
CPanel:AddControl( "Button", params )
end
CPanel:AddControl("TextBox", {
Label = "Filename:",
Command = "adv_duplicator_save_filename"})
CPanel:AddControl("TextBox", {
Label = "Description:",
Command = "adv_duplicator_file_desc"})
2007-02-26 05:35:07 +00:00
CPanel:AddControl( "Slider", {
Label = "Spawn Delay",
Type = "Float",
Min = "0",
Max = "100",
Command = "adv_duplicator_delay"})
CPanel:AddControl( "Slider", {
Label = "Automatic Undo Delay",
Type = "Float",
Min = "0",
Max = "100",
Command = "adv_duplicator_undo_delay"})
CPanel:AddControl("Slider", {
Label = "Range",
Type = "Float",
Min = "0",
Max = "1000",
Command = "adv_duplicator_range"})
CPanel:AddControl("CheckBox", {
Label = "Show Beam",
Command = "adv_duplicator_show_beam"})
if (!SinglePlayer()) then
local dir = "adv_duplicator"
local ndir = dir.."/"..string.gsub(LocalPlayer():GetName(), ":", "_")
local params = {}
params.Label = "ClientSide Files"
params.Height = 180
params.Options = {}
if ( file.Exists(dir) && file.IsDir(dir) ) then
for key, val in pairs( file.Find( dir.."/*" ) ) do
if ( !file.IsDir( dir.."/"..val ) ) then
params.Options[val] = {}
params.Options[val].adv_duplicator_load_filename_cl = val
end
end
end
if (ndir && file.Exists(ndir) && file.IsDir(ndir) ) then
for key, val in pairs( file.Find( ndir.."/*" ) ) do
if ( !file.IsDir( ndir.."/"..val ) ) then
params.Options[val] = {}
params.Options[val].adv_duplicator_load_filename_cl = val
end
end
end
CPanel:AddControl( "ListBox", params )
CPanel:AddControl( "Button", {
Text = "Upload File to server",
Command = "adv_duplicator_load_cl"})
end
end
end
2007-02-26 05:35:07 +00:00
local function AdvDupeCS_UpLoad( pl, command, args )
if !pl:IsValid()
or !pl:IsPlayer()
then return end
local filename = ""
if !args[1] //if a filename wasn't passed with an arg, then get the selection in the panel
then filename = pl:GetInfo( "adv_duplicator_load_filename_cl" )
else filename = tostring(args[1]) end
duplicatorclient.UpLoadFile( pl, filename )
end
concommand.Add( "adv_duplicator_load_cl", AdvDupeCS_UpLoad )
local function AdvDuplicator_ClientGhostDataHead( um )
Msg("=========\ngot ghost head data ========== ")
duplicator.HeadEntID = um:ReadShort()
duplicator.offset = um:ReadVector()
duplicator.HoldAngle = um:ReadAngle()
local tool = LocalPlayer():GetActiveWeapon():GetTable():GetToolObject()
tool:ReleaseGhostEntity()
tool.GhostEntities = {}
tool.GhostOffset = {}
tool.NumToRecieve = um:ReadShort()
tool.NumRecieved = 0
Msg("HeadEntID= "..duplicator.HeadEntID.."\nNumToRecieve= "..tool.NumToRecieve.."\n==========\n")
end
usermessage.Hook("ClientGhostDataHead", AdvDuplicator_ClientGhostDataHead)
local function AdvDuplicator_ClientGhostData( um )
local entID = um:ReadShort()
Msg("getting ghost data for ent: "..entID.." ")
local tool = LocalPlayer():GetActiveWeapon():GetTable():GetToolObject()
tool.NumRecieved = tool.NumRecieved + 1
if (entID == 0) then return end
local model = um:ReadString()
local pos = um:ReadVector()
local ang = um:ReadAngle()
Msg("model: "..model.."\n")
local GhostEntity = tool:MakeGhostEntity_Duplicator( model, (pos + duplicator.offset), ang )
if ( GhostEntity ) then
//Msg("ghost ent good\n")
tool.GhostEntities[entID] = GhostEntity
tool.GhostOffset[entID] = pos
if ( entID == duplicator.HeadEntID ) then
Msg("ghost head ent found \n")
tool.GhostHead = GhostEntity
tool.GhostHeadOff = pos
elseif (tool.GhostHead) then //cause we send the head first, we can do this
GhostEntity:SetParent( tool.GhostHead )
end
end
if (tool.NumRecieved >= tool.NumToRecieve) then
Msg("client ghost finishing\n")
if (!tool.GhostHead) then
Msg("client Duplicator Error, no head entity!\n")
tool:ReleaseGhostEntity()
return
end
for k, ent in pairs ( tool.GhostEntities ) do
if ( tool.GhostHead != ent ) then
ent:SetParent( tool.GhostHead )
end
end
end
end
usermessage.Hook("ClientGhostData", AdvDuplicator_ClientGhostData)
end