299 lines
8.5 KiB
Lua
299 lines
8.5 KiB
Lua
--[[
|
|
|
|
SmartLine
|
|
=========
|
|
|
|
Copyright (C) 2017-2020 Joachim Stolberg
|
|
|
|
AGPL v3
|
|
See LICENSE.txt for more information
|
|
|
|
sequencer.lua:
|
|
Derived from Tubelib sequencer
|
|
|
|
]]--
|
|
|
|
-- Load support for I18n
|
|
local S = smartline.S
|
|
|
|
local RUNNING_STATE = 1
|
|
local STOP_STATE = 0
|
|
local NUM_SLOTS = 8
|
|
|
|
local sHELP = "label[0,0;"..
|
|
S([[SmartLine Sequencer Help
|
|
|
|
Define a sequence of commands to control other machines.
|
|
Numbers(s) are the node numbers, the command shall sent to.
|
|
The commands 'on'/'off' are used for machines and other nodes.
|
|
Offset is the time to the next line in seconds (1..999).
|
|
If endless is set, the Sequencer restarts again and again.
|
|
The command ' ' does nothing, only consuming the offset time.
|
|
]])..
|
|
"]"
|
|
|
|
|
|
local sAction = ",on,off"
|
|
local kvAction = {[""]=1, ["on"]=2, ["off"]=3}
|
|
local tAction = {nil, "on", "off"}
|
|
|
|
local function formspec(state, rules, endless)
|
|
endless = endless == 1 and "true" or "false"
|
|
local tbl = {"size[8,9.2]"..
|
|
default.gui_bg..
|
|
default.gui_bg_img..
|
|
default.gui_slots..
|
|
"label[0,0;"..S("Number(s)").."]label[2.1,0;"..S("Command").."]label[6.4,0;"..S("Offset/s").."]"}
|
|
|
|
for idx, rule in ipairs(rules or {}) do
|
|
tbl[#tbl+1] = "field[0.2,"..(-0.2+idx)..";2,1;num"..idx..";;"..(rule.num or "").."]"
|
|
tbl[#tbl+1] = "dropdown[2,"..(-0.4+idx)..";3.9,1;act"..idx..";"..sAction..";"..(rule.act or "").."]"
|
|
tbl[#tbl+1] = "field[6.2,"..(-0.2+idx)..";2,1;offs"..idx..";;"..(rule.offs or "").."]"
|
|
end
|
|
tbl[#tbl+1] = "checkbox[0,8.5;endless;"..S("Run endless")..";"..endless.."]"
|
|
tbl[#tbl+1] = "button[4.5,8.5;1.5,1;help;"..S("help").."]"
|
|
tbl[#tbl+1] = "image_button[6.5,8.5;1,1;".. tubelib.state_button(state) ..";button;]"
|
|
|
|
return table.concat(tbl)
|
|
end
|
|
|
|
local function formspec_help()
|
|
return "size[13,10]"..
|
|
default.gui_bg..
|
|
default.gui_bg_img..
|
|
default.gui_slots..
|
|
"field[0,0;0,0;_type_;;"..S("help").."]"..
|
|
sHELP..
|
|
--"label[0.2,0;test]"..
|
|
"button[11.5,9;1.5,1;close;"..S("close").."]"
|
|
end
|
|
|
|
local function stop_the_sequencer(pos)
|
|
local node = minetest.get_node(pos)
|
|
local meta = minetest.get_meta(pos)
|
|
local number = meta:get_string("number")
|
|
meta:set_int("running", STOP_STATE)
|
|
meta:set_string("infotext", S("SmartLine Sequencer").." "..number..": "..S("stopped"))
|
|
local rules = minetest.deserialize(meta:get_string("rules"))
|
|
local endless = meta:get_int("endless") or 0
|
|
meta:set_string("formspec", formspec(tubelib.STOPPED, rules, endless))
|
|
minetest.get_node_timer(pos):stop()
|
|
return false
|
|
end
|
|
|
|
local function get_next_slot(idx, rules, endless)
|
|
idx = idx + 1
|
|
if idx <= #rules and rules[idx].offs ~= "" and rules[idx].num ~= "" then
|
|
return idx
|
|
elseif endless == 1 then
|
|
return 1
|
|
end
|
|
return nil
|
|
end
|
|
|
|
local function restart_timer(pos, time)
|
|
local timer = minetest.get_node_timer(pos)
|
|
if timer:is_started() then
|
|
timer:stop()
|
|
end
|
|
if type(time) == "number" then
|
|
timer:start(time)
|
|
end
|
|
end
|
|
|
|
local function check_rules(pos, elapsed)
|
|
if tubelib.data_not_corrupted(pos) then
|
|
local meta = minetest.get_meta(pos)
|
|
local rules = minetest.deserialize(meta:get_string("rules"))
|
|
if rules then
|
|
local running = meta:get_int("running")
|
|
local index = meta:get_int("index") or 1
|
|
local number = meta:get_string("number")
|
|
local endless = meta:get_int("endless") or 0
|
|
local placer_name = meta:get_string("placer_name")
|
|
while true do -- process all rules as long as offs == 0
|
|
local rule = rules[index]
|
|
local offs = rules[index].offs
|
|
if type(offs) == "string" then
|
|
offs = 0
|
|
end
|
|
tubelib.send_message(rule.num, placer_name, nil, tAction[rule.act], number)
|
|
index = get_next_slot(index, rules, endless)
|
|
if index ~= nil and offs ~= nil and running == 1 then
|
|
-- after the last rule a pause with 2 or more sec is required
|
|
if index == 1 and offs < 1 then
|
|
offs = 2
|
|
end
|
|
meta:set_string("infotext", S("SmartLine Sequencer").." "..number..": "..S("running").." ("..index.."/"..NUM_SLOTS..")")
|
|
meta:set_int("index", index)
|
|
if offs > 0 then
|
|
minetest.after(0, restart_timer, pos, offs)
|
|
return false
|
|
end
|
|
else
|
|
return stop_the_sequencer(pos)
|
|
end
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
return false
|
|
end
|
|
|
|
local function start_the_sequencer(pos)
|
|
local node = minetest.get_node(pos)
|
|
local meta = minetest.get_meta(pos)
|
|
local number = meta:get_string("number")
|
|
meta:set_int("running", RUNNING_STATE)
|
|
meta:set_int("index", 1)
|
|
meta:set_string("infotext", S("SmartLine Sequencer").." "..number..": "..S("running").." (1/"..NUM_SLOTS..")")
|
|
local rules = minetest.deserialize(meta:get_string("rules"))
|
|
local endless = meta:get_int("endless") or 0
|
|
meta:set_string("formspec", formspec(tubelib.RUNNING, rules, endless))
|
|
minetest.get_node_timer(pos):start(0.1)
|
|
return false
|
|
end
|
|
|
|
local function on_receive_fields(pos, formname, fields, player)
|
|
local meta = minetest.get_meta(pos)
|
|
local running = meta:get_int("running")
|
|
if minetest.is_protected(pos, player:get_player_name()) then
|
|
return
|
|
end
|
|
if fields.help ~= nil then
|
|
meta:set_string("formspec", formspec_help())
|
|
return
|
|
end
|
|
|
|
local endless = meta:get_int("endless") or 0
|
|
if fields.endless ~= nil then
|
|
endless = fields.endless == "true" and 1 or 0
|
|
meta:set_int("index", 1)
|
|
end
|
|
meta:set_int("endless", endless)
|
|
|
|
local rules = minetest.deserialize(meta:get_string("rules"))
|
|
if fields.exit ~= nil then
|
|
meta:set_string("formspec", formspec(tubelib.state(running), rules, endless))
|
|
return
|
|
end
|
|
|
|
for idx = 1,NUM_SLOTS do
|
|
if fields["offs"..idx] ~= nil then
|
|
rules[idx].offs = tonumber(fields["offs"..idx]) or ""
|
|
end
|
|
if fields["num"..idx] ~= nil and tubelib.check_numbers(fields["num"..idx]) then
|
|
rules[idx].num = fields["num"..idx]
|
|
end
|
|
if fields["act"..idx] ~= nil then
|
|
rules[idx].act = kvAction[fields["act"..idx]]
|
|
end
|
|
end
|
|
meta:set_string("rules", minetest.serialize(rules))
|
|
|
|
if fields.button ~= nil then
|
|
if running > STOP_STATE then
|
|
stop_the_sequencer(pos)
|
|
else
|
|
start_the_sequencer(pos)
|
|
end
|
|
elseif fields.num1 ~= nil then -- any other change?
|
|
stop_the_sequencer(pos)
|
|
else
|
|
local endless = meta:get_int("endless") or 0
|
|
meta:set_string("formspec", formspec(tubelib.state(running), rules, endless))
|
|
end
|
|
end
|
|
|
|
minetest.register_node("smartline:sequencer", {
|
|
description = S("SmartLine Sequencer"),
|
|
inventory_image = "smartline_sequencer_inventory.png",
|
|
wield_image = "smartline_sequencer_inventory.png",
|
|
stack_max = 1,
|
|
tiles = {
|
|
-- up, down, right, left, back, front
|
|
"smartline.png",
|
|
"smartline.png",
|
|
"smartline.png",
|
|
"smartline.png",
|
|
"smartline.png",
|
|
"smartline.png^smartline_sequencer.png",
|
|
},
|
|
|
|
drawtype = "nodebox",
|
|
node_box = {
|
|
type = "fixed",
|
|
fixed = {
|
|
{ -6/32, -6/32, 14/32, 6/32, 6/32, 16/32},
|
|
},
|
|
},
|
|
after_place_node = function(pos, placer)
|
|
local meta = minetest.get_meta(pos)
|
|
local number = tubelib.add_node(pos, "smartline:sequencer")
|
|
local rules = {}
|
|
for idx = 1,NUM_SLOTS do
|
|
rules[idx] = {offs = "", num = "", act = 1}
|
|
end
|
|
meta:set_string("placer_name", placer:get_player_name())
|
|
meta:set_string("rules", minetest.serialize(rules))
|
|
meta:set_string("number", number)
|
|
meta:set_int("index", 1)
|
|
meta:set_int("endless", 0)
|
|
meta:get_int("running", STOP_STATE)
|
|
meta:set_string("formspec", formspec(tubelib.STOPPED, rules, 0))
|
|
meta:set_string("infotext", S("SmartLine Sequencer").." "..number)
|
|
end,
|
|
|
|
on_receive_fields = on_receive_fields,
|
|
|
|
on_dig = function(pos, node, puncher, pointed_thing)
|
|
if minetest.is_protected(pos, puncher:get_player_name()) then
|
|
return
|
|
end
|
|
local meta = minetest.get_meta(pos)
|
|
local running = meta:get_int("running")
|
|
if running ~= 1 then
|
|
minetest.node_dig(pos, node, puncher, pointed_thing)
|
|
tubelib.remove_node(pos)
|
|
end
|
|
end,
|
|
|
|
on_timer = check_rules,
|
|
|
|
paramtype = "light",
|
|
sunlight_propagates = true,
|
|
paramtype2 = "facedir",
|
|
groups = {choppy=2, cracky=2, crumbly=2},
|
|
is_ground_content = false,
|
|
sounds = default.node_sound_stone_defaults(),
|
|
})
|
|
|
|
|
|
minetest.register_craft({
|
|
output = "smartline:sequencer",
|
|
recipe = {
|
|
{"", "default:mese_crystal", ""},
|
|
{"dye:blue", "default:copper_ingot", "tubelib:wlanchip"},
|
|
{"", "", ""},
|
|
},
|
|
})
|
|
|
|
tubelib.register_node("smartline:sequencer", {}, {
|
|
on_recv_message = function(pos, topic, payload)
|
|
local node = minetest.get_node(pos)
|
|
if topic == "on" then
|
|
start_the_sequencer(pos)
|
|
elseif topic == "off" then
|
|
-- do not stop immediately
|
|
local meta = minetest.get_meta(pos)
|
|
meta:set_int("endless", 0)
|
|
end
|
|
end,
|
|
on_node_load = function(pos)
|
|
local meta = minetest.get_meta(pos)
|
|
if meta:get_int("running") ~= STOP_STATE then
|
|
meta:set_int("running", RUNNING_STATE)
|
|
minetest.get_node_timer(pos):start(1)
|
|
end
|
|
end,
|
|
})
|