267 lines
8.0 KiB
Lua
267 lines
8.0 KiB
Lua
--[[
|
|
|
|
TechAge
|
|
=======
|
|
|
|
Copyright (C) 2019-2022 Joachim Stolberg
|
|
|
|
AGPL v3
|
|
See LICENSE.txt for more information
|
|
|
|
TA4 Solar Power DC/AC Inverter
|
|
|
|
]]--
|
|
|
|
-- for lazy programmers
|
|
local M = minetest.get_meta
|
|
local S = techage.S
|
|
local P2S = function(pos) if pos then return minetest.pos_to_string(pos) end end
|
|
local S2P = minetest.string_to_pos
|
|
|
|
local Cable = techage.ElectricCable
|
|
local Solar = techage.TA4_Cable
|
|
local power = networks.power
|
|
local control = networks.control
|
|
|
|
local CYCLE_TIME = 2
|
|
local PWR_PERF = 100
|
|
local COUNTDOWN_TICKS = 1
|
|
|
|
local function determine_power(pos, nvm)
|
|
-- determine DC node position
|
|
local outdir = M(pos):get_int("leftdir")
|
|
local netw = networks.get_network_table(pos, Solar, outdir) or {}
|
|
local num_inv = #(netw.con or {})
|
|
local max_power = 0
|
|
for _, power in ipairs(control.request(pos, Solar, outdir, "junc", "power")) do
|
|
max_power = max_power + power
|
|
end
|
|
|
|
if num_inv == 1 then -- only one inverter is allowed
|
|
nvm.max_power = math.min(PWR_PERF, max_power)
|
|
else
|
|
nvm.max_power = 0
|
|
end
|
|
return max_power, num_inv
|
|
end
|
|
|
|
local function has_dc_power(pos, nvm)
|
|
local time = minetest.get_timeofday() or 0
|
|
if time < 6.00/24.00 or time > 18.00/24.00 then
|
|
nvm.ticks = 0
|
|
nvm.max_power = 0
|
|
return false
|
|
end
|
|
nvm.ticks = nvm.ticks or 0
|
|
if (nvm.ticks % 30) == 0 then -- calculate max_power not to often
|
|
determine_power(pos, nvm)
|
|
else
|
|
nvm.max_power = nvm.max_power or 0
|
|
end
|
|
nvm.ticks = nvm.ticks + 1
|
|
return nvm.max_power > 0
|
|
end
|
|
|
|
local function formspec(self, pos, nvm)
|
|
local max_power = nvm.max_power or 0
|
|
local provided = nvm.provided or 0
|
|
local arrow = "image[2.5,1.5;1,1;techage_form_arrow_bg.png^[transformR270]"
|
|
if techage.is_running(nvm) then
|
|
arrow = "image[2.5,1.5;1,1;techage_form_arrow_fg.png^[transformR270]"
|
|
end
|
|
return "size[6,4]"..
|
|
default.gui_bg..
|
|
default.gui_bg_img..
|
|
default.gui_slots..
|
|
"box[0,-0.1;5.8,0.5;#c6e8ff]"..
|
|
"label[2.5,-0.1;"..minetest.colorize( "#000000", S("Inverter")).."]"..
|
|
techage.formspec_power_bar(pos, 0, 0.8, S("Power DC"), max_power, PWR_PERF)..
|
|
techage.formspec_power_bar(pos, 3.5, 0.8, S("Power AC"), provided, max_power)..
|
|
arrow..
|
|
"image_button[2.5,3;1,1;".. self:get_state_button_image(nvm) ..";state_button;]"..
|
|
"tooltip[2.5,3;1,1;"..self:get_state_tooltip(nvm).."]"
|
|
end
|
|
|
|
local function can_start(pos, nvm, state)
|
|
local max_power, num_inverter = determine_power(pos, nvm)
|
|
if num_inverter > 1 then return S("solar network error") end
|
|
if max_power == 0 then return S("no solar power") end
|
|
return true
|
|
end
|
|
|
|
local function start_node(pos, nvm, state)
|
|
local meta = M(pos)
|
|
nvm.provided = 0
|
|
nvm.ticks = 0
|
|
local outdir = meta:get_int("outdir")
|
|
power.start_storage_calc(pos, Cable, outdir)
|
|
techage.evaluate_charge_termination(nvm, meta)
|
|
end
|
|
|
|
local function stop_node(pos, nvm, state)
|
|
nvm.provided = 0
|
|
nvm.running = nil -- legacy
|
|
local outdir = M(pos):get_int("outdir")
|
|
power.start_storage_calc(pos, Cable, outdir)
|
|
end
|
|
|
|
local State = techage.NodeStates:new({
|
|
node_name_passive = "techage:ta4_solar_inverter",
|
|
cycle_time = CYCLE_TIME,
|
|
standby_ticks = 2,
|
|
formspec_func = formspec,
|
|
infotext_name = S("TA4 Solar Inverter"),
|
|
can_start = can_start,
|
|
start_node = start_node,
|
|
stop_node = stop_node,
|
|
})
|
|
|
|
local function node_timer(pos, elapsed)
|
|
local nvm = techage.get_nvm(pos)
|
|
local running = techage.is_running(nvm)
|
|
local has_power = has_dc_power(pos, nvm)
|
|
if running and not has_power then
|
|
State:standby(pos, nvm)
|
|
stop_node(pos, nvm, State)
|
|
elseif not running and has_power then
|
|
State:start(pos, nvm)
|
|
-- start_node() is called implicit
|
|
elseif running then
|
|
local meta = M(pos)
|
|
local outdir = meta:get_int("outdir")
|
|
local tp1 = tonumber(meta:get_string("termpoint1"))
|
|
local tp2 = tonumber(meta:get_string("termpoint2"))
|
|
nvm.provided = power.provide_power(pos, Cable, outdir, nvm.max_power, tp1, tp2)
|
|
local val = power.get_storage_load(pos, Cable, outdir, nvm.max_power)
|
|
if val > 0 then
|
|
nvm.load = val
|
|
end
|
|
State:keep_running(pos, nvm, COUNTDOWN_TICKS)
|
|
end
|
|
if techage.is_activeformspec(pos) then
|
|
M(pos):set_string("formspec", formspec(State, pos, nvm))
|
|
end
|
|
return State:is_active(nvm)
|
|
end
|
|
|
|
local function on_receive_fields(pos, formname, fields, player)
|
|
if minetest.is_protected(pos, player:get_player_name()) then
|
|
return
|
|
end
|
|
local nvm = techage.get_nvm(pos)
|
|
State:state_button_event(pos, nvm, fields)
|
|
end
|
|
|
|
local function on_rightclick(pos, node, clicker)
|
|
local nvm = techage.get_nvm(pos)
|
|
determine_power(pos, nvm)
|
|
techage.set_activeformspec(pos, clicker)
|
|
M(pos):set_string("formspec", formspec(State, pos, nvm))
|
|
end
|
|
|
|
local function get_generator_data(pos, outdir, tlib2)
|
|
local nvm = techage.get_nvm(pos)
|
|
if techage.is_running(nvm) then
|
|
return {level = (nvm.load or 0) / nvm.max_power, perf = nvm.max_power, capa = nvm.max_power * 2}
|
|
end
|
|
end
|
|
|
|
minetest.register_node("techage:ta4_solar_inverter", {
|
|
description = S("TA4 Solar Inverter"),
|
|
tiles = {
|
|
-- up, down, right, left, back, front
|
|
"techage_filling_ta4.png^techage_frame_ta4_top.png^techage_appl_arrow.png",
|
|
"techage_filling_ta4.png^techage_frame_ta4.png",
|
|
"techage_filling_ta4.png^techage_frame_ta4.png^techage_appl_hole_electric.png",
|
|
"techage_filling_ta4.png^techage_frame_ta4.png^techage_appl_ta4_cable.png",
|
|
"techage_filling_ta4.png^techage_frame_ta4.png^techage_appl_inverter.png",
|
|
"techage_filling_ta4.png^techage_frame_ta4.png^techage_appl_inverter.png",
|
|
},
|
|
paramtype2 = "facedir",
|
|
groups = {cracky=2, crumbly=2, choppy=2},
|
|
on_rotate = screwdriver.disallow,
|
|
is_ground_content = false,
|
|
|
|
after_place_node = function(pos)
|
|
local nvm = techage.get_nvm(pos)
|
|
M(pos):set_int("outdir", networks.side_to_outdir(pos, "R"))
|
|
M(pos):set_int("leftdir", networks.side_to_outdir(pos, "L"))
|
|
Cable:after_place_node(pos)
|
|
Solar:after_place_node(pos)
|
|
local number = techage.add_node(pos, "techage:ta4_solar_inverter")
|
|
State:node_init(pos, nvm, number)
|
|
M(pos):set_string("formspec", formspec(State, pos, nvm))
|
|
end,
|
|
|
|
after_dig_node = function(pos, oldnode)
|
|
Cable:after_dig_node(pos)
|
|
Solar:after_dig_node(pos)
|
|
techage.del_mem(pos)
|
|
end,
|
|
|
|
on_receive_fields = on_receive_fields,
|
|
on_rightclick = on_rightclick,
|
|
on_timer = node_timer,
|
|
get_generator_data = get_generator_data,
|
|
ta3_formspec = techage.generator_settings("ta4", PWR_PERF)
|
|
})
|
|
|
|
power.register_nodes({"techage:ta4_solar_inverter"}, Cable, "gen", {"R"})
|
|
power.register_nodes({"techage:ta4_solar_inverter"}, Solar, "con", {"L"})
|
|
|
|
techage.register_node({"techage:ta4_solar_inverter"}, {
|
|
on_recv_message = function(pos, src, topic, payload)
|
|
local nvm = techage.get_nvm(pos)
|
|
if topic == "delivered" then
|
|
return math.floor((nvm.provided or 0) + 0.5)
|
|
else
|
|
return State:on_receive_message(pos, topic, payload)
|
|
end
|
|
end,
|
|
on_beduino_receive_cmnd = function(pos, src, topic, payload)
|
|
return State:on_beduino_receive_cmnd(pos, topic, payload)
|
|
end,
|
|
on_beduino_request_data = function(pos, src, topic, payload)
|
|
local nvm = techage.get_nvm(pos)
|
|
if topic == 135 then -- Delivered Power
|
|
return 0, {math.floor((nvm.provided or 0) + 0.5)}
|
|
else
|
|
return State:on_beduino_request_data(pos, topic, payload)
|
|
end
|
|
end,
|
|
})
|
|
|
|
control.register_nodes({"techage:ta4_solar_inverter"}, {
|
|
on_receive = function(pos, tlib2, topic, payload)
|
|
end,
|
|
on_request = function(pos, tlib2, topic)
|
|
if topic == "info" then
|
|
local nvm = techage.get_nvm(pos)
|
|
local meta = M(pos)
|
|
return {
|
|
type = S("TA4 Solar Inverter"),
|
|
number = meta:get_string("node_number") or "",
|
|
running = techage.is_running(nvm) or false,
|
|
available = nvm.max_power or 0,
|
|
provided = nvm.provided or 0,
|
|
termpoint = meta:get_string("termpoint"),
|
|
}
|
|
end
|
|
return false
|
|
end,
|
|
}
|
|
)
|
|
|
|
minetest.register_craft({
|
|
output = "techage:ta4_solar_inverter",
|
|
recipe = {
|
|
{'default:steel_ingot', 'dye:green', 'default:steel_ingot'},
|
|
{'', 'techage:ta4_wlanchip', 'techage:electric_cableS'},
|
|
{'default:steel_ingot', "techage:baborium_ingot", 'default:steel_ingot'},
|
|
},
|
|
})
|
|
|
|
techage.register_node_for_v1_transition({"techage:ta4_solar_inverter"}, function(pos, node)
|
|
power.update_network(pos, nil, Solar)
|
|
end)
|