2018-01-13 15:19:06 +01:00
|
|
|
local update_interval = 0.2
|
|
|
|
local level_delta = 2
|
2018-05-08 16:52:08 +02:00
|
|
|
local shiny_items = {}
|
2018-01-10 10:04:36 +01:00
|
|
|
|
2018-05-08 16:52:08 +02:00
|
|
|
--- Shining API ---
|
2018-01-13 15:19:06 +01:00
|
|
|
wielded_light = {}
|
|
|
|
|
|
|
|
function wielded_light.update_light(pos, light_level)
|
2018-09-04 22:13:37 +02:00
|
|
|
local around_vector = {
|
|
|
|
{x=0, y=0, z=0},
|
|
|
|
{x=0, y=1, z=0}, {x=0, y=-1, z=0},
|
|
|
|
{x=1, y=0, z=0}, {x=-1, y=0, z=0},
|
|
|
|
{x=0, y=0, z=1}, {x=0, y=0, z=1},
|
|
|
|
}
|
2018-01-13 15:19:06 +01:00
|
|
|
local do_update = false
|
|
|
|
local old_value = 0
|
|
|
|
local timer
|
2018-09-04 22:13:37 +02:00
|
|
|
local light_pos
|
|
|
|
for _, around in ipairs(around_vector) do
|
|
|
|
light_pos = vector.add(pos, around)
|
|
|
|
local name = minetest.get_node(light_pos).name
|
|
|
|
if name == "air" and (minetest.get_node_light(light_pos) or 0) < light_level then
|
2018-01-13 15:19:06 +01:00
|
|
|
do_update = true
|
2018-09-04 22:13:37 +02:00
|
|
|
break
|
|
|
|
elseif name:sub(1,13) == "wielded_light" then -- Update existing light node and timer
|
|
|
|
old_value = tonumber(name:sub(15))
|
|
|
|
if light_level > old_value then
|
2018-01-13 15:19:06 +01:00
|
|
|
do_update = true
|
2018-09-04 22:13:37 +02:00
|
|
|
else
|
|
|
|
timer = minetest.get_node_timer(light_pos)
|
|
|
|
local elapsed = timer:get_elapsed()
|
|
|
|
if elapsed > (update_interval * 1.5) then
|
|
|
|
-- The timer is set to 3x update_interval
|
|
|
|
-- This node was not updated the last interval and may
|
|
|
|
-- is disabled before the next step
|
|
|
|
-- Therefore the light should be re-set to avoid flicker
|
|
|
|
do_update = true
|
|
|
|
end
|
2018-01-13 15:19:06 +01:00
|
|
|
end
|
2018-09-04 22:13:37 +02:00
|
|
|
break
|
2018-01-13 15:19:06 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
if do_update then
|
2018-09-04 22:13:37 +02:00
|
|
|
timer = timer or minetest.get_node_timer(light_pos)
|
2018-01-13 15:19:06 +01:00
|
|
|
if light_level ~= old_value then
|
2018-09-04 22:13:37 +02:00
|
|
|
minetest.swap_node(light_pos, {name = "wielded_light:"..light_level})
|
2018-01-13 15:19:06 +01:00
|
|
|
end
|
|
|
|
timer:start(update_interval*3)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-05-08 16:52:08 +02:00
|
|
|
function wielded_light.update_light_by_item(item, pos)
|
|
|
|
local stack = ItemStack(item)
|
|
|
|
local light_level = shiny_items[stack:get_name()]
|
|
|
|
local itemdef = stack:get_definition()
|
|
|
|
if not light_level and not itemdef then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
light_level = light_level or ((itemdef.light_source or 0) - level_delta)
|
|
|
|
|
|
|
|
if light_level > 0 then
|
|
|
|
wielded_light.update_light(pos, light_level)
|
|
|
|
end
|
|
|
|
end
|
2018-01-13 15:19:06 +01:00
|
|
|
|
|
|
|
function wielded_light.register_item_light(itemname, light_level)
|
|
|
|
shiny_items[itemname] = light_level
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2018-05-08 16:52:08 +02:00
|
|
|
-- Register helper nodes
|
2018-01-13 15:19:06 +01:00
|
|
|
for i=1, 14 do
|
2018-01-10 10:04:36 +01:00
|
|
|
minetest.register_node("wielded_light:"..i, {
|
|
|
|
drawtype = "airlike",
|
|
|
|
groups = {not_in_creative_inventory = 1},
|
|
|
|
walkable = false,
|
|
|
|
paramtype = "light",
|
|
|
|
sunlight_propagates = true,
|
|
|
|
light_source = i,
|
|
|
|
pointable = false,
|
|
|
|
buildable_to = true,
|
|
|
|
drops = {},
|
|
|
|
on_timer = function(pos, elapsed)
|
|
|
|
minetest.swap_node(pos, {name = "air"})
|
|
|
|
end,
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2018-01-13 15:19:06 +01:00
|
|
|
-- Wielded item shining globalstep
|
2018-01-10 10:04:36 +01:00
|
|
|
local timer = 0
|
|
|
|
minetest.register_globalstep(function(dtime)
|
|
|
|
timer = timer + dtime;
|
|
|
|
if timer < update_interval then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
timer = 0
|
|
|
|
|
|
|
|
for _, player in pairs(minetest.get_connected_players()) do
|
2018-05-12 14:33:52 -07:00
|
|
|
-- predict where the player will be the next time we place the light
|
|
|
|
-- assume that on average we're slightly past 1/2 of the next interval, hence 1.5
|
|
|
|
-- (since the scheduling is a bit behind)
|
|
|
|
-- experimentally this also works nicely
|
|
|
|
local pos = vector.add (
|
|
|
|
vector.add({x = 0, y = 1, z = 0}, vector.round(player:getpos())),
|
|
|
|
vector.round(vector.multiply(player:get_player_velocity(), update_interval * 1.5))
|
|
|
|
)
|
|
|
|
|
|
|
|
wielded_light.update_light_by_item(player:get_wielded_item(), pos)
|
2018-01-10 10:04:36 +01:00
|
|
|
end
|
|
|
|
end)
|
2018-01-13 15:19:06 +01:00
|
|
|
|
|
|
|
|
2018-05-08 16:52:08 +02:00
|
|
|
-- Dropped item on_step override
|
|
|
|
-- https://github.com/minetest/minetest/issues/6909
|
|
|
|
local builtin_item = minetest.registered_entities["__builtin:item"]
|
|
|
|
local item = { }
|
|
|
|
for k,v in pairs(builtin_item) do
|
|
|
|
item[k] = v
|
|
|
|
end
|
|
|
|
item.on_step = function(self, dtime)
|
|
|
|
builtin_item.on_step(self, dtime)
|
|
|
|
|
|
|
|
self.shining_timer = (self.shining_timer or 0) + dtime
|
|
|
|
if self.shining_timer >= update_interval then
|
|
|
|
self.shining_timer = 0
|
|
|
|
local pos = self.object:get_pos()
|
|
|
|
if pos then
|
|
|
|
wielded_light.update_light_by_item(self.itemstring, pos)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
minetest.register_entity(":__builtin:item", item)
|
|
|
|
|
|
|
|
|
2018-01-13 15:19:06 +01:00
|
|
|
---TEST
|
|
|
|
--wielded_light.register_item_light('default:dirt', 14)
|
2018-05-08 16:52:08 +02:00
|
|
|
|