116 lines
3.4 KiB
Lua
116 lines
3.4 KiB
Lua
-- LUALOCALS < ---------------------------------------------------------
|
|
local math, minetest, nodecore, pairs, table, type, vector
|
|
= math, minetest, nodecore, pairs, table, type, vector
|
|
local math_pi, table_remove
|
|
= math.pi, table.remove
|
|
-- LUALOCALS > ---------------------------------------------------------
|
|
|
|
local modname = minetest.get_current_modname()
|
|
|
|
local seethru = {}
|
|
minetest.after(0, function()
|
|
for name, def in pairs(minetest.registered_nodes) do
|
|
if (minetest.get_item_group(name, "witness_opaque") == 0)
|
|
and (
|
|
minetest.get_item_group(name, "witness_transparent") > 0
|
|
or def.sunlight_propagates
|
|
or (
|
|
def.paramtype == "light"
|
|
and def.alpha
|
|
and def.alpha > 0
|
|
and def.alpha < 255
|
|
)
|
|
|
|
) then
|
|
seethru[name] = true
|
|
end
|
|
end
|
|
end)
|
|
|
|
local metakey = modname .. "_witness"
|
|
local cache = {}
|
|
local function witnessdata(player)
|
|
local pname = player:get_player_name()
|
|
local meta = player:get_meta()
|
|
local data = cache[pname]
|
|
if not data then
|
|
data = meta:get_string(metakey) or ""
|
|
data = data and data ~= "" and minetest.deserialize(data)
|
|
or {queue = {}, lookup = {}}
|
|
cache[pname] = data
|
|
end
|
|
return data, function() return meta:set_string(metakey, minetest.serialize(data)) end
|
|
end
|
|
|
|
local function canwitnessnow(player, pos)
|
|
local ppos = player:get_pos()
|
|
ppos.y = ppos.y + player:get_properties().eye_height
|
|
|
|
local look = player:get_look_dir()
|
|
local targ = vector.normalize(vector.subtract(pos, ppos))
|
|
if vector.angle(look, targ) > math_pi / 8 then return end
|
|
|
|
local rp = vector.round(pos)
|
|
for pt in minetest.raycast(pos, ppos, false, true) do
|
|
if not pt.under then return end
|
|
if vector.equals(pt.under, rp) then return true end
|
|
local node = minetest.get_node(pt.under)
|
|
if not seethru[node.name] then return end
|
|
end
|
|
return true
|
|
end
|
|
|
|
local function witnesslater(player, pos, disc)
|
|
disc = nodecore.flatkeys(disc)
|
|
local data, save = witnessdata(player)
|
|
local newdata = {
|
|
node = minetest.get_node(pos).name,
|
|
stack = nodecore.stack_get(pos):get_name(),
|
|
disc = disc
|
|
}
|
|
local posstr = minetest.pos_to_string(pos)
|
|
local olddata = data.lookup[posstr]
|
|
if olddata and (olddata.node == newdata.node) and (olddata.stack
|
|
== newdata.stack) and (type(olddata.disc) == "table") then
|
|
for k in pairs(disc) do
|
|
olddata.disc[k] = true
|
|
end
|
|
else
|
|
data.queue[#data.queue + 1] = pos
|
|
while #data.queue > 100 do table_remove(data.queue, 1) end
|
|
data.lookup[posstr] = newdata
|
|
end
|
|
save()
|
|
end
|
|
|
|
function nodecore.witness(pos, disc, maxdist)
|
|
maxdist = maxdist or 16
|
|
for _, player in pairs(minetest.get_connected_players()) do
|
|
local ppos = player:get_pos()
|
|
if vector.distance(ppos, pos) <= maxdist then
|
|
if canwitnessnow(player, pos) then
|
|
nodecore.player_discover(player, disc, "witness:")
|
|
else
|
|
witnesslater(player, pos, disc)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function delayed_witness(pos, node, player)
|
|
local data, save = witnessdata(player)
|
|
local posstr = minetest.pos_to_string(pos)
|
|
local found = data.lookup[posstr]
|
|
if not found then return end
|
|
data.lookup[posstr] = nil
|
|
save()
|
|
node = node or minetest.get_node(pos)
|
|
if (found.node ~= node.name) or (nodecore.stack_get(pos):get_name()
|
|
~= found.stack) then return end
|
|
return nodecore.player_discover(player, found.disc, "witness:")
|
|
end
|
|
minetest.register_on_punchnode(delayed_witness)
|
|
nodecore.register_on_node_stare(function(player, pt)
|
|
delayed_witness(pt.under, minetest.get_node(pt.under), player)
|
|
end)
|