2021-08-30 19:34:20 -04:00

134 lines
4.1 KiB
Lua

--[[
Copyright (C) 2021 Jude Melton-Houghton
This file is part of area_containers. It implements mesecons functionality.
area_containers is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
area_containers is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with area_containers. If not, see <https://www.gnu.org/licenses/>.
]]
--[[
OVERVIEW
Port nodes inside the chamber correspond to faces of the container.
A Mesecons signal can conduct between the horizontal container faces and
the ports. The port nodes have the same param1 and param2 as the container
node for the purpose of conductance. NOTE: Port nodes are assumed to be on
the -X side of the chamber.
See also container.lua and nodes.lua.
]]
local use = ...
local ALL_CONTAINER_STATES, PORT_NAME_PREFIX,
PORT_OFFSETS, PORT_IDS_HORIZ, get_port_id_from_name,
MESECON_STATE_ON, MESECON_STATE_OFF, vec2table = use("misc", {
"ALL_CONTAINER_STATES", "PORT_NAME_PREFIX",
"PORT_OFFSETS", "PORT_IDS_HORIZ", "get_port_id_from_name",
"MESECON_STATE_ON", "MESECON_STATE_OFF", "vec2table",
})
local get_related_container, get_related_inside = use("relation", {
"get_related_container", "get_related_inside"
})
local exports = {}
exports.container = {}
exports.ports = {}
-- A container is a conductor to its insides. The position of its insides can
-- be determined from param1 and param2.
exports.container.mesecons = {conductor = {
states = ALL_CONTAINER_STATES,
}}
local function container_rules_add_port(rules, port_id, self_pos, inside_pos)
local port_pos = vector.add(inside_pos, PORT_OFFSETS[port_id])
local offset_to_port = vector.subtract(port_pos, self_pos)
rules[#rules + 1] = vec2table(offset_to_port)
end
function exports.container.mesecons.conductor.rules(node)
local rules = {
{
{x = 1, y = 1, z = 0},
{x = 1, y = 0, z = 0},
{x = 1, y = -1, z = 0},
},
{
{x = -1, y = 1, z = 0},
{x = -1, y = 0, z = 0},
{x = -1, y = -1, z = 0},
},
{
{x = 0, y = 1, z = 1},
{x = 0, y = 0, z = 1},
{x = 0, y = -1, z = 1},
},
{
{x = 0, y = 1, z = -1},
{x = 0, y = 0, z = -1},
{x = 0, y = -1, z = -1},
},
}
local self_pos = get_related_container(node.param1, node.param2)
if self_pos then
local inside_pos = get_related_inside(node.param1, node.param2)
container_rules_add_port(rules[1], "px", self_pos, inside_pos)
container_rules_add_port(rules[2], "nx", self_pos, inside_pos)
container_rules_add_port(rules[3], "pz", self_pos, inside_pos)
container_rules_add_port(rules[4], "nz", self_pos, inside_pos)
end
return rules
end
-- The ports conduct in a similar way to the container, using param1 and param2.
local function get_port_rules(node)
local rules = {
{x = 1, y = -1, z = 0},
{x = 1, y = 0, z = 0},
{x = 1, y = 1, z = 0},
}
local container_pos = get_related_container(node.param1, node.param2)
if container_pos then
local id = get_port_id_from_name(node.name)
local inside_pos = get_related_inside(node.param1, node.param2)
local self_pos = vector.add(inside_pos, PORT_OFFSETS[id])
local container_offset =
vector.subtract(container_pos, self_pos)
rules[#rules + 1] = vec2table(container_offset)
end
return rules
end
-- mesecons information for port nodes that have it, with node names as keys.
for _, id in ipairs(PORT_IDS_HORIZ) do
local on_state = PORT_NAME_PREFIX .. id .. "_on"
local off_state = PORT_NAME_PREFIX .. id .. "_off"
exports.ports[on_state] = {
mesecons = {conductor = {
state = MESECON_STATE_ON,
offstate = off_state,
rules = get_port_rules,
}},
}
exports.ports[off_state] = {
mesecons = {conductor = {
state = MESECON_STATE_OFF,
onstate = on_state,
rules = get_port_rules,
}},
}
end
return exports