Better tempaltes
authorNovatux <nathanael.courant@laposte.net>
Sat, 19 Oct 2013 09:27:47 +0000 (11:27 +0200)
committerNovatux <nathanael.courant@laposte.net>
Sat, 19 Oct 2013 09:28:16 +0000 (11:28 +0200)
technic/machines/other/frames.lua

index df117dbdaf2dbb3042df6497f0df6a5b6a80fdd5..80bf9e5d10575ddcbfebc8278258787daa847b1d 100644 (file)
@@ -40,6 +40,25 @@ local function get_face(pos,ppos,pvect)
        end
 end
 
+function lines(str)
+       local t = {}
+       local function helper(line) table.insert(t, line) return "" end
+       helper((str:gsub("(.-)\r?\n", helper)))
+       return t
+end
+
+local function pos_to_string(pos)
+       if pos.x == 0 then pos.x = 0 end -- Fix for signed 0
+       if pos.y == 0 then pos.y = 0 end -- Fix for signed 0
+       if pos.z == 0 then pos.z = 0 end -- Fix for signed 0
+       return tostring(pos.x).."\n"..tostring(pos.y).."\n"..tostring(pos.z)
+end
+
+local function pos_from_string(str)
+       local l = lines(str)
+       return {x = tonumber(l[1]), y = tonumber(l[2]), z = tonumber(l[3])}
+end
+
 local function pos_in_list(l,pos)
        for _,p in ipairs(l) do
                if p.x==pos.x and p.y==pos.y and p.z==pos.z then return true end
@@ -47,6 +66,13 @@ local function pos_in_list(l,pos)
        return false
 end
 
+local function table_empty(table)
+       for _, __ in pairs(table) do
+               return false
+       end
+       return true
+end
+
 local function add_table(table,toadd)
        local i=1
        while true do
@@ -280,29 +306,56 @@ minetest.register_node("technic:frame_motor",{
 
 
 -- Templates
-local function template_connected(pos,c)
+local function template_connected(pos,c,connectors)
        for _,vect in ipairs({{x=0,y=1,z=0},{x=0,y=0,z=1},{x=0,y=0,z=-1},{x=1,y=0,z=0},{x=-1,y=0,z=0},{x=0,y=-1,z=0}}) do
                local pos1=vector.add(pos,vect)
                local nodename=minetest.get_node(pos1).name
-               if not(pos_in_list(c,pos1)) and nodename=="technic:template" then
+               if not(pos_in_list(c,pos1)) and (nodename=="technic:template" or nodename == "technic:template_connector")then
                        local meta = minetest.get_meta(pos1)
                        if meta:get_string("connected") == "" then
                                c[#(c)+1]=pos1
-                               template_connected(pos1,c)
+                               template_connected(pos1,c,connectors)
+                               if nodename == "technic:template_connector" then
+                                       connectors[#connectors+1] = pos1
+                               end
                        end
                end
        end
 end
 
 local function get_templates(pos)
-       c={pos}
-       template_connected(pos,c)
-       return c
+       local c = {pos}
+       local connectors
+       if minetest.get_node(pos).name == "technic:template_connector" then
+               connectors = {pos}
+       else
+               connectors = {}
+       end
+       template_connected(pos,c,connectors)
+       return c, connectors
+end
+
+local function swap_template(pos, new)
+       local meta = minetest.get_meta(pos)
+       local saved_node = meta:get_string("saved_node")
+       meta:set_string("saved_node", "")
+       hacky_swap_node(pos, new)
+       local meta = minetest.get_meta(pos)
+       meta:set_string("saved_node", saved_node)
 end
 
 local function save_node(pos)
        local node = minetest.get_node(pos)
-       if node.name == "air" or node.name == "technic:template" then return end
+       if node.name == "air" then
+               minetest.set_node(pos, {name="technic:template"})
+               return
+       end
+       if node.name == "technic:template" then
+               swap_template(pos, "technic:template_connector")
+               local meta = minetest.get_meta(pos)
+               meta:set_string("connected", "")
+               return
+       end
        local meta = minetest.get_meta(pos)
        local meta0 = meta:to_table()
        for _, list in pairs(meta0.inventory) do
@@ -311,6 +364,7 @@ local function save_node(pos)
                end
        end
        node.meta = meta0
+       minetest.set_node(pos, {name="technic:template"})
        return node
 end
 
@@ -329,58 +383,108 @@ local function expand_template(pos)
        local meta = minetest.get_meta(pos)
        local c = meta:get_string("connected")
        if c == "" then return end
-       meta:set_string("connected", "")
        c = minetest.deserialize(c)
        for _, vect in ipairs(c) do
-               if vect.x ~= 0 or vect.y ~= 0 or vect.z ~= 0 then
-                       local pos1 = vector.add(pos, vect)
-                       local saved_node = save_node(pos1)
-                       minetest.set_node(pos1, {name="technic:template"})
-                       local meta1 = minetest.get_meta(pos1)
-                       if saved_node ~= nil then
-                               meta1:set_string("saved_node", minetest.serialize(saved_node))
-                       else
-                               meta1:set_string("saved_node", "")
+               local pos1 = vector.add(pos, vect)
+               local saved_node = save_node(pos1)
+               local meta1 = minetest.get_meta(pos1)
+               if saved_node ~= nil then
+                       meta1:set_string("saved_node", minetest.serialize(saved_node))
+               else
+                       --meta1:set_string("saved_node", "")
+               end
+       end
+end
+
+local function compress_templates(pos)
+       local templates, connectors = get_templates(pos)
+       if #connectors == 0 then
+               connectors = {pos}
+       end
+       for _, cn in ipairs(connectors) do
+               local meta = minetest.get_meta(cn)
+               local c = {}
+               for _,p in ipairs(templates) do
+                       local np = vector.subtract(p, cn)
+                       if not pos_in_list(c,np) then
+                               c[#c+1] = np
+                       end
+               end
+               local cc = {}
+               for _,p in ipairs(connectors) do
+                       local np = vector.subtract(p, cn)
+                       if (np.x ~= 0 or np.y ~= 0 or np.z ~= 0) then
+                               cc[pos_to_string(np)] = true
                        end
                end
+               swap_template(cn, "technic:template")
+               meta:set_string("connected", minetest.serialize(c))
+               meta:set_string("connectors_connected", minetest.serialize(cc))
+       end
+       
+       for _,p in ipairs(templates) do
+               if not pos_in_list(connectors, p) then
+                       minetest.set_node(p, {name = "air"})
+               end
        end
 end
 
 local function template_drops(pos, node, oldmeta, digger)
        local c = oldmeta.fields.connected
+       local cc = oldmeta.fields.connectors_connected
        local drops
        if c == "" or c == nil then
                drops = {"technic:template 1"}
        else
-               local stack_max = 99
-               local num = #(minetest.deserialize(c))
-               drops = {}
-               while num > stack_max do
-                       drops[#drops+1] = "technic:template "..stack_max
-                       num = num - stack_max
+               if cc == "" or cc == nil then
+                       drops = {"technic:template 1"}
+               else
+                       local dcc = minetest.deserialize(cc)
+                       if not table_empty(dcc) then
+                               drops = {}
+                               for sp, _ in pairs(dcc) do
+                                       local ssp = pos_from_string(sp)
+                                       local p = vector.add(ssp, pos)
+                                       local meta = minetest.get_meta(p)
+                                       local d = minetest.deserialize(meta:get_string("connectors_connected"))
+                                       if d ~= nil then
+                                               d[pos_to_string({x=-ssp.x, y=-ssp.y, z=-ssp.z})] = nil
+                                               meta:set_string("connectors_connected", minetest.serialize(d))
+                                       end
+                               end
+                       else
+                               local stack_max = 99
+                               local num = #(minetest.deserialize(c))
+                               drops = {}
+                               while num > stack_max do
+                                       drops[#drops+1] = "technic:template "..stack_max
+                                       num = num - stack_max
+                               end
+                               drops[#drops+1] = "technic:template "..num
+                       end
                end
-               drops[#drops+1] = "technic:template "..num
        end
-       print(dump(drops))
        minetest.handle_node_drops(pos, drops, digger)
 end
 
+local function template_on_destruct(pos, node)
+       local meta = minetest.get_meta(pos)
+       local saved_node = meta:get_string("saved_node")
+       if saved_node ~= "" then
+               local nnode = minetest.deserialize(saved_node)
+               minetest.after(0, restore_node, pos, nnode)
+       end
+end
+
 minetest.register_node("technic:template",{
        description = "Template",
        tiles = {"technic_mv_cable.png"},
        drop = "",
        groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2},
-       on_destruct = function(pos, node)
-               local meta = minetest.get_meta(pos)
-               local saved_node = meta:get_string("saved_node")
-               if saved_node ~= "" then
-                       local nnode = minetest.deserialize(saved_node)
-                       minetest.after(0, restore_node, pos, nnode)
-               end
-       end,
+       on_destruct = template_on_destruct,
        after_dig_node = template_drops,
        on_punch = function(pos,node,puncher)
-               hacky_swap_node(pos, "technic:template_disabled")
+               swap_template(pos, "technic:template_disabled")
        end
 })
 
@@ -388,18 +492,24 @@ minetest.register_node("technic:template_disabled",{
        description = "Template",
        tiles = {"technic_hv_cable.png"},
        drop = "",
-       groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2},
-       on_destruct = function(pos, node)
-               local meta = minetest.get_meta(pos)
-               local saved_node = meta:get_string("saved_node")
-               if saved_node ~= "" then
-                       local nnode = minetest.deserialize(saved_node)
-                       minetest.after(0, restore_node, pos, nnode)
-               end
-       end,
+       groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2,not_in_creative_inventory=1},
+       on_destruct = template_on_destruct,
        after_dig_node = template_drops,
        on_punch = function(pos,node,puncher)
-               hacky_swap_node(pos, "technic:template")
+       local meta = minetest.get_meta(pos)
+               swap_template(pos, "technic:template_connector")
+       end
+})
+
+minetest.register_node("technic:template_connector",{
+       description = "Template",
+       tiles = {"technic_lv_cable.png"},
+       drop = "",
+       groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2,not_in_creative_inventory=1},
+       on_destruct = template_on_destruct,
+       after_dig_node = template_drops,
+       on_punch = function(pos,node,puncher)
+               swap_template(pos, "technic:template")
        end
 })
 
@@ -414,7 +524,6 @@ minetest.register_craftitem("technic:template_replacer",{
                local node = minetest.get_node(p)
                if node.name == "technic:template" then return end
                local saved_node = save_node(p)
-               minetest.set_node(p, {name="technic:template"})
                itemstack:take_item()
                if saved_node ~= nil then
                        local meta = minetest.get_meta(p)
@@ -433,25 +542,15 @@ minetest.register_tool("technic:template_tool",{
                        return nil
                end
                local node = minetest.get_node(pos)
-               if node.name ~= "technic:template" then return end
+               if node.name ~= "technic:template" and node.name ~= "technic:template_connector" then return end
                local meta = minetest.get_meta(pos)
                local c2 = meta:get_string("connected")
                if c2 ~= "" then
                        expand_template(pos)
-                       return
+               else
+                       compress_templates(pos)
                end
-               local templates = get_templates(pos)
-               local c = {}
-               for _,p in ipairs(templates) do
-                       local np = {x=p.x-pos.x, y=p.y-pos.y, z=p.z-pos.z}
-                       if not pos_in_list(c,np) then
-                               c[#c+1] = np
-                       end
-                       if p.x ~= pos.x or p.y ~= pos.y or p.z ~= pos.z then
-                               minetest.set_node(p, {name = "air"})
-                       end
-               end
-               meta:set_string("connected", minetest.serialize(c))
+               
        end
 })