techage/basis/node_states.lua

562 lines
16 KiB
Lua
Raw Permalink Normal View History

2019-03-02 14:24:48 +03:00
--[[
TechAge
=======
Copyright (C) 2019-2023 Joachim Stolberg
2019-03-02 14:24:48 +03:00
2020-10-19 20:09:17 +03:00
AGPL v3
2019-03-02 14:24:48 +03:00
See LICENSE.txt for more information
A state model/class for TechAge nodes.
]]--
2022-01-03 23:40:31 +03:00
--[[
2019-03-02 14:24:48 +03:00
Node states:
+-----------------------------------+ +------------+
2022-01-03 23:40:31 +03:00
| | | |
| V V |
| +---------+ |
| | | |
| +---------| STOPPED | |
| | | | |
| button | +---------+ |
| | ^ |
button | V | button |
| +---------+ | | button
| +--------->| |---------+ |
| | power | RUNNING | |
| | +------| |---------+ |
| | | +---------+ | |
| | | ^ | | |
| | | | | | |
| | V | V V |
| +---------+ +----------+ +---------+ |
| | | | | | | |
+---| NOPOWER | | STANDBY/ | | FAULT |----------+
| | | BLOCKED | | |
+---------+ +----------+ +---------+
2019-03-02 14:24:48 +03:00
| cycle time operational needs power
+---------+------------+-------------+-------------
| RUNNING normal yes yes
| BLOCKED long yes no
| STANDBY long yes no
| NOPOWER long no no
| FAULT none no no
| STOPPED none no no
2020-01-26 01:15:44 +03:00
Node nvm data:
"techage_state" - node state, like "RUNNING"
2020-01-26 01:15:44 +03:00
"techage_countdown" - countdown to standby mode
2019-03-02 14:24:48 +03:00
]]--
-- for lazy programmers
local S = function(pos) if pos then return minetest.pos_to_string(pos) end end
local P = minetest.string_to_pos
local M = minetest.get_meta
local N = techage.get_node_lvm
2019-03-02 14:24:48 +03:00
2022-08-09 22:15:28 +03:00
local MAX_CYCLE_TIME = 20
2019-03-02 14:24:48 +03:00
--
-- TechAge machine states
--
2022-06-05 22:20:43 +03:00
techage.RUNNING = 1 -- in normal operation/turned on
techage.BLOCKED = 2 -- a pushing node is blocked due to a full destination inventory
techage.STANDBY = 3 -- nothing to do (e.g. no input items), or node (world) not loaded
techage.NOPOWER = 4 -- only for power consuming nodes, no operation
techage.FAULT = 5 -- any fault state (e.g. wrong source items), which can be fixed by the player
techage.STOPPED = 6 -- not operational/turned off
techage.UNLOADED = 7 -- Map block unloaded
techage.INACTIVE = 8 -- Map block loaded but node is not actively working
2019-03-02 14:24:48 +03:00
techage.StatesImg = {
2022-01-03 23:40:31 +03:00
"techage_inv_button_on.png",
"techage_inv_button_warning.png",
2022-01-03 23:40:31 +03:00
"techage_inv_button_standby.png",
"techage_inv_button_nopower.png",
"techage_inv_button_error.png",
2022-01-03 23:40:31 +03:00
"techage_inv_button_off.png",
2019-03-02 14:24:48 +03:00
}
local function error(pos, msg)
minetest.log("error", "[TA states] "..msg.." at "..S(pos).." "..N(pos).name)
end
2019-03-02 14:24:48 +03:00
-- Return state button image for the node inventory
function techage.state_button(state)
if state and state < 7 and state > 0 then
return techage.StatesImg[state]
end
return "techage_inv_button_off.png"
2019-03-02 14:24:48 +03:00
end
2020-01-26 01:15:44 +03:00
function techage.get_power_image(pos, nvm)
local node = techage.get_node_lvm(pos)
local s = "3" -- electrical power
if string.find(node.name, "techage:ta2") then
s = "2" -- axles power
end
return "techage_inv_powerT"..s..".png"
end
2019-03-02 14:24:48 +03:00
-- State string based on button states
techage.StateStrings = {"running", "blocked", "standby", "nopower", "fault", "stopped"}
2019-03-02 14:24:48 +03:00
--
-- Local States
--
local RUNNING = techage.RUNNING
local BLOCKED = techage.BLOCKED
2019-03-02 14:24:48 +03:00
local STANDBY = techage.STANDBY
2019-05-21 14:15:13 +03:00
local NOPOWER = techage.NOPOWER
2019-03-02 14:24:48 +03:00
local FAULT = techage.FAULT
local STOPPED = techage.STOPPED
2019-03-02 14:24:48 +03:00
--
-- NodeStates Class Functions
--
techage.NodeStates = {}
local NodeStates = techage.NodeStates
2020-01-26 01:15:44 +03:00
local function can_start(pos, nvm)
--if false, node goes in FAULT
return true
end
2020-01-26 01:15:44 +03:00
local function has_power(pos, nvm)
--if false, node goes in NOPOWER
2019-03-02 14:24:48 +03:00
return true
end
local function swap_node(pos, new_name, old_name)
local node = techage.get_node_lvm(pos)
if node.name == new_name then
2019-03-06 00:42:31 +03:00
return
end
if node.name == old_name then
node.name = new_name
minetest.swap_node(pos, node)
end
2019-03-06 00:42:31 +03:00
end
-- true if node_timer should be executed
2020-01-26 01:15:44 +03:00
function techage.is_operational(nvm)
local state = nvm.techage_state or STOPPED
return state < NOPOWER
end
function techage.is_running(nvm)
return (nvm.techage_state or STOPPED) == RUNNING
end
2019-08-17 00:44:11 +03:00
-- consumes power
2020-01-26 01:15:44 +03:00
function techage.needs_power(nvm)
local state = nvm.techage_state or STOPPED
2021-06-05 12:42:30 +03:00
return state == RUNNING or state == NOPOWER
end
2021-06-09 23:09:59 +03:00
-- consumes power
function techage.needs_power2(state)
state = state or STOPPED
return state == RUNNING or state == NOPOWER
end
2020-01-26 01:15:44 +03:00
function techage.get_state_string(nvm)
return techage.StateStrings[nvm.techage_state or STOPPED]
2019-08-17 00:44:11 +03:00
end
2019-03-02 14:24:48 +03:00
function NodeStates:new(attr)
local o = {
-- mandatory
cycle_time = attr.cycle_time, -- for running state
standby_ticks = attr.standby_ticks, -- for standby state
-- optional
countdown_ticks = attr.countdown_ticks or 1,
2019-03-02 14:24:48 +03:00
node_name_passive = attr.node_name_passive,
2022-01-03 23:40:31 +03:00
node_name_active = attr.node_name_active,
2019-03-02 14:24:48 +03:00
infotext_name = attr.infotext_name,
has_power = attr.has_power or has_power,
can_start = attr.can_start or can_start,
start_node = attr.start_node,
stop_node = attr.stop_node,
2019-03-02 14:24:48 +03:00
formspec_func = attr.formspec_func,
2019-06-09 16:02:17 +03:00
on_state_change = attr.on_state_change,
2021-09-27 22:34:24 +03:00
quick_start = attr.quick_start,
2019-03-02 14:24:48 +03:00
}
setmetatable(o, self)
self.__index = self
return o
end
2020-01-26 01:15:44 +03:00
function NodeStates:node_init(pos, nvm, number)
nvm.techage_state = STOPPED
2019-03-09 21:26:15 +03:00
M(pos):set_string("node_number", number)
2019-03-02 14:24:48 +03:00
if self.infotext_name then
M(pos):set_string("infotext", self.infotext_name.." "..number..": stopped")
2019-03-02 14:24:48 +03:00
end
if self.formspec_func then
2020-01-26 01:15:44 +03:00
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-03-02 14:24:48 +03:00
end
end
-- to be used to re-start the timer outside of node_timer()
local function start_timer_delayed(pos, cycle_time)
local t = minetest.get_node_timer(pos)
t:stop()
if cycle_time > 0.9 then
minetest.after(0.1, t.start, t, cycle_time)
else
error(pos, "invalid cycle_time")
end
end
2020-01-26 01:15:44 +03:00
function NodeStates:stop(pos, nvm)
local state = nvm.techage_state or STOPPED
nvm.techage_state = STOPPED
2019-06-16 22:06:16 +03:00
if self.stop_node then
2020-01-26 01:15:44 +03:00
self.stop_node(pos, nvm, state)
2019-03-02 14:24:48 +03:00
end
2019-06-16 22:06:16 +03:00
if self.node_name_passive then
swap_node(pos, self.node_name_passive, self.node_name_active)
2019-06-16 22:06:16 +03:00
end
if self.infotext_name then
local number = M(pos):get_string("node_number")
M(pos):set_string("infotext", self.infotext_name.." "..number..": stopped")
end
if self.formspec_func then
2020-01-26 01:15:44 +03:00
nvm.ta_state_tooltip = "stopped"
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-06-16 22:06:16 +03:00
end
if self.on_state_change then
self.on_state_change(pos, state, STOPPED)
end
if minetest.get_node_timer(pos):is_started() then
minetest.get_node_timer(pos):stop()
end
return true
2019-03-02 14:24:48 +03:00
end
2020-01-26 01:15:44 +03:00
function NodeStates:start(pos, nvm)
local state = nvm.techage_state or STOPPED
if state ~= RUNNING and state ~= FAULT then
2020-01-26 01:15:44 +03:00
local res = self.can_start(pos, nvm, state)
if res ~= true then
2020-01-26 01:15:44 +03:00
self:fault(pos, nvm, res)
2019-03-02 14:24:48 +03:00
return false
end
2020-01-26 01:15:44 +03:00
if not self.has_power(pos, nvm, state) then
self:nopower(pos, nvm)
return false
end
2020-01-26 01:15:44 +03:00
nvm.techage_state = RUNNING
if self.start_node then
2020-01-26 01:15:44 +03:00
self.start_node(pos, nvm, state)
2019-03-02 14:24:48 +03:00
end
2022-01-02 23:00:46 +03:00
nvm.techage_countdown = self.countdown_ticks
2019-03-02 14:24:48 +03:00
if self.node_name_active then
swap_node(pos, self.node_name_active, self.node_name_passive)
2019-03-02 14:24:48 +03:00
end
if self.infotext_name then
2019-03-09 21:26:15 +03:00
local number = M(pos):get_string("node_number")
M(pos):set_string("infotext", self.infotext_name.." "..number..": running")
2019-03-02 14:24:48 +03:00
end
if self.formspec_func then
2020-01-26 01:15:44 +03:00
nvm.ta_state_tooltip = "running"
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-03-02 14:24:48 +03:00
end
2019-03-10 13:36:00 +03:00
if minetest.get_node_timer(pos):is_started() then
minetest.get_node_timer(pos):stop()
2019-03-05 23:20:44 +03:00
end
2019-06-09 16:02:17 +03:00
if self.on_state_change then
self.on_state_change(pos, state, RUNNING)
end
start_timer_delayed(pos, self.cycle_time)
2022-01-03 23:40:31 +03:00
if self.quick_start and state == STOPPED then
2021-09-27 22:34:24 +03:00
self.quick_start(pos, 0)
end
2022-08-09 22:15:28 +03:00
self:trigger_state(pos, nvm)
2019-03-02 14:24:48 +03:00
return true
end
return false
end
2020-02-02 00:00:58 +03:00
function NodeStates:standby(pos, nvm, err_string)
2020-01-26 01:15:44 +03:00
local state = nvm.techage_state or STOPPED
if state == RUNNING or state == BLOCKED then
2020-01-26 01:15:44 +03:00
nvm.techage_state = STANDBY
2019-03-02 14:24:48 +03:00
if self.node_name_passive then
swap_node(pos, self.node_name_passive, self.node_name_active)
2019-03-02 14:24:48 +03:00
end
if self.infotext_name then
2019-03-09 21:26:15 +03:00
local number = M(pos):get_string("node_number")
2020-02-02 00:00:58 +03:00
M(pos):set_string("infotext", self.infotext_name.." "..number..": "..(err_string or "standby"))
2019-03-02 14:24:48 +03:00
end
if self.formspec_func then
2020-02-02 00:00:58 +03:00
nvm.ta_state_tooltip = err_string or "standby"
2020-01-26 01:15:44 +03:00
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-03-02 14:24:48 +03:00
end
2019-06-09 16:02:17 +03:00
if self.on_state_change then
self.on_state_change(pos, state, STANDBY)
end
start_timer_delayed(pos, self.cycle_time * self.standby_ticks)
2019-03-02 14:24:48 +03:00
return true
end
return false
2022-01-03 23:40:31 +03:00
end
2019-03-02 14:24:48 +03:00
-- special case of standby for pushing nodes
2020-02-10 23:20:54 +03:00
function NodeStates:blocked(pos, nvm, err_string)
2020-01-26 01:15:44 +03:00
local state = nvm.techage_state or STOPPED
2019-06-09 16:02:17 +03:00
if state == RUNNING then
2020-01-26 01:15:44 +03:00
nvm.techage_state = BLOCKED
2019-03-02 14:24:48 +03:00
if self.node_name_passive then
swap_node(pos, self.node_name_passive, self.node_name_active)
2019-03-02 14:24:48 +03:00
end
if self.infotext_name then
2019-03-09 21:26:15 +03:00
local number = M(pos):get_string("node_number")
2020-02-10 23:20:54 +03:00
M(pos):set_string("infotext", self.infotext_name.." "..number..": "..(err_string or "blocked"))
2019-03-02 14:24:48 +03:00
end
if self.formspec_func then
2020-02-10 23:20:54 +03:00
nvm.ta_state_tooltip = err_string or "blocked"
2020-01-26 01:15:44 +03:00
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-03-02 14:24:48 +03:00
end
2019-06-09 16:02:17 +03:00
if self.on_state_change then
self.on_state_change(pos, state, BLOCKED)
end
start_timer_delayed(pos, self.cycle_time * self.standby_ticks)
2019-03-02 14:24:48 +03:00
return true
end
return false
2022-01-03 23:40:31 +03:00
end
2019-03-02 14:24:48 +03:00
2020-01-26 01:15:44 +03:00
function NodeStates:nopower(pos, nvm, err_string)
local state = nvm.techage_state or RUNNING
if state ~= NOPOWER then
2020-01-26 01:15:44 +03:00
nvm.techage_state = NOPOWER
2019-05-21 14:15:13 +03:00
if self.node_name_passive then
swap_node(pos, self.node_name_passive, self.node_name_active)
2019-05-21 14:15:13 +03:00
end
if self.infotext_name then
local number = M(pos):get_string("node_number")
2020-02-02 00:00:58 +03:00
M(pos):set_string("infotext", self.infotext_name.." "..number..": "..(err_string or "no power"))
2019-05-21 14:15:13 +03:00
end
if self.formspec_func then
2020-01-26 01:15:44 +03:00
nvm.ta_state_tooltip = err_string or "no power"
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-05-21 14:15:13 +03:00
end
2019-06-09 16:02:17 +03:00
if self.on_state_change then
self.on_state_change(pos, state, NOPOWER)
end
start_timer_delayed(pos, self.cycle_time * self.standby_ticks)
2019-05-21 14:15:13 +03:00
return true
end
return false
2022-01-03 23:40:31 +03:00
end
2019-05-21 14:15:13 +03:00
2020-01-26 01:15:44 +03:00
function NodeStates:fault(pos, nvm, err_string)
local state = nvm.techage_state or STOPPED
err_string = err_string or "fault"
2019-06-09 16:02:17 +03:00
if state == RUNNING or state == STOPPED then
2020-01-26 01:15:44 +03:00
nvm.techage_state = FAULT
2019-03-02 14:24:48 +03:00
if self.node_name_passive then
swap_node(pos, self.node_name_passive, self.node_name_active)
2019-03-02 14:24:48 +03:00
end
if self.infotext_name then
2019-03-09 21:26:15 +03:00
local number = M(pos):get_string("node_number")
M(pos):set_string("infotext", self.infotext_name.." "..number..": "..err_string)
2019-03-02 14:24:48 +03:00
end
if self.formspec_func then
2020-01-26 01:15:44 +03:00
nvm.ta_state_tooltip = err_string or "fault"
M(pos):set_string("formspec", self.formspec_func(self, pos, nvm))
2019-03-02 14:24:48 +03:00
end
2019-06-09 16:02:17 +03:00
if self.on_state_change then
self.on_state_change(pos, state, FAULT)
end
2019-03-02 14:24:48 +03:00
minetest.get_node_timer(pos):stop()
return true
end
return false
2022-01-03 23:40:31 +03:00
end
2019-03-02 14:24:48 +03:00
2020-01-26 01:15:44 +03:00
function NodeStates:get_state(nvm)
return nvm.techage_state or techage.STOPPED
2019-03-02 14:24:48 +03:00
end
-- keep the timer running?
2020-01-26 01:15:44 +03:00
function NodeStates:is_active(nvm)
local state = nvm.techage_state or STOPPED
return state < FAULT
2019-03-02 14:24:48 +03:00
end
2019-05-02 00:01:14 +03:00
function NodeStates:start_if_standby(pos)
2020-01-26 01:15:44 +03:00
local nvm = techage.get_nvm(pos)
if nvm.techage_state == STANDBY then
self:start(pos, nvm)
2019-05-02 00:01:14 +03:00
end
end
2019-03-02 14:24:48 +03:00
-- To be called if node is idle.
-- If countdown reaches zero, the node is set to STANDBY.
2020-01-26 01:15:44 +03:00
function NodeStates:idle(pos, nvm)
local countdown = (nvm.techage_countdown or 0) - 1
nvm.techage_countdown = countdown
2019-05-02 00:01:14 +03:00
if countdown <= 0 then
2020-01-26 01:15:44 +03:00
self:standby(pos, nvm)
2019-03-02 14:24:48 +03:00
end
end
-- To be called after successful node action to raise the timer
-- and keep the node in state RUNNING
function NodeStates:keep_running(pos, nvm, val)
2019-03-02 14:24:48 +03:00
-- set to RUNNING if not already done
2020-01-26 01:15:44 +03:00
if nvm.techage_state ~= RUNNING then
self:start(pos, nvm)
end
2020-01-26 01:15:44 +03:00
nvm.techage_countdown = val or 4
nvm.last_active = minetest.get_gametime()
2019-03-02 14:24:48 +03:00
end
2021-04-30 20:00:59 +03:00
function NodeStates:trigger_state(pos, nvm)
nvm.last_active = minetest.get_gametime()
end
2019-03-02 14:24:48 +03:00
-- Start/stop node based on button events.
-- if function returns false, no button was pressed
2020-01-26 01:15:44 +03:00
function NodeStates:state_button_event(pos, nvm, fields)
2019-03-02 14:24:48 +03:00
if fields.state_button ~= nil then
2020-01-26 01:15:44 +03:00
local state = nvm.techage_state or STOPPED
2019-03-02 14:24:48 +03:00
if state == STOPPED or state == STANDBY or state == BLOCKED then
2020-01-26 01:15:44 +03:00
if not self:start(pos, nvm) and (state == STANDBY or state == BLOCKED) then
self:stop(pos, nvm)
2022-01-03 23:40:31 +03:00
end
2019-05-21 14:15:13 +03:00
elseif state == RUNNING or state == FAULT or state == NOPOWER then
2020-01-26 01:15:44 +03:00
self:stop(pos, nvm)
2019-03-02 14:24:48 +03:00
end
return true
end
return false
end
2020-01-26 01:15:44 +03:00
function NodeStates:get_state_button_image(nvm)
local state = nvm.techage_state or STOPPED
2019-03-02 14:24:48 +03:00
return techage.state_button(state)
end
2020-01-26 01:15:44 +03:00
function NodeStates:get_state_tooltip(nvm)
local tp = nvm.ta_state_tooltip or ""
return tp..";#0C3D32;#FFFFFF"
end
2019-03-02 14:24:48 +03:00
-- command interface
function NodeStates:on_receive_message(pos, topic, payload)
2020-01-26 01:15:44 +03:00
local nvm = techage.get_nvm(pos)
2019-03-02 14:24:48 +03:00
if topic == "on" then
2020-01-26 01:15:44 +03:00
self:start(pos, techage.get_nvm(pos))
2019-03-02 14:24:48 +03:00
return true
elseif topic == "off" then
2020-01-26 01:15:44 +03:00
self:stop(pos, techage.get_nvm(pos))
2019-03-02 14:24:48 +03:00
return true
elseif topic == "state" then
2020-02-28 00:33:42 +03:00
local node = minetest.get_node(pos)
2019-03-02 14:24:48 +03:00
if node.name == "ignore" then -- unloaded node?
2019-09-14 00:01:55 +03:00
return "unloaded"
elseif nvm.techage_state == RUNNING then
2022-08-09 22:15:28 +03:00
local ttl = (nvm.last_active or 0) + MAX_CYCLE_TIME
if ttl < minetest.get_gametime() then
return "inactive"
end
2019-03-02 14:24:48 +03:00
end
2020-01-26 01:15:44 +03:00
return techage.get_state_string(techage.get_nvm(pos))
elseif topic == "fuel" then
2020-02-28 00:33:42 +03:00
return techage.fuel.get_fuel_amount(nvm)
elseif topic == "load" then
return techage.liquid.get_liquid_amount(nvm)
else
return "unsupported"
2019-03-02 14:24:48 +03:00
end
end
2022-01-03 23:40:31 +03:00
2022-06-05 22:20:43 +03:00
function NodeStates:on_beduino_receive_cmnd(pos, topic, payload)
if topic == 1 then
if payload[1] == 0 then
self:stop(pos, techage.get_nvm(pos))
return 0
else
self:start(pos, techage.get_nvm(pos))
return 0
end
else
return 2 -- unknown or invalid topic
end
end
function NodeStates:on_beduino_request_data(pos, topic, payload)
local nvm = techage.get_nvm(pos)
if topic == 128 then
return 0, techage.get_node_lvm(pos).name
elseif topic == 129 then
local node = minetest.get_node(pos)
if node.name == "ignore" then -- unloaded node?
return 0, {techage.UNLOADED}
elseif nvm.techage_state == RUNNING then
2022-08-09 22:15:28 +03:00
local ttl = (nvm.last_active or 0) + MAX_CYCLE_TIME
2022-06-05 22:20:43 +03:00
if ttl < minetest.get_gametime() then
return 0, {techage.INACTIVE}
end
end
return 0, {nvm.techage_state or STOPPED}
else
return 2, "" -- topic is unknown or invalid
end
end
function NodeStates.get_beduino_state(pos)
local node = minetest.get_node(pos)
local nvm = techage.get_nvm(pos)
if node.name == "ignore" then -- unloaded node?
return 0, {techage.UNLOADED}
elseif nvm.techage_state == RUNNING then
local ttl = (nvm.last_active or 0) + MAX_CYCLE_TIME
if ttl < minetest.get_gametime() then
return 0, {techage.INACTIVE}
end
end
return 0, {nvm.techage_state or STOPPED}
end
-- restart timer
function NodeStates:on_node_load(pos)
local nvm = techage.get_nvm(pos)
local state = nvm.techage_state or STOPPED
if state == RUNNING then
minetest.get_node_timer(pos):start(self.cycle_time)
elseif state < FAULT then
minetest.get_node_timer(pos):start(self.cycle_time * self.standby_ticks)
end
2019-03-02 14:24:48 +03:00
end
2019-03-09 21:26:15 +03:00
minetest.register_node("techage:defect_dummy", {
description = "Corrupted Node (to be replaced)",
tiles = {
"techage_filling_ta2.png^techage_frame_ta2.png",
"techage_filling_ta2.png^techage_frame_ta2.png",
2019-03-10 15:53:53 +03:00
"techage_filling_ta2.png^techage_frame_ta2.png^techage_appl_defect.png",
"techage_filling_ta2.png^techage_frame_ta2.png^techage_appl_defect.png",
"techage_filling_ta2.png^techage_frame_ta2.png^techage_appl_defect.png",
"techage_filling_ta2.png^techage_frame_ta2.png^techage_appl_defect.png",
2019-03-09 21:26:15 +03:00
},
drop = "",
groups = {cracky=2, crumbly=2, choppy=2, not_in_creative_inventory=1},
is_ground_content = false,
})