From 1a248ff49cba8c31e65550a6f1b87c0e2be88e0d Mon Sep 17 00:00:00 2001 From: Joachim Stolberg Date: Fri, 8 Nov 2019 21:10:05 +0100 Subject: [PATCH] bugfixes for power terminal and tank/pump lines distiller protopype added --- basis/mark2.lua | 4 +- basis/networks.lua | 35 ++- chemistry/ta4_reactor.lua | 2 +- coal_power_station/boiler_base.lua | 2 +- coal_power_station/boiler_top.lua | 2 +- coal_power_station/firebox.lua | 4 +- coal_power_station/power_terminal.lua | 10 +- doc/manual_DE.lua | 10 + energy_storage/inlet.lua | 1 + init.lua | 3 +- liquids/liquid.lua | 14 +- liquids/liquid_pipe.lua | 1 + liquids/pump.lua | 32 ++- liquids/tank.lua | 6 +- manuals/manual_ta3_DE.md | 13 + models/size.py | 9 + ...age_pillar.obj => techage_cylinder_07.obj} | 0 ...age_boiler.obj => techage_cylinder_09.obj} | 0 ...iler_large.obj => techage_cylinder_12.obj} | 0 ...iler_huge.obj => techage_cylinder_12h.obj} | 0 models/techage_cylinder_14.obj | 124 ++++++++++ nodes/pillar.lua | 2 +- oil/distiller.lua | 226 ++++++++++++++++++ oil/reboiler.lua | 190 +++++++++++++++ power/ta4_junction.lua | 77 ------ steam_engine/boiler.lua | 4 +- textures/techage_appl_arrowXL.png | Bin 0 -> 175 bytes textures/techage_appl_reboiler.png | Bin 0 -> 204 bytes textures/techage_appl_reboiler4.png | Bin 0 -> 324 bytes textures/techage_distiller1.png | Bin 0 -> 3911 bytes textures/techage_distiller2.png | Bin 0 -> 3431 bytes textures/techage_distiller3.png | Bin 0 -> 3488 bytes textures/techage_distiller4.png | Bin 0 -> 4064 bytes 33 files changed, 656 insertions(+), 115 deletions(-) create mode 100644 models/size.py rename models/{techage_pillar.obj => techage_cylinder_07.obj} (100%) rename models/{techage_boiler.obj => techage_cylinder_09.obj} (100%) rename models/{techage_boiler_large.obj => techage_cylinder_12.obj} (100%) rename models/{techage_boiler_huge.obj => techage_cylinder_12h.obj} (100%) create mode 100644 models/techage_cylinder_14.obj create mode 100644 oil/distiller.lua create mode 100644 oil/reboiler.lua delete mode 100644 power/ta4_junction.lua create mode 100644 textures/techage_appl_arrowXL.png create mode 100644 textures/techage_appl_reboiler.png create mode 100644 textures/techage_appl_reboiler4.png create mode 100644 textures/techage_distiller1.png create mode 100644 textures/techage_distiller2.png create mode 100644 textures/techage_distiller3.png create mode 100644 textures/techage_distiller4.png diff --git a/basis/mark2.lua b/basis/mark2.lua index 19cda66..6902c14 100644 --- a/basis/mark2.lua +++ b/basis/mark2.lua @@ -50,8 +50,8 @@ minetest.register_entity(":techage:position_cube", { }, --use_texture_alpha = true, physical = false, - visual_size = {x = 1.2, y = 1.2}, - collisionbox = {-0.6,-0.6,-0.6, 0.6,0.6,0.6}, + visual_size = {x = 1.1, y = 1.1}, + collisionbox = {-0.55,-0.55,-0.55, 0.55,0.55,0.55}, glow = 8, }, on_step = function(self, dtime) diff --git a/basis/networks.lua b/basis/networks.lua index 492fa80..2de2939 100644 --- a/basis/networks.lua +++ b/basis/networks.lua @@ -64,6 +64,16 @@ local function net_def2(node_name, net_name) return ndef and ndef.networks and ndef.networks[net_name] or {} end +local function connected(tlib2, pos, dir) + local param2, npos = tlib2:get_primary_node_param2(pos, dir) + if param2 then + local d1, d2, num = tlib2:decode_param2(npos, param2) + if not num then return end + return Flip[dir] == d1 or Flip[dir] == d2 + end + return false +end + -- Calculate the node outdir based on node.param2 and nominal dir (according to side) local function dir_to_outdir(dir, param2) if dir < 5 then @@ -115,7 +125,7 @@ local function node_connections(pos, tlib2) val = val * 2 local side = DirToSide[outdir_to_dir(dir, node.param2)] if sides[side] then - if tlib2:connected(pos, dir) then + if connected(tlib2, pos, dir) then val = val + 1 end end @@ -139,7 +149,7 @@ end -- check if the given pipe dir into the node is valid local function valid_indir(indir, node, net_name) local ndef = net_def2(node.name, net_name) - if not ndef or not ndef.sides or ndef.blocker then return false end + if not ndef or not ndef.sides then return false end local side = DirToSide[indir_to_dir(indir, node.param2)] if not ndef.sides[side] then return false end return true @@ -148,14 +158,17 @@ end -- do the walk through the tubelib2 network -- indir is the direction which should not be covered by the walk -- (coming from there or is a different network) -local function connection_walk(pos, indir, node, tlib2, clbk) +local function connection_walk(pos, indir, node, tlib2, lvl, clbk) if clbk then clbk(pos, indir, node) end - for _,outdir in pairs(get_node_connections(pos, tlib2.tube_type)) do - if outdir ~= Flip[indir] then - local pos2, indir2 = tlib2:get_connected_node_pos(pos, outdir) - local node = techage.get_node_lvm(pos2) - if pos2 and not pos_already_reached(pos2) and valid_indir(indir2, node, tlib2.tube_type) then - connection_walk(pos2, indir2, node, tlib2, clbk) + --techage.mark_position("singleplayer", pos, "walk", "", 1) + if lvl == 1 or net_def2(node.name, tlib2.tube_type).ntype == "junc" then + for _,outdir in pairs(get_node_connections(pos, tlib2.tube_type)) do + if outdir ~= Flip[indir] then + local pos2, indir2 = tlib2:get_connected_node_pos(pos, outdir) + local node = techage.get_node_lvm(pos2) + if pos2 and not pos_already_reached(pos2) and valid_indir(indir2, node, tlib2.tube_type) then + connection_walk(pos2, indir2, node, tlib2, lvl + 1, clbk) + end end end end @@ -170,7 +183,7 @@ local function collect_network_nodes(pos, outdir, tlib2) local node = techage.get_node_lvm(pos) local net_name = tlib2.tube_type -- outdir corresponds to the indir coming from - connection_walk(pos, outdir, node, tlib2, function(pos, indir, node) + connection_walk(pos, outdir, node, tlib2, 1, function(pos, indir, node) local ntype = net_def2(node.name, net_name).ntype if ntype then if not netw[ntype] then netw[ntype] = {} end @@ -233,7 +246,7 @@ function techage.networks.connection_walk(pos, outdir, tlib2, clbk) NumNodes = 0 pos_already_reached(pos) -- don't consider the start pos local node = techage.get_node_lvm(pos) - connection_walk(pos, outdir, node, tlib2, clbk) + connection_walk(pos, outdir, node, tlib2, 1, clbk) return NumNodes end diff --git a/chemistry/ta4_reactor.lua b/chemistry/ta4_reactor.lua index af11433..4c15f2c 100644 --- a/chemistry/ta4_reactor.lua +++ b/chemistry/ta4_reactor.lua @@ -30,7 +30,7 @@ minetest.register_node("techage:ta4_reactor", { description = S("TA4 Reactor"), tiles = {"techage_reactor_side.png"}, drawtype = "mesh", - mesh = "techage_boiler_huge.obj", + mesh = "techage_cylinder_12h.obj", selection_box = { type = "fixed", fixed = {-1/2, -23/32, -1/2, 1/2, 32/32, 1/2}, diff --git a/coal_power_station/boiler_base.lua b/coal_power_station/boiler_base.lua index ec68701..fc4d9cc 100644 --- a/coal_power_station/boiler_base.lua +++ b/coal_power_station/boiler_base.lua @@ -22,7 +22,7 @@ minetest.register_node("techage:coalboiler_base", { description = S("TA3 Boiler Base"), tiles = {"techage_coal_boiler_mesh_base.png"}, drawtype = "mesh", - mesh = "techage_boiler_large.obj", + mesh = "techage_cylinder_12.obj", selection_box = { type = "fixed", fixed = {-13/32, -16/32, -13/32, 13/32, 16/32, 13/32}, diff --git a/coal_power_station/boiler_top.lua b/coal_power_station/boiler_top.lua index a044ac4..6a3c1e5 100644 --- a/coal_power_station/boiler_top.lua +++ b/coal_power_station/boiler_top.lua @@ -223,7 +223,7 @@ minetest.register_node("techage:coalboiler_top", { description = S("TA3 Boiler Top"), tiles = {"techage_coal_boiler_mesh_top.png"}, drawtype = "mesh", - mesh = "techage_boiler_large.obj", + mesh = "techage_cylinder_12.obj", selection_box = { type = "fixed", fixed = {-13/32, -48/32, -13/32, 13/32, 16/32, 13/32}, diff --git a/coal_power_station/firebox.lua b/coal_power_station/firebox.lua index a663efa..b583d4d 100644 --- a/coal_power_station/firebox.lua +++ b/coal_power_station/firebox.lua @@ -81,7 +81,7 @@ minetest.register_node("techage:coalfirebox", { inventory_image = "techage_coal_boiler_inv.png", tiles = {"techage_coal_boiler_mesh_top.png"}, drawtype = "mesh", - mesh = "techage_boiler_large.obj", + mesh = "techage_cylinder_12.obj", selection_box = { type = "fixed", fixed = {-13/32, -16/32, -13/32, 13/32, 16/32, 13/32}, @@ -235,7 +235,7 @@ minetest.register_node("techage:oilfirebox", { inventory_image = "techage_oil_boiler_inv.png", tiles = {"techage_coal_boiler_mesh_top.png"}, drawtype = "mesh", - mesh = "techage_boiler_large.obj", + mesh = "techage_cylinder_12.obj", selection_box = { type = "fixed", fixed = {-13/32, -16/32, -13/32, 13/32, 16/32, 13/32}, diff --git a/coal_power_station/power_terminal.lua b/coal_power_station/power_terminal.lua index 421fa59..a4df465 100644 --- a/coal_power_station/power_terminal.lua +++ b/coal_power_station/power_terminal.lua @@ -13,7 +13,7 @@ ]]-- -- for lazy programmers -local P2P = minetest.string_to_pos +local S2P = minetest.string_to_pos local P2S = function(pos) if pos then return minetest.pos_to_string(pos) end end local M = minetest.get_meta local N = function(pos) return minetest.get_node(pos).name end @@ -62,8 +62,10 @@ local function collect_network_data(pos, mem) add(data.fcel, mem, mem.pwr_available, mem.provided) elseif node.name == "techage:ta4_electrolyzer" or node.name == "techage:ta4_electrolyzer_on" then add(data.elec, mem, -(mem.pwr_could_need or 0), -(mem.consumed or 0)) - elseif mem.pwr_needed and mem.pwr_needed > 0 and (mem.pwr_node_alive_cnt or 0) > 0 then - add(data.other, mem, -(mem.pwr_needed or 0), (-mem.pwr_needed or 0)) + elseif mem.pwr_needed and mem.pwr_needed > 0 then + local nomi = -mem.pwr_needed + local real = (mem.pwr_node_alive_cnt or 0) > 0 and -mem.pwr_needed + add(data.other, mem, nomi, real) end end ) @@ -74,7 +76,7 @@ local function formspec(pos, mem) local data, nnodes = collect_network_data(pos, mem) local get = function(kind) return (data[kind].load or 0).." / "..(data[kind].num or 0).." : ".. - (data[kind].curr or 0).." / "..(data[kind].nomi or 0).. " ku" + (data[kind].real or 0).." / "..(data[kind].nomi or 0).. " ku" end local alarm = "" diff --git a/doc/manual_DE.lua b/doc/manual_DE.lua index 7c8ff24..c4553b5 100644 --- a/doc/manual_DE.lua +++ b/doc/manual_DE.lua @@ -416,6 +416,8 @@ techage.manual_DE.aText = { "Teil des Kraftwerks. \n".. "Der Ölbrenner kann nur mit Öl gefüllt werden. Die Brenndauer ist abhängig von der Leistung\\, die vom Kraftwerk angefordert wird. Unter Volllast brennt Öl 20 s. Unter Teillast entsprechend länger (50% Last = doppelte Zeit).\n".. "\n".. + "Der Ölbrenner kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam.\n".. + "\n".. "\n".. "\n", "Teil des Kraftwerk. Muss mit Wasser gefüllt werden. Wem kein Wasser mehr vorhanden ist oder die Temperatur zu weit absinkt\\, schaltet sich der Boiler ab.\n".. @@ -497,6 +499,8 @@ techage.manual_DE.aText = { "\n", "Der kleine Stromgenerator wird mit Erdöl betrieben und kann für kleine Verbraucher mit bis zu 12 ku genutzt werden. Unter Volllast brennt Erdöl 100 s. Unter Teillast entsprechend länger (50% Last = doppelte Zeit).\n".. "\n".. + "Der Stromgenerator kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam.\n".. + "\n".. "\n".. "\n", "Der Akku Block dient zur Speicherung von überschüssiger Energie und gibt bei Stromausfall automatisch Strom ab (soweit vorhanden).\n".. @@ -524,6 +528,8 @@ techage.manual_DE.aText = { "\n", "Ist Teil des TA3 Industrieofen. Muss mit Öl befeuert werden.\n".. "\n".. + "Der Ölbrenner kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam.\n".. + "\n".. "\n".. "\n", "Ist Teil des TA3 Industrieofen. Siehe TA3 Industrieofen.\n".. @@ -790,6 +796,10 @@ techage.manual_DE.aText = { "\n".. "Die TA3 Pumpe pumpt 4 Einheiten Flüssigkeit alle zwei Sekunden.\n".. "\n".. + "Hinweis 1: Die Pumpe darf nicht direkt neben den Tank platziert werden. Es muss immer mindestens ein Stück gelbe Leitung dazwischen sein.\n".. + "\n".. + "Hinweis 2: Nach dem Starten markiert die Pumpe 10 x die Blöcke\\, von und zu denen gepumpt wird.\n".. + "\n".. "\n".. "\n", "", diff --git a/energy_storage/inlet.lua b/energy_storage/inlet.lua index b0d04c1..f8cbbff 100644 --- a/energy_storage/inlet.lua +++ b/energy_storage/inlet.lua @@ -75,6 +75,7 @@ local function get_radius(pos, in_dir) return radius end end + return 1 end local function check_volume(pos, in_dir, owner) diff --git a/init.lua b/init.lua index eee810f..8c3aef4 100644 --- a/init.lua +++ b/init.lua @@ -164,6 +164,8 @@ else dofile(MP.."/oil/drillbox.lua") dofile(MP.."/oil/pumpjack.lua") dofile(MP.."/oil/generator.lua") + --dofile(MP.."/oil/distiller.lua") + --dofile(MP.."/oil/reboiler.lua") -- Nodes2 if techage.basalt_stone_enabled then @@ -216,7 +218,6 @@ else -- Chemistry --dofile(MP.."/chemistry/ta4_reactor.lua") --dofile(MP.."/chemistry/ta4_doser.lua") - --dofile(MP.."/chemistry/ta4_distiller.lua") -- Hydrogen dofile(MP.."/hydrogen/hydrogen.lua") diff --git a/liquids/liquid.lua b/liquids/liquid.lua index 4c48551..742522a 100644 --- a/liquids/liquid.lua +++ b/liquids/liquid.lua @@ -111,14 +111,17 @@ end -- Add given amount of liquid to the remote inventory. -- return leftover amount -function techage.liquid.put(pos, outdir, name, amount) +function techage.liquid.put(pos, outdir, name, amount, player_name) for _,item in ipairs(get_network_table(pos, outdir, "tank")) do local liquid = LQD(item.pos) if liquid and liquid.put and liquid.peek then -- wrong items? local peek = liquid.peek(item.pos, item.indir) if peek and peek ~= name then return amount or 0 end - --techage.mark_position("singleplayer", item.pos, "put", "", 1) ------------------- debug + if player_name then + local num = techage.get_node_number(pos) + techage.mark_position(player_name, item.pos, "("..num..") put", "", 1) + end amount = liquid.put(item.pos, item.indir, name, amount) if not amount or amount == 0 then break end end @@ -128,13 +131,16 @@ end -- Take given amount of liquid for the remote inventory. -- return taken amount and item name -function techage.liquid.take(pos, outdir, name, amount) +function techage.liquid.take(pos, outdir, name, amount, player_name) local taken = 0 local item_name = nil for _,item in ipairs(get_network_table(pos, outdir, "tank")) do local liquid = LQD(item.pos) if liquid and liquid.take then - --techage.mark_position("singleplayer", item.pos, "take", "", 1) ------------------- debug + if player_name then + local num = techage.get_node_number(pos) + techage.mark_position(player_name, item.pos, "("..num..") take", "", 1) + end local val, name = liquid.take(item.pos, item.indir, name, amount - taken) if val and name then taken = taken + val diff --git a/liquids/liquid_pipe.lua b/liquids/liquid_pipe.lua index d8cc143..3327108 100644 --- a/liquids/liquid_pipe.lua +++ b/liquids/liquid_pipe.lua @@ -25,6 +25,7 @@ local Pipe = tubelib2.Tube:new({ dirs_to_check = {1,2,3,4,5,6}, max_tube_length = MAX_PIPE_LENGHT, show_infotext = false, + force_to_use_tubes = true, tube_type = "pipe", primary_node_names = {"techage:ta3_pipeS", "techage:ta3_pipeA"}, secondary_node_names = {}, diff --git a/liquids/pump.lua b/liquids/pump.lua index d2b2472..911048a 100644 --- a/liquids/pump.lua +++ b/liquids/pump.lua @@ -21,11 +21,31 @@ local networks = techage.networks local liquid = techage.liquid local Flip = techage.networks.Flip -local STANDBY_TICKS = 10 -local COUNTDOWN_TICKS = 10 +local STANDBY_TICKS = 5 +local COUNTDOWN_TICKS = 5 local CYCLE_TIME = 2 local CAPA = 4 +-- to mark the pump source and destinstion node +local DebugCache = {} + +local function set_starter_name(pos, clicker) + local key = minetest.hash_node_position(pos) + DebugCache[key] = {starter = clicker:get_player_name(), count = 10} +end + +local function get_starter_name(pos) + local key = minetest.hash_node_position(pos) + local def = DebugCache[key] + if def then + def.count = (def.count or 0) - 1 + if def.count > 0 then + return def.starter + end + DebugCache[key] = nil + end +end + local State3 = techage.NodeStates:new({ node_name_passive = "techage:t3_pump", node_name_active = "techage:t3_pump_on", @@ -44,9 +64,10 @@ local State4 = techage.NodeStates:new({ local function pumping(pos, mem, state, capa) local outdir = M(pos):get_int("outdir") - local taken, name = liquid.take(pos, Flip[outdir], nil, capa) + local starter = get_starter_name(pos) + local taken, name = liquid.take(pos, Flip[outdir], nil, capa, starter) if taken > 0 then - local leftover = liquid.put(pos, outdir, name, taken) + local leftover = liquid.put(pos, outdir, name, taken, starter) if leftover and leftover > 0 then liquid.put(pos, Flip[outdir], name, leftover) state:blocked(pos, mem) @@ -89,10 +110,12 @@ end local function on_rightclick(pos, node, clicker) local mem = tubelib2.get_mem(pos) if node.name == "techage:t3_pump" then + set_starter_name(pos, clicker) State3:start(pos, mem) elseif node.name == "techage:t3_pump_on" then State3:stop(pos, mem) elseif node.name == "techage:t4_pump" then + set_starter_name(pos, clicker) State4:start(pos, mem) elseif node.name == "techage:t4_pump_on" then State4:stop(pos, mem) @@ -187,7 +210,6 @@ local nworks = { pipe = { sides = {L = 1, R = 1}, -- Pipe connection side ntype = "pump", - blocker = true, -- for the connection_walk }, } diff --git a/liquids/tank.lua b/liquids/tank.lua index e92ff45..6dfc280 100644 --- a/liquids/tank.lua +++ b/liquids/tank.lua @@ -238,15 +238,15 @@ minetest.register_node("techage:oiltank", { paramtype = "light", node_box = { type = "fixed", - fixed = {-6/8, -6/8, -6/8, 6/8, 6/8, 6/8}, + fixed = {-6/8, -4/8, -6/8, 6/8, 6/8, 6/8}, }, selection_box = { type = "fixed", - fixed = {-6/8, -6/8, -6/8, 6/8, 6/8, 6/8}, + fixed = {-6/8, -4/8, -6/8, 6/8, 6/8, 6/8}, }, collision_box = { type = "fixed", - fixed = {-6/8, -6/8, -6/8, 6/8, 6/8, 6/8}, + fixed = {-6/8, -4/8, -6/8, 6/8, 6/8, 6/8}, }, on_construct = function(pos) local meta = minetest.get_meta(pos) diff --git a/manuals/manual_ta3_DE.md b/manuals/manual_ta3_DE.md index 103b61d..2ae0665 100644 --- a/manuals/manual_ta3_DE.md +++ b/manuals/manual_ta3_DE.md @@ -38,6 +38,9 @@ Die Feuerbox muss mit Kohle, Holzkohle oder Erdöl gefüllt werden. Die Brenndau Teil des Kraftwerks. Der Ölbrenner kann nur mit Öl gefüllt werden. Die Brenndauer ist abhängig von der Leistung, die vom Kraftwerk angefordert wird. Unter Volllast brennt Öl 20 s. Unter Teillast entsprechend länger (50% Last = doppelte Zeit). +Der Ölbrenner kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam. + + [ta3_oilbox|image] @@ -162,6 +165,9 @@ siehe TA Stromschalter. Der kleine Stromgenerator wird mit Erdöl betrieben und kann für kleine Verbraucher mit bis zu 12 ku genutzt werden. Unter Volllast brennt Erdöl 100 s. Unter Teillast entsprechend länger (50% Last = doppelte Zeit). +Der Stromgenerator kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam. + + [ta3_tinygenerator|image] @@ -194,6 +200,7 @@ Der Industrieofen benötigt Strom (für das Gebläse) sowie Öl für den Brenner Die Brennzeit für Öl beträgt 80 s. + [ta3_furnace|plan] @@ -201,6 +208,8 @@ Die Brennzeit für Öl beträgt 80 s. Ist Teil des TA3 Industrieofen. Muss mit Öl befeuert werden. +Der Ölbrenner kann nur 50 Einheiten Öl aufnehmen. Ein zusätzlicher Öltank und eine Ölpumpe sind daher ratsam. + [ta3_furnacefirebox|image] @@ -568,6 +577,10 @@ Mit der Pumpe können Flüssigkeiten von Tanks oder Behältern zu anderen Tanks Die TA3 Pumpe pumpt 4 Einheiten Flüssigkeit alle zwei Sekunden. +Hinweis 1: Die Pumpe darf nicht direkt neben den Tank platziert werden. Es muss immer mindestens ein Stück gelbe Leitung dazwischen sein. + +Hinweis 2: Nach dem Starten markiert die Pumpe 10 x die Blöcke, von und zu denen gepumpt wird. + [ta3_pump|image] diff --git a/models/size.py b/models/size.py new file mode 100644 index 0000000..620294c --- /dev/null +++ b/models/size.py @@ -0,0 +1,9 @@ +lOut = [] +for line in file("techage_boiler_large.obj"): + words = line.split(" ") + if words[0] == "v": + words[1] = "%1.6f" % (float(words[1]) * 1.2) + words[3] = "%1.6f" % (float(words[3]) * 1.2) + line = " ".join(words) + lOut.append(line.strip()) +file("techage_boiler_bigger.obj", "wt").write("\n".join(lOut)) diff --git a/models/techage_pillar.obj b/models/techage_cylinder_07.obj similarity index 100% rename from models/techage_pillar.obj rename to models/techage_cylinder_07.obj diff --git a/models/techage_boiler.obj b/models/techage_cylinder_09.obj similarity index 100% rename from models/techage_boiler.obj rename to models/techage_cylinder_09.obj diff --git a/models/techage_boiler_large.obj b/models/techage_cylinder_12.obj similarity index 100% rename from models/techage_boiler_large.obj rename to models/techage_cylinder_12.obj diff --git a/models/techage_boiler_huge.obj b/models/techage_cylinder_12h.obj similarity index 100% rename from models/techage_boiler_huge.obj rename to models/techage_cylinder_12h.obj diff --git a/models/techage_cylinder_14.obj b/models/techage_cylinder_14.obj new file mode 100644 index 0000000..85c47e8 --- /dev/null +++ b/models/techage_cylinder_14.obj @@ -0,0 +1,124 @@ +# Blender v2.78 (sub 0) OBJ File: 'oven.blend' +# www.blender.org +o Cylinder +v 0.000000 -0.500000 -0.720000 +v 0.000000 0.500000 -0.720000 +v 0.275532 -0.500000 -0.665194 +v 0.275532 0.500000 -0.665194 +v 0.509117 -0.500000 -0.509117 +v 0.509117 0.500000 -0.509117 +v 0.665194 -0.500000 -0.275532 +v 0.665194 0.500000 -0.275532 +v 0.720000 -0.500000 0.000000 +v 0.720000 0.500000 0.000000 +v 0.665194 -0.500000 0.275532 +v 0.665194 0.500000 0.275532 +v 0.509117 -0.500000 0.509117 +v 0.509117 0.500000 0.509117 +v 0.275532 -0.500000 0.665194 +v 0.275532 0.500000 0.665194 +v 0.000000 -0.500000 0.720000 +v 0.000000 0.500000 0.720000 +v -0.275532 -0.500000 0.665194 +v -0.275532 0.500000 0.665194 +v -0.509117 -0.500000 0.509117 +v -0.509117 0.500000 0.509117 +v -0.665194 -0.500000 0.275532 +v -0.665194 0.500000 0.275532 +v -0.720000 -0.500000 -0.000000 +v -0.720000 0.500000 -0.000000 +v -0.665194 -0.500000 -0.275532 +v -0.665194 0.500000 -0.275532 +v -0.509117 -0.500000 -0.509117 +v -0.509117 0.500000 -0.509117 +v -0.275532 -0.500000 -0.665194 +v -0.275532 0.500000 -0.665194 +vt 0.5486 0.5000 +vt 0.5486 1.0000 +vt 0.4725 1.0000 +vt 0.4725 0.5000 +vt 0.4142 1.0000 +vt 0.4142 0.5000 +vt 1.0000 0.5000 +vt 1.0000 1.0000 +vt 0.9239 1.0000 +vt 0.9239 0.5000 +vt 0.8415 1.0000 +vt 0.8415 0.5000 +vt 0.7654 1.0000 +vt 0.7654 0.5000 +vt 0.4142 0.5000 +vt 0.4142 0.0000 +vt 0.4725 0.0000 +vt 0.4725 0.5000 +vt 0.5486 0.0000 +vt 0.5486 0.5000 +vt 0.6310 0.0000 +vt 0.6310 0.5000 +vt 0.7071 0.0000 +vt 0.7071 0.5000 +vt 0.7654 0.0000 +vt 0.7654 0.5000 +vt 0.8415 0.0000 +vt 0.8415 0.5000 +vt 0.9239 0.0000 +vt 0.9239 0.5000 +vt 1.0000 0.0000 +vt 1.0000 0.5000 +vt 0.7654 0.5000 +vt 0.7654 1.0000 +vt 0.7071 1.0000 +vt 0.7071 0.5000 +vt 0.3244 0.4749 +vt 0.3827 0.5370 +vt 0.4142 0.6181 +vt 0.4142 0.7059 +vt 0.3827 0.7870 +vt 0.3244 0.8491 +vt 0.2483 0.8827 +vt 0.1659 0.8827 +vt 0.0898 0.8491 +vt 0.0315 0.7870 +vt 0.0000 0.7059 +vt 0.0000 0.6181 +vt 0.0315 0.5370 +vt 0.0898 0.4749 +vt 0.1659 0.4414 +vt 0.2483 0.4414 +vt 0.6310 1.0000 +vt 0.6310 0.5000 +vt 0.0000 0.2646 +vt 0.0000 0.1768 +vt 0.0315 0.0957 +vt 0.0898 0.0336 +vt 0.1659 0.0000 +vt 0.2483 0.0000 +vt 0.3244 0.0336 +vt 0.3827 0.0957 +vt 0.4142 0.1768 +vt 0.4142 0.2646 +vt 0.3827 0.3457 +vt 0.3244 0.4078 +vt 0.2483 0.4414 +vt 0.1659 0.4414 +vt 0.0898 0.4078 +vt 0.0315 0.3457 +s off +f 1/1 2/2 4/3 3/4 +f 3/4 4/3 6/5 5/6 +f 5/7 6/8 8/9 7/10 +f 7/10 8/9 10/11 9/12 +f 9/12 10/11 12/13 11/14 +f 11/15 12/16 14/17 13/18 +f 13/18 14/17 16/19 15/20 +f 15/20 16/19 18/21 17/22 +f 17/22 18/21 20/23 19/24 +f 19/24 20/23 22/25 21/26 +f 21/26 22/25 24/27 23/28 +f 23/28 24/27 26/29 25/30 +f 25/30 26/29 28/31 27/32 +f 27/33 28/34 30/35 29/36 +f 4/37 2/38 32/39 30/40 28/41 26/42 24/43 22/44 20/45 18/46 16/47 14/48 12/49 10/50 8/51 6/52 +f 29/36 30/35 32/53 31/54 +f 31/54 32/53 2/2 1/1 +f 1/55 3/56 5/57 7/58 9/59 11/60 13/61 15/62 17/63 19/64 21/65 23/66 25/67 27/68 29/69 31/70 \ No newline at end of file diff --git a/nodes/pillar.lua b/nodes/pillar.lua index 94c3082..72cca96 100644 --- a/nodes/pillar.lua +++ b/nodes/pillar.lua @@ -18,7 +18,7 @@ minetest.register_node("techage:pillar", { description = S("TA4 Pillar"), tiles = {"techage_concrete.png"}, drawtype = "mesh", - mesh = "techage_pillar.obj", + mesh = "techage_cylinder_07.obj", selection_box = { type = "fixed", fixed = {-8/32, -16/32, -8/32, 8/32, 16/32, 8/32}, diff --git a/oil/distiller.lua b/oil/distiller.lua new file mode 100644 index 0000000..cd87753 --- /dev/null +++ b/oil/distiller.lua @@ -0,0 +1,226 @@ +--[[ + + TechAge + ======= + + Copyright (C) 2019 Joachim Stolberg + + GPL v3 + See LICENSE.txt for more information + + TA3 Distillation Tower + +]]-- + +local S2P = minetest.string_to_pos +local P2S = minetest.pos_to_string +local M = minetest.get_meta +local S = techage.S +local Pipe = techage.LiquidPipe +local networks = techage.networks +local liquid = techage.liquid + + +local function orientation(pos, names) + local node = minetest.get_node({x=pos.x, y=pos.y-1, z=pos.z}) + for _,name in ipairs(names) do + if node.name == name then + local param2 = node.param2 + node = minetest.get_node(pos) + node.param2 = param2 + minetest.swap_node(pos, node) + return + end + end + minetest.remove_node(pos) + return true +end + +local function after_place_node(pos, placer) + Pipe:after_place_node(pos) +end + +local function tubelib2_on_update2(pos, dir, tlib2, node) + liquid.update_network(pos) +end + +local function after_dig_node(pos, oldnode, oldmetadata, digger) + Pipe:after_dig_node(pos) +end + +minetest.register_node("techage:ta3_distiller_base", { + description = S("TA3 Distillation Tower Base"), + tiles = { + -- up, down, right, left, back, front + "techage_concrete.png^techage_appl_arrowXL.png^techage_appl_hole_pipe.png", + "techage_concrete.png", + "techage_concrete.png", + "techage_concrete.png", + "techage_concrete.png^techage_appl_hole_pipe.png", + "techage_concrete.png", + }, + drawtype = "nodebox", + paramtype = "light", + node_box = { + type = "fixed", + fixed = {-6/8, -4/8, -6/8, 6/8, 4/8, 6/8}, + }, + selection_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + collision_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + + after_place_node = after_place_node, + tubelib2_on_update2 = tubelib2_on_update2, + after_dig_node = after_dig_node, + + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +minetest.register_node("techage:ta3_distiller1", { + description = S("TA3 Distillation Tower 1"), + tiles = {"techage_distiller1.png"}, + drawtype = "mesh", + mesh = "techage_cylinder_14.obj", + selection_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + collision_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + + after_place_node = function(pos, placer) + local res = orientation(pos, {"techage:ta3_distiller_base"}) + M(pos):set_int("outdir", networks.side_to_outdir(pos, "R")) + after_place_node(pos, placer) + return res + end, + + tubelib2_on_update2 = tubelib2_on_update2, + after_dig_node = after_dig_node, + + paramtype = "light", + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +minetest.register_node("techage:ta3_distiller2", { + description = S("TA3 Distillation Tower 2"), + tiles = {"techage_distiller2.png"}, + drawtype = "mesh", + mesh = "techage_cylinder_14.obj", + selection_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + collision_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + + after_place_node = function(pos, placer) + return orientation(pos, {"techage:ta3_distiller1", "techage:ta3_distiller3"}) + end, + + paramtype = "light", + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +minetest.register_node("techage:ta3_distiller3", { + description = S("TA3 Distillation Tower 3"), + tiles = {"techage_distiller3.png"}, + drawtype = "mesh", + mesh = "techage_cylinder_14.obj", + selection_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + collision_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + + after_place_node = function(pos, placer) + local res = orientation(pos, {"techage:ta3_distiller2"}) + return res + end, + + tubelib2_on_update2 = tubelib2_on_update2, + after_dig_node = after_dig_node, + + paramtype = "light", + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +minetest.register_node("techage:ta3_distiller4", { + description = S("TA3 Distillation Tower 4"), + tiles = {"techage_distiller4.png"}, + drawtype = "mesh", + mesh = "techage_cylinder_14.obj", + selection_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + collision_box = { + type = "fixed", + fixed = {-1/2, -1/2, -1/2, 1/2, 1/2, 1/2}, + }, + + after_place_node = function(pos, placer) + local res = orientation(pos, {"techage:ta3_distiller3"}) + after_place_node(pos, placer) + return res + end, + + tubelib2_on_update2 = tubelib2_on_update2, + after_dig_node = after_dig_node, + + paramtype = "light", + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + + +Pipe:add_secondary_node_names({ + "techage:ta3_distiller_base", + "techage:ta3_distiller1", "techage:ta3_distiller2", + "techage:ta3_distiller3", "techage:ta3_distiller4", +}) + +techage.register_node({"techage:ta3_distiller1"}, { + on_recv_message = function(pos, src, topic, payload) + if topic == "put" then + local pos2 = {x = pos.x, y = pos.y + payload.height, z = pos.z} + local outdir = 6 + if payload.height ~= 7 then + outdir = M(pos):get_int("outdir") + end + techage.mark_position("singleplayer", pos2, "put", "", 1)------------------------debug + return liquid.put(pos2, outdir, payload.name, payload.amount) + end + end, +}) diff --git a/oil/reboiler.lua b/oil/reboiler.lua new file mode 100644 index 0000000..b254fde --- /dev/null +++ b/oil/reboiler.lua @@ -0,0 +1,190 @@ +--[[ + + TechAge + ======= + + Copyright (C) 2019 Joachim Stolberg + + GPL v3 + See LICENSE.txt for more information + + TA3 Oil Reboiler + +]]-- + +local S2P = minetest.string_to_pos +local P2S = minetest.pos_to_string +local M = minetest.get_meta +local S = techage.S +local Pipe = techage.LiquidPipe +local networks = techage.networks +local liquid = techage.liquid +local Flip = techage.networks.Flip + +local CYCLE_TIME = 4 +local CAPA = 12 + +local function swap_node(pos) + local node = techage.get_node_lvm(pos) + if node.name == "techage:ta3_reboiler" then + node.name = "techage:ta3_reboiler_on" + else + node.name = "techage:ta3_reboiler" + end + minetest.swap_node(pos, node) +end + +--local function pumping(pos, mem, state, capa) +-- local outdir = M(pos):get_int("outdir") +-- local taken, name = liquid.take(pos, Flip[outdir], nil, capa) +-- if taken > 0 then +-- local leftover = liquid.put(pos, outdir, name, taken) +-- if leftover and leftover > 0 then +-- liquid.put(pos, Flip[outdir], name, leftover) +-- state:blocked(pos, mem) +-- return +-- end +-- state:keep_running(pos, mem, COUNTDOWN_TICKS) +-- return +-- end +-- state:idle(pos, mem) +--end + +local function after_place_node(pos, placer) + M(pos):set_int("outdir", networks.side_to_outdir(pos, "R")) + Pipe:after_place_node(pos) +end + +local function node_timer(pos, elapsed) + local mem = tubelib2.get_mem(pos) + pumping(pos, mem, State3, CAPA) + return State3:is_active(mem) +end + +local function tubelib2_on_update2(pos, outdir, tlib2, node) + liquid.update_network(pos, outdir) +end + +local function after_dig_node(pos, oldnode, oldmetadata, digger) + Pipe:after_dig_node(pos) +end + +local _liquid = { + capa = CAPA, + peek = liquid.srv_peek, + put = function(pos, indir, name, amount) + --start timer + return liquid.srv_put(pos, indir, name, amount) + end, + take = liquid.srv_take, +} + +local _networks = { + pipe = { + sides = {L = true, R = true}, -- Pipe connection sides + ntype = "tank", + }, +} + +local function on_rightclick(pos, node, clicker) + swap_node(pos) +end + +minetest.register_node("techage:ta3_reboiler", { + description = S("TA3 Oil Reboiler"), + tiles = { + -- up, down, right, left, back, front + "techage_filling_ta3.png^techage_frame_ta3_top.png^techage_appl_arrow.png", + "techage_filling_ta3.png^techage_frame_ta3.png", + "techage_filling_ta3.png^techage_appl_hole_pipe.png^techage_frame_ta3.png", + "techage_filling_ta3.png^techage_appl_hole_pipe.png^techage_frame_ta3.png", + "techage_filling_ta3.png^techage_appl_reboiler.png^techage_frame_ta3.png^[transformFX", + "techage_filling_ta3.png^techage_appl_reboiler.png^techage_frame_ta3.png", + }, + + after_place_node = function(pos, placer) + local mem = tubelib2.init_mem(pos) + mem.liquid = {} + local meta = M(pos) + meta:set_string("infotext", S("TA3 Oil Reboiler")) + meta:set_int("outdir", networks.side_to_outdir(pos, "R")) + Pipe:after_place_node(pos) + end, + + tubelib2_on_update2 = tubelib2_on_update2, + on_timer = node_timer, + after_dig_node = after_dig_node, + on_rightclick = on_rightclick, + liquid = _liquid, + networks = _networks, + + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +minetest.register_node("techage:ta3_reboiler_on", { + description = S("TA3 Oil Reboiler"), + tiles = { + -- up, down, right, left, back, front + "techage_filling_ta3.png^techage_frame_ta3_top.png^techage_appl_arrow.png", + "techage_filling_ta3.png^techage_frame_ta3.png", + "techage_filling_ta3.png^techage_appl_hole_pipe.png^techage_frame_ta3.png", + "techage_filling_ta3.png^techage_appl_hole_pipe.png^techage_frame_ta3.png", + { + image = "techage_filling4_ta3.png^techage_appl_reboiler4.png^techage_frame4_ta3.png^[transformFX", + backface_culling = false, + animation = { + type = "vertical_frames", + aspect_w = 32, + aspect_h = 32, + length = 2.0, + }, + }, + { + image = "techage_filling4_ta3.png^techage_appl_reboiler4.png^techage_frame4_ta3.png", + backface_culling = false, + animation = { + type = "vertical_frames", + aspect_w = 32, + aspect_h = 32, + length = 2.0, + }, + }, + }, + + tubelib2_on_update2 = tubelib2_on_update2, + on_timer = node_timer, + after_dig_node = after_dig_node, + on_rightclick = on_rightclick, + liquid = _liquid, + networks = _networks, + + paramtype2 = "facedir", + on_rotate = screwdriver.disallow, + groups = {cracky=2}, + is_ground_content = false, + sounds = default.node_sound_metal_defaults(), +}) + +Pipe:add_secondary_node_names({"techage:ta3_reboiler", "techage:ta3_reboiler_on"}) + +--minetest.register_craft({ +-- output = "techage:t3_pump 2", +-- recipe = { +-- {"group:wood", "techage:iron_ingot", "group:wood"}, +-- {"techage:ta3_pipeS", "techage:usmium_nuggets", "techage:ta3_pipeS"}, +-- {"group:wood", "techage:iron_ingot", "group:wood"}, +-- }, +--}) + +--minetest.register_craft({ +-- output = "techage:t4_pump", +-- recipe = { +-- {"default:tin_ingot", "dye:blue", "default:steel_ingot"}, +-- {"", "techage:t3_pump", ""}, +-- {"", "", ""}, +-- }, +--}) diff --git a/power/ta4_junction.lua b/power/ta4_junction.lua deleted file mode 100644 index aad3c06..0000000 --- a/power/ta4_junction.lua +++ /dev/null @@ -1,77 +0,0 @@ ---[[ - - TechAge - ======= - - Copyright (C) 2019 Joachim Stolberg - - GPL v3 - See LICENSE.txt for more information - - TA4 Junction Pipes - -]]-- - --- for lazy programmers -local M = minetest.get_meta -local S = techage.S - -local Pipe = techage.BiogasPipe -local power = techage.power - -local size1 = 1/8 -local size2 = 2/8 -local size3 = 13/32 ---local size3 = 1/8 -local Boxes = { - { - {-size1, -size1, size1, size1, size1, 0.5 }, -- z+ - {-size2, -size2, size3, size2, size2, 0.5 }, -- z+ - }, - { - {-size1, -size1, -size1, 0.5, size1, size1}, -- x+ - { size3, -size2, -size2, 0.5, size2, size2}, -- x+ - }, - { - {-size1, -size1, -0.5, size1, size1, size1}, -- z- - {-size2, -size2, -0.5, size2, size2, -size3}, -- z- - }, - { - {-0.5, -size1, -size1, size1, size1, size1}, -- x- - {-0.5, -size2, -size2, -size3, size2, size2}, -- x- - }, - { - {-size1, -0.5, -size1, size1, size1, size1}, -- y- - {-size2, -0.5, -size2, size2, -size3, size2}, -- y- - }, - { - {-size1, -size1, -size1, size1, 0.5, size1}, -- y+ - {-size2, size3, -size2, size2, 0.5, size2}, -- y+ - } -} - -techage.register_junction("techage:ta4_junctionpipe", 1/8, Boxes, Pipe, { - description = S("TA4 Junction Pipe"), - tiles = {"techage_gaspipe_junction.png"}, - groups = {crumbly = 2, cracky = 2, snappy = 2, techage_trowel = 1}, - sounds = default.node_sound_metal_defaults(), - - on_construct = tubelib2.init_mem, - after_tube_update = function(node, pos, out_dir, peer_pos, peer_in_dir) - local name = "techage:ta4_junctionpipe"..techage.junction_type(pos, Pipe) - minetest.swap_node(pos, {name = name, param2 = 0}) - power.network_changed(pos, tubelib2.get_mem(pos)) - end, - is_power_available = function(pos) - return techage.power.power_accounting(pos, tubelib2.get_mem(pos)) - end, -}, 25) - -minetest.register_craft({ - output = "techage:ta4_junctionpipe25 2", - recipe = { - {"", "techage:ta4_pipeS", ""}, - {"techage:ta4_pipeS", "", "techage:ta4_pipeS"}, - {"", "techage:ta4_pipeS", ""}, - }, -}) \ No newline at end of file diff --git a/steam_engine/boiler.lua b/steam_engine/boiler.lua index 4233a55..172a56e 100644 --- a/steam_engine/boiler.lua +++ b/steam_engine/boiler.lua @@ -214,7 +214,7 @@ minetest.register_node("techage:boiler1", { description = S("TA2 Boiler Base"), tiles = {"techage_boiler.png"}, drawtype = "mesh", - mesh = "techage_boiler.obj", + mesh = "techage_cylinder_09.obj", selection_box = { type = "fixed", fixed = {-8/32, -16/32, -8/32, 8/32, 16/32, 8/32}, @@ -232,7 +232,7 @@ minetest.register_node("techage:boiler2", { description = S("TA2 Boiler Top"), tiles = {"techage_boiler2.png"}, drawtype = "mesh", - mesh = "techage_boiler.obj", + mesh = "techage_cylinder_09.obj", selection_box = { type = "fixed", fixed = {-10/32, -48/32, -10/32, 10/32, 16/32, 10/32}, diff --git a/textures/techage_appl_arrowXL.png b/textures/techage_appl_arrowXL.png new file mode 100644 index 0000000000000000000000000000000000000000..18b1cc0a69ba7a9ecd2e02a711357e18156395c6 GIT binary patch literal 175 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnL3?x0byx0z;SkfJR9T^xl_H+M9WCijWi-X*q z7}lMWc?slj7I;J!Gca%qgD@k*tT_@uLAC&&5LchfLLhTi%tBd^o{}KHUBj)WG2B L>gTe~DWM4fA`UG` literal 0 HcmV?d00001 diff --git a/textures/techage_appl_reboiler.png b/textures/techage_appl_reboiler.png new file mode 100644 index 0000000000000000000000000000000000000000..f95dd3ec8dd296ed1e93791519e3535d31626da5 GIT binary patch literal 204 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnL3?x0byx0z;SkfJR9T^xl_H+M9WCijWi-X*q z7}lMWc?slj7I;J!Gca%qgD@k*tT_@uLAC&&5LchfLLhTi%tBd^o{}KHUv&86 literal 0 HcmV?d00001 diff --git a/textures/techage_distiller1.png b/textures/techage_distiller1.png new file mode 100644 index 0000000000000000000000000000000000000000..12314906a498b95262c58b416c0b9da57c464a20 GIT binary patch literal 3911 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 NSee_KRWN)K{s&=&<+%U= literal 0 HcmV?d00001