--[[ TechAge ======= Copyright (C) 2019-2020 Joachim Stolberg AGPL v3 See LICENSE.txt for more information TA4 Doser ]]-- 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 recipes = techage.recipes local Liquids = {} -- {hash(pos) = {name = outdir},...} local STANDBY_TICKS = 2 local COUNTDOWN_TICKS = 3 local CYCLE_TIME = 10 -- 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 function formspec(self, pos, nvm) return "size[6,3.6]".. default.gui_bg.. default.gui_bg_img.. default.gui_slots.. "box[0,-0.1;5.8,0.5;#c6e8ff]".. "label[2.5,-0.1;"..minetest.colorize( "#000000", S("Doser")).."]".. recipes.formspec(0.1, 0.8, "ta4_doser", nvm).. "image_button[5,2;1,1;".. self:get_state_button_image(nvm) ..";state_button;]".. "tooltip[5,2;1,1;"..self:get_state_tooltip(nvm).."]" end local function get_liquids(pos) local hash = minetest.hash_node_position(pos) if Liquids[hash] then return Liquids[hash] end -- determine the available input liquids local tbl = {} for outdir = 1,4 do local name, num = liquid.peek(pos, outdir) if name then tbl[name] = outdir end end Liquids[hash] = tbl return Liquids[hash] end local function del_liquids(pos) local hash = minetest.hash_node_position(pos) Liquids[hash] = nil end -- if liquids are missing, update the cached liquid table local function reload_liquids(pos) local hash = minetest.hash_node_position(pos) -- determine the available input liquids local tbl = {} for outdir = 1,4 do local name, num = liquid.peek(pos, outdir) if name then tbl[name] = outdir end end Liquids[hash] = tbl return Liquids[hash] end local function reactor_cmnd(pos, cmnd, payload) return techage.transfer( pos, 6, -- outdir cmnd, -- topic payload, -- payload Pipe, -- network {"techage:ta4_reactor_fillerpipe"}) end local function can_start(pos, nvm, state) -- check reactor local res = reactor_cmnd(pos, "check") if not res then return S("reactor defect") end res = reactor_cmnd(pos, "can_start") if not res then return S("reactor defect or no power") end local recipe = recipes.get(nvm, "ta4_doser") if recipe.catalyst then res = reactor_cmnd(pos, "catalyst") if not res or res == "" then return S("catalyst missing") end if res ~= recipe.catalyst then return S("wrong catalyst") end end return true end local function start_node(pos, nvm, state) reactor_cmnd(pos, "start") del_liquids(pos) nvm.running = true end local function stop_node(pos, nvm, state) reactor_cmnd(pos, "stop") nvm.running = false end local State = techage.NodeStates:new({ node_name_passive = "techage:ta4_doser", node_name_active = "techage:ta4_doser_on", cycle_time = CYCLE_TIME, standby_ticks = STANDBY_TICKS, formspec_func = formspec, infotext_name = "TA4 Doser", can_start = can_start, start_node = start_node, stop_node = stop_node, }) local function untake(pos, taken) for _,item in pairs(taken) do liquid.untake(pos, item.outdir, item.name, item.num) end end local function dosing(pos, nvm, elapsed) -- trigger reactor (power) if not reactor_cmnd(pos, "power") then if not nvm.techage_countdown or nvm.techage_countdown < 3 then reactor_cmnd(pos, "stop") State:nopower(pos, nvm, S("reactor has no power")) return end State:idle(pos, nvm) return end -- available liquids local liquids = get_liquids(pos) local recipe = recipes.get(nvm, "ta4_doser") if not liquids or not recipe then return end -- check from time to time nvm.check_cnt = (nvm.check_cnt or 0) + 1 if nvm.check_cnt >= 4 then nvm.check_cnt = 0 local res = reactor_cmnd(pos, "check") if not res then State:fault(pos, nvm, S("reactor defect")) reactor_cmnd(pos, "stop") return end if recipe.catalyst then res = reactor_cmnd(pos, "catalyst") if not res then State:fault(pos, nvm, S("catalyst missing")) reactor_cmnd(pos, "stop") return end if res ~= recipe.catalyst then State:fault(pos, nvm, S("wrong catalyst")) reactor_cmnd(pos, "stop") return end end end -- inputs local starter = get_starter_name(pos) local taken = {} for _,item in pairs(recipe.input) do if item.name ~= "" then local outdir = liquids[item.name] or reload_liquids(pos)[item.name] if not outdir then State:standby(pos, nvm) reactor_cmnd(pos, "stop") untake(pos, taken) return end local num = liquid.take(pos, outdir, item.name, item.num, starter) if num < item.num then taken[#taken + 1] = {outdir = outdir, name = item.name, num = num} State:standby(pos, nvm) reactor_cmnd(pos, "stop") untake(pos, taken) return end taken[#taken + 1] = {outdir = outdir, name = item.name, num = item.num} end end -- waste local leftover if recipe.waste.name ~= "" then leftover = reactor_cmnd(pos, "waste", { name = recipe.waste.name, amount = recipe.waste.num}) if not leftover or (tonumber(leftover) or 1) > 0 then untake(pos, taken) State:blocked(pos, nvm) reactor_cmnd(pos, "stop") return end end -- output leftover = reactor_cmnd(pos, "output", { name = recipe.output.name, amount = recipe.output.num}) if not leftover or (tonumber(leftover) or 1) > 0 then untake(pos, taken) State:blocked(pos, nvm) reactor_cmnd(pos, "stop") return end State:keep_running(pos, nvm, COUNTDOWN_TICKS) end local function node_timer(pos, elapsed) local nvm = techage.get_nvm(pos) dosing(pos, nvm, elapsed) return State:is_active(nvm) end local function on_rightclick(pos) local nvm = techage.get_nvm(pos) M(pos):set_string("formspec", formspec(State, pos, nvm)) end local function on_receive_fields(pos, formname, fields, player) if minetest.is_protected(pos, player:get_player_name()) then return end local nvm = techage.get_nvm(pos) if not nvm.running then recipes.on_receive_fields(pos, formname, fields, player) end set_starter_name(pos, player) State:state_button_event(pos, nvm, fields) M(pos):set_string("formspec", formspec(State, pos, nvm)) end local nworks = { pipe2 = { sides = techage.networks.AllSides, -- Pipe connection sides ntype = "pump", }, } minetest.register_node("techage:ta4_doser", { description = S("TA4 Doser"), tiles = { -- up, down, right, left, back, front "techage_filling_ta4.png^techage_frame_ta4_top.png^techage_appl_hole_pipe.png", "techage_filling_ta4.png^techage_frame_ta4.png", "techage_filling_ta4.png^techage_frame_ta4.png^techage_appl_pump_up.png", }, after_place_node = function(pos, placer) local meta = M(pos) local nvm = techage.get_nvm(pos) local number = techage.add_node(pos, "techage:ta4_doser") meta:set_string("node_number", number) meta:set_string("owner", placer:get_player_name()) meta:set_string("formspec", formspec(State, pos, nvm)) meta:set_string("infotext", S("TA4 Doser").." "..number) State:node_init(pos, nvm, number) Pipe:after_place_node(pos) end, tubelib2_on_update2 = function(pos, dir, tlib2, node) liquid.update_network(pos, dir) del_liquids(pos) end, after_dig_node = function(pos, oldnode, oldmetadata, digger) techage.remove_node(pos, oldnode, oldmetadata) Pipe:after_dig_node(pos) liquid.after_dig_pump(pos) techage.del_mem(pos) end, on_receive_fields = on_receive_fields, on_rightclick = on_rightclick, on_timer = node_timer, networks = nworks, paramtype2 = "facedir", on_rotate = screwdriver.disallow, groups = {cracky=2}, is_ground_content = false, sounds = default.node_sound_metal_defaults(), }) minetest.register_node("techage:ta4_doser_on", { description = S("TA4 Doser"), tiles = { -- up, down, right, left, back, front "techage_filling_ta4.png^techage_frame_ta4_top.png^techage_appl_hole_pipe.png", "techage_filling_ta4.png^techage_frame_ta4.png", { image = "techage_filling8_ta4.png^techage_frame8_ta4.png^techage_appl_pump_up8.png", backface_culling = false, animation = { type = "vertical_frames", aspect_w = 32, aspect_h = 32, length = 2.0, }, }, }, tubelib2_on_update2 = function(pos, dir, tlib2, node) liquid.update_network(pos, dir) del_liquids(pos) end, on_receive_fields = on_receive_fields, on_rightclick = on_rightclick, on_timer = node_timer, networks = nworks, paramtype2 = "facedir", on_rotate = screwdriver.disallow, diggable = false, groups = {not_in_creative_inventory=1}, is_ground_content = false, sounds = default.node_sound_metal_defaults(), }) techage.register_node({"techage:ta4_doser", "techage:ta4_doser_on"}, { on_recv_message = function(pos, src, topic, payload) return State:on_receive_message(pos, topic, payload) end, }) Pipe:add_secondary_node_names({"techage:ta4_doser", "techage:ta4_doser_on"}) if minetest.global_exists("unified_inventory") then unified_inventory.register_craft_type("ta4_doser", { description = S("TA4 Reactor"), icon = 'techage_reactor_filler_plan.png', width = 2, height = 2, }) end minetest.register_craft({ output = "techage:ta4_doser", recipe = { {"", "techage:ta3_pipeS", ""}, {"techage:ta3_pipeS", "techage:t4_pump", "techage:ta3_pipeS"}, {"", "techage:ta4_wlanchip", ""}, }, })