mirror of
https://git.minetest.land/Mineclonia/Mineclonia.git
synced 2024-11-09 19:25:05 +00:00
161 lines
5.2 KiB
Lua
161 lines
5.2 KiB
Lua
-- Dig and place services
|
|
|
|
mesecon.on_placenode = function(pos, node)
|
|
mesecon.execute_autoconnect_hooks_now(pos, node)
|
|
|
|
-- Receptors: Send on signal when active
|
|
if mesecon.is_receptor_on(node.name) then
|
|
mesecon.receptor_on(pos, mesecon.receptor_get_rules(node))
|
|
end
|
|
|
|
-- Conductors: Send turnon signal when powered or replace by respective offstate conductor
|
|
-- if placed conductor is an onstate one
|
|
if mesecon.is_conductor(node.name) then
|
|
local sources = mesecon.is_powered(pos)
|
|
if sources then
|
|
-- also call receptor_on if itself is powered already, so that neighboring
|
|
-- conductors will be activated (when pushing an on-conductor with a piston)
|
|
for _, s in ipairs(sources) do
|
|
local rule = vector.subtract(pos, s)
|
|
mesecon.turnon(pos, rule)
|
|
end
|
|
mesecon.receptor_on (pos, mesecon.conductor_get_rules(node))
|
|
elseif mesecon.is_conductor_on(node) then
|
|
minetest.swap_node(pos, {name = mesecon.get_conductor_off(node)})
|
|
end
|
|
end
|
|
|
|
-- Effectors: Send changesignal and activate or deactivate
|
|
if mesecon.is_effector(node.name) then
|
|
local powered_rules = {}
|
|
local unpowered_rules = {}
|
|
|
|
-- for each input rule, check if powered
|
|
for _, r in ipairs(mesecon.effector_get_rules(node)) do
|
|
local powered = mesecon.is_powered(pos, r)
|
|
if powered then table.insert(powered_rules, r)
|
|
else table.insert(unpowered_rules, r) end
|
|
|
|
local state = powered and mesecon.state.on or mesecon.state.off
|
|
mesecon.changesignal(pos, node, r, state, 1)
|
|
end
|
|
|
|
if (#powered_rules > 0) then
|
|
for _, r in ipairs(powered_rules) do
|
|
mesecon.activate(pos, node, r, 1)
|
|
end
|
|
else
|
|
for _, r in ipairs(unpowered_rules) do
|
|
mesecon.deactivate(pos, node, r, 1)
|
|
end
|
|
end
|
|
end
|
|
|
|
if minetest.get_item_group(node.name, "opaque") == 1 then
|
|
local neighbors = mesecon.mcl_get_neighbors(pos)
|
|
local is_powered, direct_source = mesecon.is_powered(pos)
|
|
if is_powered and direct_source then
|
|
for n=1, #neighbors do
|
|
local npos = neighbors[n].pos
|
|
local nnode = minetest.get_node(npos)
|
|
if mesecon.is_conductor_off(nnode) then
|
|
mesecon.receptor_on(npos, mesecon.conductor_get_rules(nnode))
|
|
-- Redstone torch is a special case and must be ignored
|
|
elseif mesecon.is_effector_on(nnode.name) and minetest.get_item_group(nnode.name, "redstone_torch") == 0 then
|
|
mesecon.changesignal(npos, nnode, neighbors[n].link, mesecon.state.on, 1)
|
|
mesecon.activate(npos, nnode, neighbors[n].link, 1)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
mesecon.on_dignode = function(pos, node)
|
|
if mesecon.is_conductor_on(node) then
|
|
mesecon.receptor_off(pos, mesecon.conductor_get_rules(node))
|
|
elseif mesecon.is_receptor_on(node.name) then
|
|
mesecon.receptor_off(pos, mesecon.receptor_get_rules(node))
|
|
end
|
|
if minetest.get_item_group(node.name, "opaque") == 1 then
|
|
local sources = mesecon.is_powered(pos)
|
|
local neighbors = mesecon.mcl_get_neighbors(pos)
|
|
for n=1, #neighbors do
|
|
local npos = neighbors[n].pos
|
|
local nlink = neighbors[n].link
|
|
local nnode = minetest.get_node(npos)
|
|
if mesecon.is_conductor_on(nnode) then
|
|
mesecon.receptor_off(npos, mesecon.conductor_get_rules(nnode))
|
|
-- Disable neighbor effectors unless they are in a special ignore group
|
|
elseif mesecon.is_effector_on(nnode.name) and mesecon.is_powered(npos) == false and minetest.get_item_group(nnode.name, "mesecon_ignore_opaque_dig") == 0 then
|
|
mesecon.changesignal(npos, nnode, nlink, mesecon.state.off, 1)
|
|
mesecon.deactivate(npos, nnode, nlink, 1)
|
|
end
|
|
end
|
|
end
|
|
mesecon.execute_autoconnect_hooks_queue(pos, node)
|
|
end
|
|
|
|
minetest.register_on_placenode(mesecon.on_placenode)
|
|
minetest.register_on_dignode(mesecon.on_dignode)
|
|
|
|
-- Overheating service for fast circuits
|
|
local OVERHEAT_MAX = mesecon.setting("overheat_max", 8)
|
|
local COOLDOWN_TIME = mesecon.setting("cooldown_time", 3.0)
|
|
local COOLDOWN_STEP = mesecon.setting("cooldown_granularity", 0.5)
|
|
local COOLDOWN_MULTIPLIER = OVERHEAT_MAX / COOLDOWN_TIME
|
|
local cooldown_timer = 0.0
|
|
local object_heat = {}
|
|
|
|
-- returns true if heat is too high
|
|
function mesecon.do_overheat(pos)
|
|
local id = minetest.hash_node_position(pos)
|
|
local heat = (object_heat[id] or 0) + 1
|
|
object_heat[id] = heat
|
|
if heat >= OVERHEAT_MAX then
|
|
minetest.log("action", "Node overheats at " .. minetest.pos_to_string(pos))
|
|
object_heat[id] = nil
|
|
return true
|
|
end
|
|
return false
|
|
end
|
|
|
|
function mesecon.do_cooldown(pos)
|
|
local id = minetest.hash_node_position(pos)
|
|
object_heat[id] = nil
|
|
end
|
|
|
|
function mesecon.get_heat(pos)
|
|
local id = minetest.hash_node_position(pos)
|
|
return object_heat[id] or 0
|
|
end
|
|
|
|
function mesecon.move_hot_nodes(moved_nodes)
|
|
local new_heat = {}
|
|
for _, n in ipairs(moved_nodes) do
|
|
local old_id = minetest.hash_node_position(n.oldpos)
|
|
local new_id = minetest.hash_node_position(n.pos)
|
|
new_heat[new_id] = object_heat[old_id]
|
|
object_heat[old_id] = nil
|
|
end
|
|
for id, heat in pairs(new_heat) do
|
|
object_heat[id] = heat
|
|
end
|
|
end
|
|
|
|
local function global_cooldown(dtime)
|
|
cooldown_timer = cooldown_timer + dtime
|
|
if cooldown_timer < COOLDOWN_STEP then
|
|
return -- don't overload the CPU
|
|
end
|
|
local cooldown = COOLDOWN_MULTIPLIER * cooldown_timer
|
|
cooldown_timer = 0
|
|
for id, heat in pairs(object_heat) do
|
|
heat = heat - cooldown
|
|
if heat <= 0 then
|
|
object_heat[id] = nil -- free some RAM
|
|
else
|
|
object_heat[id] = heat
|
|
end
|
|
end
|
|
end
|
|
minetest.register_globalstep(global_cooldown)
|