--[[

	TechAge
	=======

	Copyright (C) 2017-2019 Joachim Stolberg

	GPL v3
	See LICENSE.txt for more information

	TA3 Sequencer
	
]]--

-- for lazy programmers
local M = minetest.get_meta
local S = techage.S

local logic = techage.logic
local NUM_SLOTS = 8

local sAction = ",on,off"
local kvAction = {[""]=1, ["on"]=2, ["off"]=3}
local tAction = {nil, "on", "off"}

local function new_rules()
	local tbl = {}
	for idx = 1,NUM_SLOTS do
		tbl[idx] = {offs = "", num = "", act = 1}
	end
	return tbl
end

local function formspec(state, rules, endless)
	local tbl = {"size[8,9.2]"..
		default.gui_bg..
		default.gui_bg_img..
		default.gui_slots..
		"label[0,0;Number(s)]label[2.1,0;Command]label[6.4,0;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;Run endless;"..dump(endless).."]"
	tbl[#tbl+1] = "image_button[5,8.5;1,1;".. techage.state_button(state) ..";button;]"
	tbl[#tbl+1] = "button[6.2,8.5;1.5,1;help;help]"
	
	return table.concat(tbl)
end

local function formspec_help()
	return "size[8,9.2]"..
		default.gui_bg..
		default.gui_bg_img..
		default.gui_slots..
		"label[2,0;TA3 Sequencer Help]"..
		"label[0,1;Define a sequence of commands\nto control other machines.]"..
		"label[0,2.2;Numbers(s) are the node numbers,\nthe command shall sent to.]"..
		"label[0,3.4;The commands 'on'/'off' are used\n for machines and other nodes.]"..
		"label[0,4.6;Offset is the time to the\nnext line in seconds (1..999).]"..
		"label[0,5.8;If endless is set, the Sequencer\nrestarts again and again.]"..
		"label[0,7;The command '  ' does nothing,\nonly consuming the offset time.]"..
		"button[3,8;2,1;exit;close]"
end

local function stop_the_sequencer(pos)
	local mem = tubelib2.get_mem(pos)
	local meta = M(pos)
	mem.running = false
	mem.endless = mem.endless or false
	mem.rules = mem.rules or new_rules()
	logic.infotext(meta, S("TA3 Sequencer"), "stopped")
	meta:set_string("formspec", formspec(techage.STOPPED, mem.rules, mem.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 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)
	local mem = tubelib2.get_mem(pos)
	local own_num = M(pos):get_string("node_number")
	mem.rules = mem.rules or new_rules()
	mem.running = mem.running or false
	mem.index = mem.index or 1
	mem.endless = mem.endless or false
	while true do -- process all rules as long as offs == 0
		local rule = mem.rules[mem.index]
		local offs = tonumber(mem.rules[mem.index].offs or 1)
		techage.send_multi(own_num, rule.num, tAction[rule.act])
		mem.index = get_next_slot(mem.index, mem.rules, mem.endless)
		if mem.index ~= nil and offs ~= nil and mem.running then
			-- after the last rule a pause with 1 or more sec is required
			if mem.index == 1 and offs < 1 then
				offs = 1
			end
			if offs > 0 then
				minetest.after(0, restart_timer, pos, offs)
				return false
			end
		else
			return stop_the_sequencer(pos)
		end
	end
	return false
end

local function start_the_sequencer(pos)
	local mem = tubelib2.get_mem(pos)
	local meta = M(pos)
	mem.running = true
	mem.endless = mem.endless or false
	mem.rules = mem.rules or new_rules()
	logic.infotext(meta, S("TA3 Sequencer"), "running")
	meta:set_string("formspec", formspec(techage.RUNNING, mem.rules, mem.endless))
	minetest.get_node_timer(pos):start(0.1)
	return false
end

local function on_receive_fields(pos, formname, fields, player)
	if minetest.is_protected(pos, player:get_player_name()) then
		return
	end
	
	local meta = M(pos)
	local mem = tubelib2.get_mem(pos)
	mem.running = mem.running or false
	mem.endless = mem.endless or false
	mem.rules = mem.rules or new_rules()
	
	if fields.help ~= nil then
		meta:set_string("formspec", formspec_help())
		return
	end
	
	if fields.endless ~= nil then
		mem.endless = fields.endless == "true"
		mem.index = 1
	end
	
	if fields.exit ~= nil then
		if mem.running then
			meta:set_string("formspec", formspec(techage.RUNNING, mem.rules, mem.endless))
		else
			meta:set_string("formspec", formspec(techage.STOPPED, mem.rules, mem.endless))
		end
		return
	end

	for idx = 1,NUM_SLOTS do
		if fields["offs"..idx] ~= nil then
			mem.rules[idx].offs = tonumber(fields["offs"..idx]) or ""
		end
		if fields["num"..idx] ~= nil and 
				techage.check_numbers(fields["num"..idx], player:get_player_name()) then
			mem.rules[idx].num = fields["num"..idx]
		end
		if fields["act"..idx] ~= nil then
			mem.rules[idx].act = kvAction[fields["act"..idx]]
		end
	end

	if fields.button ~= nil then
		if mem.running then
			stop_the_sequencer(pos)
		else
			start_the_sequencer(pos)
		end
	elseif fields.num1 ~= nil then  -- any other change?
		stop_the_sequencer(pos)
	else
		if mem.running then
			meta:set_string("formspec", formspec(techage.RUNNING, mem.rules, mem.endless))
		else
			meta:set_string("formspec", formspec(techage.STOPPED, mem.rules, mem.endless))
		end
	end
end

minetest.register_node("techage:ta3_sequencer", {
	description = S("TA3 Sequencer"),
	tiles = {
		-- up, down, right, left, back, front
		"techage_filling_ta3.png^techage_frame_ta3_top.png",
		"techage_filling_ta3.png^techage_frame_ta3_top.png",
		"techage_filling_ta3.png^techage_frame_ta3.png^techage_appl_sequencer.png",
	},
	
	after_place_node = function(pos, placer)
		local meta = M(pos)
		local mem = tubelib2.init_mem(pos)
		logic.after_place_node(pos, placer, "techage:ta3_sequencer", S("TA3 Sequencer"))
		logic.infotext(meta, S("TA3 Sequencer"), S("stopped"))
		mem.rules = new_rules()
		mem.index = 1
		mem.running = false
		mem.endless = false
		meta:set_string("formspec", formspec(techage.STOPPED, mem.rules, mem.endless))
	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 mem = tubelib2.get_mem(pos)
		if not mem.running then
			minetest.node_dig(pos, node, puncher, pointed_thing)
			techage.remove_node(pos)
			tubelib2.del_mem(pos)
		end
	end,
	
	on_timer = check_rules,
	
	paramtype2 = "facedir",
	groups = {choppy=2, cracky=2, crumbly=2},
	is_ground_content = false,
	sounds = default.node_sound_stone_defaults(),
})


minetest.register_craft({
	output = "techage:ta3_sequencer",
	recipe = {
		{"group:wood", "group:wood", ""},
		{"default:mese_crystal", "techage:vacuum_tube", ""},
		{"group:wood", "group:wood", ""},
	},
})

techage.register_node({"techage:ta3_sequencer"}, {
	on_recv_message = function(pos, src, topic, payload)
		if topic == "on" then
			start_the_sequencer(pos)
		elseif topic == "off" then
			-- do not stop immediately
			local mem = tubelib2.get_mem(pos)
			mem.endless = false
		else
			return "unsupported"
		end
	end,
	on_node_load = function(pos)
		local mem = tubelib2.get_mem(pos)
		if mem.running then
			minetest.get_node_timer(pos):start(1)
		end
	end,
})