232 lines
7.2 KiB
Lua
232 lines
7.2 KiB
Lua
|
--[[
|
||
|
|
||
|
TechAge
|
||
|
=======
|
||
|
|
||
|
Copyright (C) 2019-2020 Joachim Stolberg
|
||
|
|
||
|
GPL v3
|
||
|
See LICENSE.txt for more information
|
||
|
|
||
|
API for Power Nodes
|
||
|
|
||
|
]]--
|
||
|
|
||
|
--local P2S = minetest.pos_to_string
|
||
|
--local M = minetest.get_meta
|
||
|
--local N = function(pos) return minetest.get_node(pos).name end
|
||
|
--local S = techage.S
|
||
|
|
||
|
local net_def = techage.networks.net_def
|
||
|
local networks = techage.networks
|
||
|
|
||
|
-- Consumer States
|
||
|
local STOPPED = 1
|
||
|
local NOPOWER = 2
|
||
|
local RUNNING = 3
|
||
|
|
||
|
techage.power = {}
|
||
|
|
||
|
techage.power.STOPPED = STOPPED
|
||
|
techage.power.NOPOWER = NOPOWER
|
||
|
techage.power.RUNNING = RUNNING
|
||
|
|
||
|
-- determine network ID (largest hash number of all generators)
|
||
|
local function determine_netID(pos, outdir, Cable)
|
||
|
local netID = 0
|
||
|
networks.connection_walk(pos, outdir, Cable, function(pos, indir, node)
|
||
|
local ntype = net_def(pos, Cable.tube_type).ntype
|
||
|
if ntype ~= "junc" then
|
||
|
local new = minetest.hash_node_position(pos)
|
||
|
if netID <= new then
|
||
|
netID = new
|
||
|
end
|
||
|
end
|
||
|
end)
|
||
|
return netID
|
||
|
end
|
||
|
|
||
|
-- store network ID on each node
|
||
|
local function store_netID(pos, outdir, netID, Cable)
|
||
|
networks.connection_walk(pos, outdir, Cable, function(pos, indir, node)
|
||
|
--techage.mark_position("singleplayer", pos, "store", "", 2)-----------------------------------------
|
||
|
--print(node.name, dump(net_def(pos, Cable.tube_type)))
|
||
|
if net_def(pos, Cable.tube_type) then
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
nvm[Cable.tube_type] = nvm[Cable.tube_type] or {}
|
||
|
nvm[Cable.tube_type]["netID"] = netID
|
||
|
end
|
||
|
end)
|
||
|
end
|
||
|
|
||
|
-- delete network and ID on each node
|
||
|
local function delete_netID(pos, outdir, Cable)
|
||
|
local netID = 0
|
||
|
networks.connection_walk(pos, outdir, Cable, function(pos, indir, node)
|
||
|
--techage.mark_position("singleplayer", pos, "delete", "", 2)----------------------------------------
|
||
|
if net_def(pos, Cable.tube_type) then
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
if nvm[Cable.tube_type] and nvm[Cable.tube_type]["netID"] then
|
||
|
netID = nvm[Cable.tube_type]["netID"]
|
||
|
nvm[Cable.tube_type]["netID"] = nil
|
||
|
end
|
||
|
end
|
||
|
end)
|
||
|
networks.delete_network(Cable.tube_type, netID)
|
||
|
end
|
||
|
|
||
|
-- Keep the network up and running
|
||
|
local function trigger_network(pos, outdir, Cable)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local netID = nvm[Cable.tube_type] and nvm[Cable.tube_type]["netID"]
|
||
|
if not netID then
|
||
|
--print("determine_netID !!!!!!!!!!!!!!!!!!!!")
|
||
|
netID = determine_netID(pos, outdir, Cable)
|
||
|
store_netID(pos, outdir, netID, Cable)
|
||
|
networks.build_network(pos, outdir, Cable, netID)
|
||
|
elseif not networks.get_network(Cable.tube_type, netID) then
|
||
|
--print("build_network !!!!!!!!!!!!!!!!!!!!")
|
||
|
netID = determine_netID(pos, outdir, Cable)
|
||
|
store_netID(pos, outdir, netID, Cable)
|
||
|
networks.build_network(pos, outdir, Cable, netID)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local function build_network_consumer(pos, Cable)
|
||
|
local outdirs = techage.networks.get_node_connections(pos, Cable.tube_type)
|
||
|
if #outdirs == 1 then
|
||
|
local netID = determine_netID(pos, outdirs[1], Cable)
|
||
|
store_netID(pos, outdirs[1], netID, Cable)
|
||
|
networks.build_network(pos, outdirs[1], Cable, netID)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
-- To be called from each node via 'tubelib2_on_update2'
|
||
|
-- 'output' is optional and only needed for nodes with dedicated
|
||
|
-- pipe sides (e.g. pumps).
|
||
|
function techage.power.update_network(pos, outdir, Cable)
|
||
|
networks.node_connections(pos, Cable) -- update node internal data
|
||
|
delete_netID(pos, outdir, Cable) -- network walk to delete all IDs
|
||
|
end
|
||
|
|
||
|
--
|
||
|
-- Read the current power value from all connected devices (used for solar cells)
|
||
|
-- Only used by the solar inverter to collect the power of all solar cells.
|
||
|
-- Only one inverter per network is allowed. Therefore, we have to check,
|
||
|
-- if additional inverters are in the network.
|
||
|
-- Function returns in addition the number of found inverters.
|
||
|
function techage.power.get_power(pos, outdir, Cable, inverter)
|
||
|
local sum = 0
|
||
|
local num_inverter = 0
|
||
|
networks.connection_walk(pos, outdir, Cable, function(pos, indir, node)
|
||
|
--techage.mark_position("singleplayer", pos, "get_power", "", 2)-----------------------------------------
|
||
|
local def = net_def(pos, Cable.tube_type)
|
||
|
if def and def.on_getpower then
|
||
|
sum = sum + def.on_getpower(pos)
|
||
|
else
|
||
|
local node = techage.get_node_lvm(pos)
|
||
|
if node.name == inverter then
|
||
|
num_inverter = num_inverter + 1
|
||
|
end
|
||
|
end
|
||
|
end)
|
||
|
return sum, num_inverter
|
||
|
end
|
||
|
|
||
|
|
||
|
|
||
|
--
|
||
|
-- Consumer related functions
|
||
|
--
|
||
|
|
||
|
-- check if there is a living network
|
||
|
function techage.power.power_available(pos, Cable)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local tlib_type = Cable.tube_type
|
||
|
local netID = nvm[Cable.tube_type] and nvm[Cable.tube_type]["netID"]
|
||
|
return networks.has_network(tlib_type, netID)
|
||
|
end
|
||
|
|
||
|
-- this is more a try to start, the start will be performed by on_power()
|
||
|
function techage.power.consumer_start(pos, Cable, cycle_time)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local tlib_type = Cable.tube_type
|
||
|
nvm[tlib_type] = nvm[tlib_type] or {}
|
||
|
nvm[tlib_type]["calive"] = (cycle_time / 2) + 1
|
||
|
nvm[tlib_type]["cstate"] = NOPOWER
|
||
|
nvm[tlib_type]["taken"] = 0
|
||
|
end
|
||
|
|
||
|
function techage.power.consumer_stop(pos, Cable)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local tlib_type = Cable.tube_type
|
||
|
nvm[tlib_type] = nvm[tlib_type] or {}
|
||
|
nvm[tlib_type]["calive"] = -1
|
||
|
nvm[tlib_type]["cstate"] = STOPPED
|
||
|
nvm[tlib_type]["taken"] = 0
|
||
|
end
|
||
|
|
||
|
function techage.power.consumer_alive(pos, Cable, cycle_time)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local def = nvm[Cable.tube_type] -- power related network data
|
||
|
if def then
|
||
|
-- if network is deleted (cable removed/placed) rebuild it to prevent flickering lights
|
||
|
if not def["netID"] or not networks.get_network(Cable.tube_type, def["netID"]) then
|
||
|
build_network_consumer(pos, Cable)
|
||
|
end
|
||
|
local rv = (cycle_time / 2) + 1
|
||
|
if def["netID"] and def["calive"] and def["calive"] < rv then -- network available
|
||
|
def["calive"] = rv
|
||
|
return def["taken"] or 0
|
||
|
elseif not def["cstate"] or def["cstate"] == RUNNING then
|
||
|
local ndef = net_def(pos, Cable.tube_type)
|
||
|
ndef.on_nopower(pos, Cable.tube_type)
|
||
|
def["cstate"] = NOPOWER
|
||
|
end
|
||
|
else
|
||
|
local ndef = net_def(pos, Cable.tube_type)
|
||
|
ndef.on_nopower(pos, Cable.tube_type)
|
||
|
end
|
||
|
return 0
|
||
|
end
|
||
|
|
||
|
--
|
||
|
-- Generator related functions
|
||
|
--
|
||
|
-- curr_power is optional, only needed for generators with variable output power
|
||
|
function techage.power.generator_start(pos, Cable, cycle_time, outdir, curr_power)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local tlib_type = Cable.tube_type
|
||
|
nvm[tlib_type] = nvm[tlib_type] or {}
|
||
|
nvm[tlib_type]["galive"] = (cycle_time / 2) + 2
|
||
|
nvm[tlib_type]["gstate"] = RUNNING
|
||
|
nvm[tlib_type]["given"] = 0
|
||
|
nvm[tlib_type]["curr_power"] = curr_power
|
||
|
trigger_network(pos, outdir, Cable)
|
||
|
end
|
||
|
|
||
|
function techage.power.generator_stop(pos, Cable, outdir)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local tlib_type = Cable.tube_type
|
||
|
nvm[tlib_type] = nvm[tlib_type] or {}
|
||
|
nvm[tlib_type]["galive"] = -1
|
||
|
nvm[tlib_type]["gstate"] = STOPPED
|
||
|
nvm[tlib_type]["given"] = 0
|
||
|
end
|
||
|
|
||
|
-- curr_power is optional, only needed for generators with variable output power
|
||
|
function techage.power.generator_alive(pos, Cable, cycle_time, outdir, curr_power)
|
||
|
local nvm = techage.get_nvm(pos)
|
||
|
local def = nvm[Cable.tube_type] -- power related network data
|
||
|
if def then
|
||
|
trigger_network(pos, outdir, Cable)
|
||
|
def["galive"] = (cycle_time / 2) + 2
|
||
|
def["curr_power"] = curr_power
|
||
|
return def["given"] or 0
|
||
|
end
|
||
|
return 0
|
||
|
end
|
||
|
|
||
|
-- function delete_netID(pos, outdir, Cable)
|
||
|
techage.power.delete_netID = delete_netID
|