2019-03-02 14:24:48 +03:00
|
|
|
--[[
|
|
|
|
|
|
|
|
TechAge
|
|
|
|
=======
|
|
|
|
|
|
|
|
Copyright (C) 2019 Joachim Stolberg
|
|
|
|
|
|
|
|
LGPLv2.1+
|
|
|
|
See LICENSE.txt for more information
|
|
|
|
|
|
|
|
A state model/class for TechAge nodes.
|
|
|
|
|
|
|
|
]]--
|
|
|
|
|
|
|
|
|
|
|
|
--[[
|
|
|
|
|
|
|
|
Node states:
|
|
|
|
|
|
|
|
+-----------------------------------+ +------------+
|
|
|
|
| | | |
|
|
|
|
| V V |
|
|
|
|
| +---------+ |
|
|
|
|
| | | |
|
|
|
|
| +---------| STOPPED | |
|
|
|
|
| | | | |
|
|
|
|
| button | +---------+ |
|
|
|
|
| | ^ |
|
|
|
|
repair | V | button |
|
|
|
|
| +---------+ | | button
|
|
|
|
| | |---------+ |
|
|
|
|
| | RUNNING | |
|
|
|
|
| +--------| |---------+ |
|
|
|
|
| | +---------+ | |
|
|
|
|
| | ^ | | |
|
|
|
|
| | | | | |
|
|
|
|
| V | V V |
|
|
|
|
| +---------+ +----------+ +---------+ |
|
|
|
|
| | | | | | | |
|
|
|
|
+---| DEFECT | | STANDBY/ | | FAULT |----------+
|
|
|
|
| | | BLOCKED | | |
|
|
|
|
+---------+ +----------+ +---------+
|
|
|
|
|
|
|
|
Node metadata:
|
2019-03-05 00:46:35 +03:00
|
|
|
"techage_number" - string with tubelib number, like "123"
|
|
|
|
"techage_state" - node state, like "RUNNING"
|
|
|
|
"techage_item_meter" - node item/runtime counter
|
|
|
|
"techage_countdown" - countdown to stadby mode
|
|
|
|
"techage_aging" - aging counter
|
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
|
|
|
|
|
|
|
|
|
|
|
|
--
|
|
|
|
-- TechAge machine states
|
|
|
|
--
|
|
|
|
|
|
|
|
techage.STOPPED = 1 -- not operational/turned off
|
|
|
|
techage.RUNNING = 2 -- in normal operation/turned on
|
|
|
|
techage.STANDBY = 3 -- nothing to do (e.g. no input items), or blocked anyhow (output jammed),
|
|
|
|
-- or node (world) not loaded
|
2019-03-05 00:46:35 +03:00
|
|
|
techage.FAULT = 4 -- any fault state (e.g. no power), which can be fixed by the player
|
2019-03-02 14:24:48 +03:00
|
|
|
techage.BLOCKED = 5 -- a pushing node is blocked due to a full destination inventory
|
|
|
|
techage.DEFECT = 6 -- a defect (broken), which has to be repaired by the player
|
|
|
|
|
|
|
|
techage.StatesImg = {
|
2019-03-05 00:46:35 +03:00
|
|
|
"techage_inv_button_off.png",
|
|
|
|
"techage_inv_button_on.png",
|
|
|
|
"techage_inv_button_standby.png",
|
|
|
|
"techage_inv_button_error.png",
|
|
|
|
"techage_inv_button_warning.png",
|
|
|
|
"techage_inv_button_off.png",
|
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
|
2019-03-05 00:46:35 +03:00
|
|
|
return "techage_inv_button_off.png"
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
-- State string based on button states
|
|
|
|
techage.StateStrings = {"stopped", "running", "standby", "fault", "blocked", "defect"}
|
|
|
|
|
|
|
|
--
|
|
|
|
-- Local States
|
|
|
|
--
|
|
|
|
local STOPPED = techage.STOPPED
|
|
|
|
local RUNNING = techage.RUNNING
|
|
|
|
local STANDBY = techage.STANDBY
|
|
|
|
local FAULT = techage.FAULT
|
|
|
|
local BLOCKED = techage.BLOCKED
|
|
|
|
local DEFECT = techage.DEFECT
|
|
|
|
|
|
|
|
|
|
|
|
local AGING_FACTOR = 4 -- defect random factor
|
|
|
|
|
|
|
|
--
|
|
|
|
-- NodeStates Class Functions
|
|
|
|
--
|
|
|
|
techage.NodeStates = {}
|
|
|
|
local NodeStates = techage.NodeStates
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
local function can_start(pos, mem)
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
function NodeStates:new(attr)
|
|
|
|
local o = {
|
|
|
|
-- mandatory
|
|
|
|
cycle_time = attr.cycle_time, -- for running state
|
|
|
|
standby_ticks = attr.standby_ticks, -- for standby state
|
|
|
|
has_item_meter = attr.has_item_meter, -- true/false
|
|
|
|
-- optional
|
|
|
|
node_name_passive = attr.node_name_passive,
|
|
|
|
node_name_active = attr.node_name_active,
|
|
|
|
node_name_defect = attr.node_name_defect,
|
|
|
|
infotext_name = attr.infotext_name,
|
2019-03-05 00:46:35 +03:00
|
|
|
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,
|
|
|
|
}
|
|
|
|
if attr.aging_factor then
|
|
|
|
o.aging_level1 = attr.aging_factor * techage.machine_aging_value
|
|
|
|
o.aging_level2 = attr.aging_factor * techage.machine_aging_value * AGING_FACTOR
|
|
|
|
end
|
|
|
|
setmetatable(o, self)
|
|
|
|
self.__index = self
|
|
|
|
return o
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:node_init(pos, mem, number)
|
|
|
|
mem.techage_state = STOPPED
|
|
|
|
mem.techage_number = number
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": stopped")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.has_item_meter then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_item_meter = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.aging_level1 then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_aging = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:stop(pos, mem)
|
|
|
|
local state = mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
if state ~= DEFECT then
|
2019-03-05 00:46:35 +03:00
|
|
|
if self.stop_node then
|
|
|
|
self.stop_node(pos, mem, state)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_state = STOPPED
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.node_name_passive then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_passive
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": stopped")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
2019-03-05 23:20:44 +03:00
|
|
|
if minetest.get_node_timer(pos):is_started() then
|
|
|
|
minetest.get_node_timer(pos):stop()
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:start(pos, mem, called_from_on_timer)
|
|
|
|
local state = mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
if state == STOPPED or state == STANDBY or state == BLOCKED then
|
2019-03-05 00:46:35 +03:00
|
|
|
if not self.can_start(pos, mem, state) then
|
|
|
|
self:fault(pos, mem)
|
2019-03-02 14:24:48 +03:00
|
|
|
return false
|
|
|
|
end
|
2019-03-05 00:46:35 +03:00
|
|
|
if self.start_node then
|
|
|
|
self.start_node(pos, mem, state)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_state = RUNNING
|
|
|
|
mem.techage_countdown = 4
|
2019-03-02 14:24:48 +03:00
|
|
|
if called_from_on_timer then
|
|
|
|
-- timer has to be stopped once to be able to be restarted
|
|
|
|
self.stop_timer = true
|
|
|
|
end
|
|
|
|
if self.node_name_active then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_active
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_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
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
2019-03-05 23:20:44 +03:00
|
|
|
if not minetest.get_node_timer(pos):is_started() then
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time)
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:standby(pos, mem)
|
|
|
|
if mem.techage_state == RUNNING then
|
|
|
|
mem.techage_state = STANDBY
|
2019-03-02 14:24:48 +03:00
|
|
|
-- timer has to be stopped once to be able to be restarted
|
|
|
|
self.stop_timer = true
|
|
|
|
if self.node_name_passive then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_passive
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": standby")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time * self.standby_ticks)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
-- special case of standby for pushing nodes
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:blocked(pos, mem)
|
|
|
|
if mem.techage_state == RUNNING then
|
|
|
|
mem.techage_state = BLOCKED
|
2019-03-02 14:24:48 +03:00
|
|
|
-- timer has to be stopped once to be able to be restarted
|
|
|
|
self.stop_timer = true
|
|
|
|
if self.node_name_passive then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_passive
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": blocked")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time * self.standby_ticks)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:fault(pos, mem)
|
2019-03-05 20:38:41 +03:00
|
|
|
if mem.techage_state == RUNNING or mem.techage_state == STOPPED then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_state = FAULT
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.node_name_passive then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_passive
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": fault")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
minetest.get_node_timer(pos):stop()
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:defect(pos, mem)
|
|
|
|
mem.techage_state = DEFECT
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.node_name_defect then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_defect
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": defect")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
minetest.get_node_timer(pos):stop()
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:get_state(mem)
|
|
|
|
return mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:get_state_string(mem)
|
|
|
|
return techage.StateStrings[mem.techage_state]
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:is_active(mem)
|
|
|
|
local state = mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.stop_timer == true then
|
|
|
|
self.stop_timer = false
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
return state == RUNNING or state == STANDBY or state == BLOCKED
|
|
|
|
end
|
|
|
|
|
|
|
|
-- To be called if node is idle.
|
|
|
|
-- If countdown reaches zero, the node is set to STANDBY.
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:idle(pos, mem)
|
|
|
|
local countdown = mem.techage_countdown - 1
|
|
|
|
mem.techage_countdown = countdown
|
2019-03-02 14:24:48 +03:00
|
|
|
if countdown < 0 then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:standby(pos, mem)
|
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
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:keep_running(pos, mem, val, num_items)
|
2019-03-02 14:24:48 +03:00
|
|
|
num_items = num_items or 1
|
|
|
|
-- set to RUNNING if not already done
|
2019-03-05 00:46:35 +03:00
|
|
|
self:start(pos, mem, true)
|
|
|
|
mem.techage_countdown = val
|
2019-03-05 23:20:44 +03:00
|
|
|
if self.has_item_meter then
|
|
|
|
mem.techage_item_meter = mem.techage_item_meter + (num_items or 1)
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.aging_level1 then
|
2019-03-05 00:46:35 +03:00
|
|
|
local cnt = mem.techage_aging + num_items
|
|
|
|
mem.techage_aging = cnt
|
2019-03-02 14:24:48 +03:00
|
|
|
if (cnt > (self.aging_level1) and math.random(self.aging_level2/num_items) == 1)
|
|
|
|
or cnt >= 999999 then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:defect(pos, mem)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Start/stop node based on button events.
|
|
|
|
-- if function returns false, no button was pressed
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:state_button_event(pos, mem, fields)
|
2019-03-02 14:24:48 +03:00
|
|
|
if fields.state_button ~= nil then
|
2019-03-05 00:46:35 +03:00
|
|
|
local state = mem.techage_state
|
|
|
|
print("on_receive_fields", state)
|
2019-03-02 14:24:48 +03:00
|
|
|
if state == STOPPED or state == STANDBY or state == BLOCKED then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:start(pos, mem)
|
2019-03-02 14:24:48 +03:00
|
|
|
elseif state == RUNNING or state == FAULT then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:stop(pos, mem)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-05 00:46:35 +03:00
|
|
|
function NodeStates:get_state_button_image(mem)
|
|
|
|
local state = mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
return techage.state_button(state)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- command interface
|
|
|
|
function NodeStates:on_receive_message(pos, topic, payload)
|
|
|
|
if topic == "on" then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:start(pos, tubelib2.get_mem(pos))
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
elseif topic == "off" then
|
2019-03-05 00:46:35 +03:00
|
|
|
self:stop(pos, tubelib2.get_mem(pos))
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
elseif topic == "state" then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
if node.name == "ignore" then -- unloaded node?
|
|
|
|
return "blocked"
|
|
|
|
end
|
2019-03-05 00:46:35 +03:00
|
|
|
return self:get_state_string(tubelib2.get_mem(pos))
|
2019-03-02 14:24:48 +03:00
|
|
|
elseif self.has_item_meter and topic == "counter" then
|
2019-03-05 00:46:35 +03:00
|
|
|
return mem.techage_item_meter
|
2019-03-02 14:24:48 +03:00
|
|
|
elseif self.has_item_meter and topic == "clear_counter" then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_item_meter = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
elseif self.aging_level1 and topic == "aging" then
|
2019-03-05 00:46:35 +03:00
|
|
|
return mem.techage_aging
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- repair corrupt node data and/or migrate node to state2
|
|
|
|
function NodeStates:on_node_load(pos, not_start_timer)
|
2019-03-05 00:46:35 +03:00
|
|
|
local mem = tubelib2.get_mem(pos)
|
2019-03-02 14:24:48 +03:00
|
|
|
|
|
|
|
-- legacy node number/state/counter?
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.number
|
2019-03-02 14:24:48 +03:00
|
|
|
if number ~= "" and number ~= nil then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_number = number
|
|
|
|
mem.techage_state = techage.state(mem.running)
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.has_item_meter then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_item_meter = mem.counter
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.aging_level1 then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_aging = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.number = nil
|
|
|
|
mem.running = 0
|
|
|
|
mem.counter = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
-- node number corrupt?
|
2019-03-05 00:46:35 +03:00
|
|
|
number = mem.techage_number
|
2019-03-02 14:24:48 +03:00
|
|
|
if number == "" then
|
|
|
|
number = techage.get_new_number(pos, self.node_name_passive)
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_number = number
|
2019-03-02 14:24:48 +03:00
|
|
|
else
|
|
|
|
local info = techage.get_node_info(number)
|
|
|
|
if not info or info.pos ~= pos then
|
|
|
|
number = techage.get_new_number(pos, self.node_name_passive)
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_number = number
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- state corrupt?
|
2019-03-05 00:46:35 +03:00
|
|
|
local state = mem.techage_state
|
2019-03-02 14:24:48 +03:00
|
|
|
if state == 0 then
|
|
|
|
if minetest.get_node_timer(pos):is_started() then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_state = RUNNING
|
2019-03-02 14:24:48 +03:00
|
|
|
else
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_state = STOPPED
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
elseif state == RUNNING and not not_start_timer then
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time)
|
|
|
|
elseif state == STANDBY then
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time * self.standby_ticks)
|
|
|
|
elseif state == BLOCKED then
|
|
|
|
minetest.get_node_timer(pos):start(self.cycle_time * self.standby_ticks)
|
|
|
|
end
|
|
|
|
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Repair of defect (feature!) nodes
|
|
|
|
function NodeStates:on_node_repair(pos)
|
2019-03-05 00:46:35 +03:00
|
|
|
local mem = tubelib2.get_mem(pos)
|
|
|
|
if mem.techage_state == DEFECT then
|
|
|
|
mem.techage_state = STOPPED
|
2019-03-02 14:24:48 +03:00
|
|
|
if self.node_name_passive then
|
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
node.name = self.node_name_passive
|
|
|
|
minetest.swap_node(pos, node)
|
|
|
|
end
|
|
|
|
if self.aging_level1 then
|
2019-03-05 00:46:35 +03:00
|
|
|
mem.techage_aging = 0
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.infotext_name then
|
2019-03-05 00:46:35 +03:00
|
|
|
local number = mem.techage_number
|
|
|
|
M(pos):set_string("infotext", self.infotext_name.." "..number..": stopped")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if self.formspec_func then
|
2019-03-05 00:46:35 +03:00
|
|
|
M(pos):set_string("formspec", self.formspec_func(self, pos, mem))
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Return working or defect machine, depending on machine lifetime
|
|
|
|
function NodeStates:after_dig_node(pos, oldnode, oldmetadata, digger)
|
2019-03-05 00:46:35 +03:00
|
|
|
local mem = tubelib2.get_mem(pos)
|
2019-03-02 14:24:48 +03:00
|
|
|
local inv = minetest.get_inventory({type="player", name=digger:get_player_name()})
|
2019-03-05 00:46:35 +03:00
|
|
|
local cnt = mem.techage_aging or 0
|
2019-03-05 23:20:44 +03:00
|
|
|
if self.aging_level1 then
|
|
|
|
local is_defect = cnt > self.aging_level1 and math.random(self.aging_level2 / cnt) == 1
|
|
|
|
if self.node_name_defect and is_defect then
|
|
|
|
inv:add_item("main", ItemStack(self.node_name_defect))
|
|
|
|
else
|
|
|
|
inv:add_item("main", ItemStack(self.node_name_passive))
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
else
|
|
|
|
inv:add_item("main", ItemStack(self.node_name_passive))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Return "full", "loaded", or "empty" depending
|
|
|
|
-- on the number of fuel stack items.
|
|
|
|
-- Function only works on fuel inventories with one stacks/99 items
|
|
|
|
function techage.fuelstate(meta, listname, item)
|
|
|
|
if meta == nil or meta.get_inventory == nil then return nil end
|
|
|
|
local inv = meta:get_inventory()
|
|
|
|
if inv:is_empty(listname) then
|
|
|
|
return "empty"
|
|
|
|
end
|
|
|
|
local list = inv:get_list(listname)
|
|
|
|
if #list == 1 and list[1]:get_count() == 99 then
|
|
|
|
return "full"
|
|
|
|
else
|
|
|
|
return "loaded"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|