161 lines
4.1 KiB
Lua
Raw Normal View History

-- LUALOCALS < ---------------------------------------------------------
local math, minetest, nodecore, pairs, vector
= math, minetest, nodecore, pairs, vector
local math_random
= math.random
-- LUALOCALS > ---------------------------------------------------------
2019-03-05 23:05:56 -05:00
local optic_queue = {}
2019-03-07 00:35:27 -05:00
local function dirname(pos)
if pos.x > 0 then return "e" end
if pos.x < 0 then return "w" end
if pos.y > 0 then return "u" end
if pos.y < 0 then return "d" end
if pos.z > 0 then return "n" end
if pos.z < 0 then return "s" end
return ""
end
local function scan(pos, dir, max)
2019-03-05 23:05:56 -05:00
local p = {x = pos.x, y = pos.y, z = pos.z}
if (not max) or (max > 16) then max = 16 end
for _ = 1, max do
2019-03-05 23:05:56 -05:00
p = vector.add(p, dir)
local node = minetest.get_node(p)
2019-03-07 00:35:27 -05:00
if node.name == "ignore" then return false, node end
local def = minetest.registered_items[node.name] or {}
if not def.sunlight_propagates then return p, node end
if def.groups and def.groups.visinv then
local stack = nodecore.stack_get(p)
def = minetest.registered_items[stack:get_name()]
if def and def.type == "node" and not def.sunlight_propagates then
return p, node
end
end
2019-03-05 23:05:56 -05:00
end
end
2019-03-07 00:35:27 -05:00
local function scan_recv(pos, dir)
local hit, node = scan(pos, dir)
if not hit then return hit, node end
local data = minetest.get_meta(hit):get_string("nc_optics")
if data == "" then return end
dir = dirname(vector.multiply(dir, -1))
if not minetest.deserialize(data)[dir] then return end
return hit, node
2019-03-05 23:05:56 -05:00
end
2019-03-07 00:35:27 -05:00
local function optic_check(pos)
optic_queue[minetest.hash_node_position(pos)] = pos
end
2019-03-07 00:35:27 -05:00
nodecore.optic_check = optic_check
local function optic_trigger(start, dir, max)
local pos, node = scan(start, dir, max)
2019-03-07 01:40:26 -05:00
if not node then return end
2019-03-07 00:35:27 -05:00
local def = minetest.registered_items[node.name] or {}
if def and def.optic_check then return optic_check(pos) end
end
local function optic_process(trans, pos)
local node = minetest.get_node(pos)
2019-03-24 19:52:14 -04:00
if node.name == "ignore" then return end
local def = minetest.registered_items[node.name] or {}
2019-03-24 18:24:37 -04:00
local ignored
2019-03-07 00:35:27 -05:00
if def and def.optic_check then
local func = function(dir)
local hit, hnode = scan_recv(pos, dir)
2019-03-24 18:24:37 -04:00
ignored = ignored or hit == false
return hit, hnode
2019-03-07 00:35:27 -05:00
end
2019-03-24 18:24:37 -04:00
local nn, res = def.optic_check(pos, node, func, def)
2019-03-24 23:15:25 -04:00
if (not ignored) and nn then
2019-03-12 20:30:59 -04:00
trans[minetest.hash_node_position(pos)] = {
pos = pos,
nn = nn,
data = res
2019-03-12 20:30:59 -04:00
}
2019-03-07 00:35:27 -05:00
end
end
end
2019-03-12 20:30:59 -04:00
local function optic_commit(v)
local meta = minetest.get_meta(v.pos)
local old = meta:get_string("nc_optics")
old = old and (old ~= "") and minetest.deserialize(old) or {}
local updated
2019-03-12 20:30:59 -04:00
local node = minetest.get_node(v.pos)
if node.name ~= v.nn then
node.name = v.nn
minetest.set_node(v.pos, node)
updated = true
2019-03-12 20:30:59 -04:00
end
local data = {}
for _, vv in pairs(v.data or {}) do
data[dirname(vv)] = 1
2019-03-12 20:30:59 -04:00
end
local dirty
2019-03-12 20:30:59 -04:00
for _, dir in pairs(nodecore.dirs()) do
local dn = dirname(dir)
if old[dn] ~= data[dn] then
optic_trigger(v.pos, dir)
dirty = true
elseif updated then
optic_trigger(v.pos, dir, 1)
end
2019-03-12 20:30:59 -04:00
end
if dirty then
meta:set_string("nc_optics", minetest.serialize(data))
end
2019-03-12 20:30:59 -04:00
end
local passive_queue = {}
minetest.register_abm({
label = "optic check",
interval = 1,
chance = 1,
nodenames = {"group:optic_check"},
action = function(pos)
passive_queue[#passive_queue + 1] = pos
end
})
local passive_batch = {}
nodecore.register_globalstep("optic check", function()
local batch = optic_queue
optic_queue = {}
2019-03-07 00:35:27 -05:00
if #passive_queue > 0 then
passive_batch = passive_queue
passive_queue = {}
for i = 1, #passive_batch do
local j = math_random(1, #passive_batch)
local t = passive_batch[i]
passive_batch[i] = passive_batch[j]
passive_batch[j] = t
end
end
local max = 25 - #batch
if max < 5 then max = 5 end
if max > #passive_batch then max = #passive_batch end
for _ = 1, max do
local pos = passive_batch[#passive_batch]
passive_batch[#passive_batch] = nil
batch[minetest.hash_node_position(pos)] = pos
end
2019-03-12 20:30:59 -04:00
local trans = {}
for _, pos in pairs(batch) do
optic_process(trans, pos)
end
2019-03-12 20:30:59 -04:00
for _, v in pairs(trans) do
optic_commit(v)
end
end)