forked from minetest-mods/nether
check for portal frames on the surface
When a surface level was known, only the volume_is_natural check was being performed, this fixes that. Also fixes bug where LBM didn't start the timer of example portals that had been disabled. Allows portal ignite to repair a stopped timer. Other misc cleanup - paramat's new Floatlands won't have lakes so I removed that from the flavortext.
This commit is contained in:
parent
48cbd87b59
commit
4985e199b1
@ -1036,12 +1036,14 @@ local function ignite_portal(ignition_pos, ignition_node_name)
|
|||||||
local anchorPos, orientation, is_ignited = is_within_portal_frame(portal_definition, ignition_pos)
|
local anchorPos, orientation, is_ignited = is_within_portal_frame(portal_definition, ignition_pos)
|
||||||
if anchorPos == nil then
|
if anchorPos == nil then
|
||||||
if DEBUG then minetest.chat_send_all("No " .. portal_definition.name .. " portal frame found at " .. minetest.pos_to_string(ignition_pos)) end
|
if DEBUG then minetest.chat_send_all("No " .. portal_definition.name .. " portal frame found at " .. minetest.pos_to_string(ignition_pos)) end
|
||||||
continue = true -- no portal is here, but perhaps there more than one portal type we need to search for
|
continue = true -- no portal is here, but perhaps there's more than one portal type we need to search for
|
||||||
elseif is_ignited then
|
elseif is_ignited then
|
||||||
|
-- Found a portal, check its metadata and timer is healthy.
|
||||||
local repair = false
|
local repair = false
|
||||||
local meta = minetest.get_meta(ignition_pos)
|
local meta = minetest.get_meta(ignition_pos)
|
||||||
if meta ~= nil then
|
if meta ~= nil then
|
||||||
if meta:get_string("p1") == "" then
|
local p1, p2, target = meta:get_string("p1"), meta:get_string("p2"), meta:get_string("target")
|
||||||
|
if p1 == "" or p2 == "" or target == "" then
|
||||||
-- metadata is missing, the portal frame node must have been removed without calling
|
-- metadata is missing, the portal frame node must have been removed without calling
|
||||||
-- on_destruct - perhaps by an ABM, then replaced - presumably by a player.
|
-- on_destruct - perhaps by an ABM, then replaced - presumably by a player.
|
||||||
-- allowing reigniting will repair the portal
|
-- allowing reigniting will repair the portal
|
||||||
@ -1049,9 +1051,19 @@ local function ignite_portal(ignition_pos, ignition_node_name)
|
|||||||
repair = true
|
repair = true
|
||||||
else
|
else
|
||||||
if DEBUG then minetest.chat_send_all("This portal links to " .. meta:get_string("target") .. ". p1=" .. meta:get_string("p1") .. " p2=" .. meta:get_string("p2")) end
|
if DEBUG then minetest.chat_send_all("This portal links to " .. meta:get_string("target") .. ". p1=" .. meta:get_string("p1") .. " p2=" .. meta:get_string("p2")) end
|
||||||
|
|
||||||
|
-- Check the portal's timer is running, and fix if it's not.
|
||||||
|
-- A portal's timer can stop running if the game is played without that portal type being
|
||||||
|
-- registered, e.g. enabling one of the example portals then later disabling it, then enabling it again.
|
||||||
|
-- (if this is a frequent problem, then change the value of "run_at_every_load" in the lbm)
|
||||||
|
local timer = minetest.get_node_timer(get_timerPos_from_p1_and_p2(minetest.string_to_pos(p1), minetest.string_to_pos(p2)))
|
||||||
|
if timer ~= nil and timer:get_timeout() == 0 then
|
||||||
|
if DEBUG then minetest.chat_send_all("Portal timer was not running: restarting the timer.") end
|
||||||
|
timer:start(1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if not repair then return false end -- portal is already ignited
|
end
|
||||||
|
if not repair then return false end -- portal is already ignited (or timer has been fixed)
|
||||||
end
|
end
|
||||||
|
|
||||||
if continue == false then
|
if continue == false then
|
||||||
@ -1508,13 +1520,14 @@ minetest.register_lbm({
|
|||||||
local meta = minetest.get_meta(pos)
|
local meta = minetest.get_meta(pos)
|
||||||
if meta ~= nil then
|
if meta ~= nil then
|
||||||
p1 = minetest.string_to_pos(meta:get_string("p1"))
|
p1 = minetest.string_to_pos(meta:get_string("p1"))
|
||||||
p2 = minetest.string_to_pos(meta:get_string("p1"))
|
p2 = minetest.string_to_pos(meta:get_string("p2"))
|
||||||
end
|
end
|
||||||
if p1 ~= nil and p2 ~= nil then
|
if p1 ~= nil and p2 ~= nil then
|
||||||
local timerPos = get_timerPos_from_p1_and_p2(p1, p2)
|
local timerPos = get_timerPos_from_p1_and_p2(p1, p2)
|
||||||
local timer = minetest.get_node_timer(timerPos)
|
local timer = minetest.get_node_timer(timerPos)
|
||||||
if timer ~= nil then
|
if timer ~= nil then
|
||||||
timer:start(1)
|
timer:start(1)
|
||||||
|
if DEBUG then minetest.chat_send_all("LBM started portal timer " .. minetest.pos_to_string(timerPos)) end
|
||||||
elseif DEBUG then
|
elseif DEBUG then
|
||||||
minetest.chat_send_all("get_node_timer" .. minetest.pos_to_string(timerPos) .. " returned null")
|
minetest.chat_send_all("get_node_timer" .. minetest.pos_to_string(timerPos) .. " returned null")
|
||||||
end
|
end
|
||||||
@ -1743,35 +1756,32 @@ end
|
|||||||
-- portal_name is optional, providing it allows existing portals on the surface to be reused.
|
-- portal_name is optional, providing it allows existing portals on the surface to be reused.
|
||||||
function nether.find_surface_target_y(target_x, target_z, portal_name)
|
function nether.find_surface_target_y(target_x, target_z, portal_name)
|
||||||
|
|
||||||
|
-- default to starting the search at -16 (probably underground) if we don't know the
|
||||||
|
-- surface, like paramat's original code from before get_spawn_level() was available:
|
||||||
|
-- https://github.com/minetest-mods/nether/issues/5#issuecomment-506983676
|
||||||
|
local start_y = -16
|
||||||
|
|
||||||
-- try to spawn on surface first
|
-- try to spawn on surface first
|
||||||
if minetest.get_spawn_level ~= nil then -- older versions of Minetest don't have this
|
if minetest.get_spawn_level ~= nil then -- older versions of Minetest don't have this
|
||||||
local surface_level = minetest.get_spawn_level(target_x, target_z)
|
surface_level = minetest.get_spawn_level(target_x, target_z)
|
||||||
if surface_level ~= nil then
|
if surface_level ~= nil then -- test this since get_spawn_level() can return nil over water or steep/high terrain
|
||||||
|
|
||||||
-- get_spawn_level() tends to err on the side of caution and sometimes spawn the player a
|
-- get_spawn_level() tends to err on the side of caution and spawns the player a
|
||||||
-- block higher than the ground level. The implementation is mapgen specific
|
-- block higher than the ground level. The implementation is mapgen specific
|
||||||
-- and -2 seems to be the right amount for v6, v5, carpathian, valleys, and flat,
|
-- and -2 seems to be the right correction for v6, v5, carpathian, valleys, and flat,
|
||||||
-- but v7 only needs -1.
|
-- but v7 only needs -1.
|
||||||
-- Perhaps this was not always the case, and -2 may be too much in older versions
|
-- Perhaps this was not always the case, and -2 may be too much in older versions
|
||||||
-- of minetest, but half-buried portals are perferable to floating ones, and they
|
-- of minetest, but half-buried portals are perferable to floating ones, and they
|
||||||
-- will clear a suitable hole around them.
|
-- will clear a suitable hole around themselves.
|
||||||
if minetest.get_mapgen_setting("mg_name") == "v7" then
|
if minetest.get_mapgen_setting("mg_name") == "v7" then
|
||||||
surface_level = surface_level - 1
|
surface_level = surface_level - 1
|
||||||
else
|
else
|
||||||
surface_level = surface_level - 2
|
surface_level = surface_level - 2
|
||||||
end
|
end
|
||||||
|
start_y = surface_level
|
||||||
-- Check volume for non-natural nodes
|
|
||||||
local minp = {x = target_x - 1, y = surface_level - 1, z = target_z - 2}
|
|
||||||
local maxp = {x = target_x + 2, y = surface_level + 3, z = target_z + 2}
|
|
||||||
if nether.volume_is_natural(minp, maxp) then
|
|
||||||
return surface_level
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
-- fallback to underground search
|
|
||||||
local start_y = -16
|
|
||||||
for y = start_y, start_y - 256, -16 do
|
for y = start_y, start_y - 256, -16 do
|
||||||
-- Check volume for non-natural nodes
|
-- Check volume for non-natural nodes
|
||||||
local minp = {x = target_x - 1, y = y - 1, z = target_z - 2}
|
local minp = {x = target_x - 1, y = y - 1, z = target_z - 2}
|
||||||
@ -1783,6 +1793,7 @@ function nether.find_surface_target_y(target_x, target_z, portal_name)
|
|||||||
-- but reigniting existing portals in portal rooms is fine - desirable even.
|
-- but reigniting existing portals in portal rooms is fine - desirable even.
|
||||||
local anchorPos, orientation, is_ignited = is_within_portal_frame(nether.registered_portals[portal_name], {x = target_x, y = y, z = target_z})
|
local anchorPos, orientation, is_ignited = is_within_portal_frame(nether.registered_portals[portal_name], {x = target_x, y = y, z = target_z})
|
||||||
if anchorPos ~= nil then
|
if anchorPos ~= nil then
|
||||||
|
if DEBUG then minetest.chat_send_all("Volume_is_natural check failed, but a portal frame is here " .. minetest.pos_to_string(anchorPos) .. ", so this is still a good target y level") end
|
||||||
return y
|
return y
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -53,7 +53,7 @@ if minetest.settings:get_bool("nether_enable_portal_example_floatlands", ENABLE
|
|||||||
FLOATLAND_LEVEL = minetest.get_mapgen_setting("mgv7_floatland_level") or 1280
|
FLOATLAND_LEVEL = minetest.get_mapgen_setting("mgv7_floatland_level") or 1280
|
||||||
|
|
||||||
if FLOATLANDS_ENABLED then
|
if FLOATLANDS_ENABLED then
|
||||||
floatlands_flavortext = "There is a floating land of hills and lakes and forests up there, the edges of which lead to a drop all the way back down to the surface. We have not found how far these strange lands extend. One day I may retire here."
|
floatlands_flavortext = S("There is a floating land of hills and forests up there, over the edges of which is a perilous drop all the way back down to sea level. We have not found how far these strange pristine lands extend. I have half a mind to retire there one day.")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -202,10 +202,8 @@ if minetest.settings:get_bool("nether_enable_portal_example_surfacetravel", ENA
|
|||||||
for attempt = 1, attemptLimit do
|
for attempt = 1, attemptLimit do
|
||||||
adj_x = math.floor(prng:rand_normal_dist(-search_radius, search_radius, 2) + 0.5)
|
adj_x = math.floor(prng:rand_normal_dist(-search_radius, search_radius, 2) + 0.5)
|
||||||
adj_z = math.floor(prng:rand_normal_dist(-search_radius, search_radius, 2) + 0.5)
|
adj_z = math.floor(prng:rand_normal_dist(-search_radius, search_radius, 2) + 0.5)
|
||||||
minetest.chat_send_all(attempt .. ": x " .. target_x + adj_x .. ", z " .. target_z + adj_z)
|
|
||||||
if minetest.get_spawn_level(target_x + adj_x, target_z + adj_z) ~= nil then
|
if minetest.get_spawn_level(target_x + adj_x, target_z + adj_z) ~= nil then
|
||||||
-- found a location which will be at ground level (unless a player has built there)
|
-- found a location which will be at ground level (unless a player has built there)
|
||||||
minetest.chat_send_all("x " .. target_x + adj_x .. ", z " .. target_z + adj_z .. " is suitable. Within " .. search_radius .. " of " .. target_x .. ", " .. target_z)
|
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
Reference in New Issue
Block a user