VoxeLibre/mods/ITEMS/mcl_portals/portal_nether.lua
2025-01-01 14:43:47 +01:00

1635 lines
54 KiB
Lua

local modname = minetest.get_current_modname()
local S = minetest.get_translator(modname)
local modpath = minetest.get_modpath(modname)
-- Localize functions for better performance
local abs = math.abs
local ceil = math.ceil
local floor = math.floor
local max = math.max
local min = math.min
local random = math.random
local dist = vector.distance
local add = vector.add
local mul = vector.multiply
local sub = vector.subtract
local log = function(level, message)
minetest.log(level, string.format("[mcl_portals] %s", message))
end
-- Resources
-- Issue that has a lot of context: https://git.minetest.land/VoxeLibre/VoxeLibre/issues/4120
-- Minecraft portal mechanics: https://minecraft.fandom.com/wiki/Tutorials/Nether_portals
-- Flow diagram: https://docs.google.com/drawings/d/1WIl4pVuxgOxI3Ncxk4g6D1pL4Fyll3bQ-fX6L9yyiLw/edit
-- Useful boundaries: https://git.minetest.land/VoxeLibre/VoxeLibre/wiki/World-structure%3A-positions%2C-boundaries%2C-blocks%2C-chunks%2C-dimensions%2C-barriers-and-the-void
-- Setup
-- === CAUTION ===
-- the following values: SEARCH_DISTANCE_OVERWORLD, SEARCH_DISTANCE_NETHER,
-- BUILD_DISTANCE_XY, W_MAX, and NETHER_COMPRESSION have been set to together
-- guarantee that standard 2-wide portals will never split into two exits.
-- Splitting will still occur rarely for portals in the overworld wider than 2,
-- will still be likely for portals in the nether wider than 16, and guaranteed
-- for portals in the nether wider than 18.
-- Changing one value without changing the others might have uninteded
-- consequences. You have been warned :-)
-- Distance compression factor for the nether. If you change this, you might
-- want to recalculate SEARCH_DISTANCE_* and W_MAX too.
local NETHER_COMPRESSION = 8
-- Maximum distance from the ideal build spot where active parts of portals are
-- allowed to be placed.
-- There is a tradeoff here between the "walking" distance (distance to walk in
-- the overworld to get a new nether portal exit, which we want to be as similar
-- to Minecraft as possible, which is 136 [16*8+8]), and the area available for
-- exit search (which we want to maximise).
-- For our mapgen performance reasons, our search area is clipped by chunk, so
-- in the unlucky corner case the worst-case build area could be a quarter of
-- the expected size.
-- For MC build distance of 16, which gives a build area of 1,089 X-Z blocks
-- [(16+16+1)*(16+16+1)]. To guarantee this area here, we'd need to pick a build
-- distance of 32 [(32+32+1)*(32+32+1)/4]. But build distance of 32 implies
-- walking distance of 264 [32*8+8], which is already quite far, so need to pick
-- the right tradeoff:
--
-- Build dist Minimum build area Minimum walk distance
-- 48 2,401 392
-- 32 1,089 264
-- 24 625 200
-- 16 289 136
local BUILD_DISTANCE_XZ = 24
-- The following two values define distance to search existing portal exits for.
-- For context, Minecraft search is "8 chunks away" for the overworld (17
-- chunks centered on the ideal position), and "1 chunk away" for the nether (3
-- chunks centered on the ideal position).
-- To prevent portal splitting on spawned portals, we add one to the build
-- distance: spawned portals are 2-wide, so we need to make sure that we can
-- reach the second exit block (which can spawn in the direction away from the
-- player).
-- The search is boundary-inclusive, meaning for position 0 in the overworld,
-- search will be from -N to N.
-- If you change this, keep in mind our exits table keying divisor is 256, so
-- small changes might have outsize performance impact. At <=128, max 4 buckets
-- are searched, at 200 max 9 buckets are searched.
local SEARCH_DISTANCE_NETHER = BUILD_DISTANCE_XZ + 1 -- 25
local SEARCH_DISTANCE_OVERWORLD = SEARCH_DISTANCE_NETHER * NETHER_COMPRESSION -- 200
-- Limits of portal sizes (includes the frame)
local W_MIN, W_MAX = 4, 23
local H_MIN, H_MAX = 5, 23
-- Limits to active nodes (mcl_portals:portal)
local N_MIN, N_MAX = 6, (W_MAX-2) * (H_MAX-2)
local LIM_MIN, LIM_MAX = mcl_vars.mapgen_edge_min, mcl_vars.mapgen_edge_max
local PLAYER_COOLOFF, MOB_COOLOFF = 3, 14 -- for this many seconds they won't teleported again
local TOUCH_CHATTER_TIME = 1 -- prevent multiple teleportation attempts caused by multiple portal touches, for this number of seconds
local CHATTER_US = TOUCH_CHATTER_TIME * 1000000
local nether_portal_creative_delay = vl_tuning.setting("gamerule:playersNetherPortalCreativeDelay", "number", {
default = 0,
})
local nether_port_survival_delay = vl_tuning.setting("gamerule:playersNetherPortalDefaultDelay", "number", {
default = 4,
})
-- Speeds up the search by allowing some non-air nodes to be replaced when
-- looking for acceptable portal locations. Setting this lower means the
-- algorithm will do more searching. Even at 0, there is no risk of finding
-- nothing - the most airy location will be used as fallback anyway.
local ACCEPTABLE_PORTAL_REPLACES = 2
local PORTAL = "mcl_portals:portal"
local OBSIDIAN = "mcl_core:obsidian"
-- Dimension-specific Y boundaries for portal exit search. Portal search
-- algorithm will only ever look at two vertically offset chunks. It will
-- select whether the second chunk is up or down based on which side has more
-- "valid" Y-space (where valid is defined as space between *_Y_MIN and
-- *_Y_MAX).
-- For nether, selection of the boundaries doesn't matter, because it fits
-- entirely within two chunks.
local N_Y_MIN = mcl_vars.mg_lava_nether_max + 1
local N_Y_MAX = mcl_vars.mg_bedrock_nether_top_min
local N_Y_SPAN = N_Y_MAX - N_Y_MIN
-- Overworld however is much taller. Let's select the boundaries so that we
-- maximise the Y-space (so align with chunk boundary), and also pick an area
-- that has a good chance of having "find_nodes_in_area_under_air" return
-- something (so ideally caves, or surface, not just sky).
-- For the bottom bound, we try for the first chunk boundary in the negative Ys (-32).
local O_Y_MIN = max(mcl_vars.mg_lava_overworld_max + 1, mcl_vars.central_chunk_offset_in_nodes)
-- Since O_Y_MIN is also used as a base for converting coordinates, we need to
-- make sure the top bound is high enough to encompass entire nether height. In
-- v7 mapgen nether is flatter than overworld, so this results in a span of
-- exactly two chunks.
-- Due to the O_Y_MIN being used as a base, the preferred build locations will
-- be in the bottom part of the overworld range (in v7 around -32 to 61), and
-- the high locations will be used as a fallback. If we see too many
-- underground portals, we may need to shift just this base upwards (new setting).
local O_Y_SPAN = max(N_Y_SPAN, 2 * mcl_vars.chunk_size_in_nodes - 1)
local O_Y_MAX = min(mcl_vars.mg_overworld_max_official, O_Y_MIN + O_Y_SPAN)
log("verbose", string.format("N_Y_MIN=%.1f, N_Y_MAX=%.1f, O_Y_MIN=%.1f, O_Y_MAX=%.1f", N_Y_MIN, N_Y_MAX, O_Y_MIN, O_Y_MAX))
log("verbose", string.format("Nether span is %.1f, overworld span is %.1f", N_Y_MAX-N_Y_MIN+1, O_Y_MAX-O_Y_MIN+1))
-- Alpha and particles
local node_particles_allowed = minetest.settings:get("mcl_node_particles") or "none"
local node_particles_levels = { none=0, low=1, medium=2, high=3 }
local PARTICLES = node_particles_levels[node_particles_allowed]
-- Table of objects (including players) which recently teleported by a
-- Nether portal. Those objects have a brief cooloff period before they
-- can teleport again. This prevents annoying back-and-forth teleportation.
local cooloff = {}
function mcl_portals.nether_portal_cooloff(object)
return cooloff[object]
end
local chatter = {}
-- Searching for portal exits, finding locations and building portals can be time consuming.
-- We can queue origins together, on the assumption they will all go to the same place in the end.
local origin_queue = {}
-- At the same time, we never want to build two portals in the same area at the
-- same time, because they will interfere and we can end up with broken portals,
-- possibly stranding the player. We won't use queue here - double queueing can
-- lead to deadlocks. We will instead interrupt the portal building, and rely
-- on the ABM to try again later.
local chunk_building = {}
local storage = mcl_portals.storage
-- `exits` is a table storing portal exits (both nether or overworld) bucketed
-- by 256x256 areas (with each bucket containing many exits, and covering entire range of Y).
-- An exit is a location of ignited PORTAL block, with another PORTAL block above
-- and obsidian below. Thus each portal will register at minimum two exits, but more
-- for bigger portals up to the maximum of W_MAX-2.
-- This table should be maintained using `add_exits`, `remove_exits` and `find_exit`.
local exits = {}
local keys = minetest.deserialize(storage:get_string("nether_exits_keys") or "return {}") or {}
for _, key in pairs(keys) do
local n = tonumber(key)
if n then
exits[key] = minetest.deserialize(storage:get_string("nether_exits_"..key) or "return {}") or {}
end
end
local get_node = mcl_vars.get_node
local set_node = minetest.set_node
local registered_nodes = minetest.registered_nodes
local is_protected = minetest.is_protected
local find_nodes_in_area = minetest.find_nodes_in_area
local find_nodes_in_area_under_air = minetest.find_nodes_in_area_under_air
local pos_to_string = minetest.pos_to_string
local is_area_protected = minetest.is_area_protected
local get_us_time = minetest.get_us_time
local dimension_to_teleport = { nether = "overworld", overworld = "nether" }
local limits = {
nether = {
pmin = {x=LIM_MIN, y = N_Y_MIN, z = LIM_MIN},
pmax = {x=LIM_MAX, y = N_Y_MAX, z = LIM_MAX},
},
overworld = {
pmin = {x=LIM_MIN, y = O_Y_MIN, z = LIM_MIN},
pmax = {x=LIM_MAX, y = O_Y_MAX, z = LIM_MAX},
},
}
-- Deletes exit from this portal's node storage.
local function delete_portal_pos(pos)
local p1 = vector.offset(pos,-5,-1,-5)
local p2 = vector.offset(pos,5,5,5)
local nn = find_nodes_in_area(p1,p2,{"mcl_portals:portal"})
for _,p in pairs(nn) do
minetest.get_meta(p):set_string("target_portal","")
end
end
-- Gets exit for this portal from node storage. After Jan 2024 this is only used
-- for old portals, so that players don't get surprises. New portals, or portals that lost
-- node storage due to destruction should use the lookup table.
local function get_portal_pos(pos)
local p1 = vector.offset(pos,-5,-1,-5)
local p2 = vector.offset(pos,5,5,5)
local nn = find_nodes_in_area(p1,p2,{"mcl_portals:portal"})
for _,p in pairs(nn) do
local m = minetest.get_meta(p):get_string("target_portal")
if m and m ~= "" and mcl_vars.get_node(p).name == "mcl_portals:portal" then
return minetest.get_position_from_hash(m)
end
end
end
-- `exits` is a lookup table bucketing exits by 256x256 areas. This function
-- returns a key for provided position p.
local function get_exit_key(p)
local x, z = floor(p.x), floor(p.z)
return floor(z/256) * 256 + floor(x/256)
end
-- Add an exit to the exits table without writing to disk. Returns the exits
-- table key if modification was done, and specifies whether the key was new.
local function add_exit(p)
local retval = {key=false, new=false}
if not p or not p.y or not p.z or not p.x then
return retval
end
local x, y, z = floor(p.x), floor(p.y), floor(p.z)
local p = {x = x, y = y, z = z}
if get_node({x=x,y=y-1,z=z}).name ~= OBSIDIAN
or get_node(p).name ~= PORTAL
or get_node({x=x,y=y+1,z=z}).name ~= PORTAL
then
return retval
end
local k = get_exit_key(p)
if not exits[k] then
exits[k]={}
retval.new = true
end
local e = exits[k]
for i = 1, #e do
local t = e[i]
if t and t.x == p.x and t.y == p.y and t.z == p.z then
return retval
end
end
e[#e+1] = p
retval.key = k
return retval
end
-- This function registers one or more exits from Nether portals and writes
-- updated table buckets to disk.
-- Exit position must be an ignited PORTAL block that sits on top of obsidian,
-- and has additional PORTAL block above it.
-- This function will silently skip exits that are invalid during the call-time.
-- If the verification passes, a new exit is added to the table of exits and
-- saved to mod storage later. Duplicate exits will be skipped and won't cause
-- writes.
local function add_exits(positions)
local keys_to_write = {}
local new_key_present = false
for _, p in ipairs(positions) do
local r = add_exit(p)
if r.key ~= false then
if r.new then
new_key_present = true
keys[#keys+1] = r.key
end
keys_to_write[#keys_to_write+1] = r.key
log("verbose", "Exit added at " .. pos_to_string(p))
end
end
for _, key in ipairs(keys_to_write) do
storage:set_string("nether_exits_"..tostring(key), minetest.serialize(exits[key]))
end
if new_key_present then
storage:set_string("nether_exits_keys", minetest.serialize(keys))
end
end
-- Removes one portal exit from the exits table without writing to disk.
-- Returns the false or bucket key if change was made.
local function remove_exit(p)
if not p or not p.y or not p.z or not p.x then
return false
end
local x, y, z = floor(p.x), floor(p.y), floor(p.z)
local p = {x = x, y = y, z = z}
local k = get_exit_key(p)
if not exits[k] then
return false
end
local e = exits[k]
if e then
for i, t in pairs(e) do
if t and t.x == x and t.y == y and t.z == z then
e[i] = nil
return k
end
end
end
return false
end
-- Removes one or more portal exits and writes updated table buckets to disk.
local function remove_exits(positions)
local keys_to_write = {}
for _, p in ipairs(positions) do
r = remove_exit(p)
if r ~= false then
keys_to_write[#keys_to_write+1] = r
log("verbose", "Exit removed from " .. pos_to_string(p))
end
end
for _, key in ipairs(keys_to_write) do
storage:set_string("nether_exits_"..tostring(key), minetest.serialize(exits[key]))
end
end
-- Searches for portal exits nearby point p within the distances specified by dx
-- and dz (but only in the same dimension). Otherwise as in Minecraft, the
-- search is bounded by X and Z, but not Y.
-- If multiple exits are found, use Euclidean distance to find the nearest. This
-- uses all three coordinates.
local function find_exit(p, dx, dz)
local dim = mcl_worlds.pos_to_dimension(p)
if not p or not p.y or not p.z or not p.x then
log("warning", "Corrupt position passed to find_exit: "..pos_to_string(p)..".")
return
end
if dx < 1 or dz < 1 then return false end
local x = floor(p.x)
local z = floor(p.z)
local x1 = x-dx
local z1 = z-dz
local x2 = x+dx
local z2 = z+dz
-- Scan the relevant hash table keys for viable exits. Dimension's entire Y is scanned.
local k1x, k2x = floor(x1/256), floor(x2/256)
local k1z, k2z = floor(z1/256), floor(z2/256)
local nearest_exit, nearest_distance
for kx = k1x, k2x do
for kz = k1z, k2z do
local k = kz*256 + kx
local e = exits[k]
if e then
for _, t0 in pairs(e) do
if mcl_worlds.pos_to_dimension(t0) == dim then
-- exit is in the correct dimension
if abs(t0.x-p.x) <= dx and abs(t0.z-p.z) <= dz then
-- exit is within the search range
local d0 = dist(p, t0)
if not nearest_distance or nearest_distance>d0 then
-- it is the only exit so far, or it is the Euclidean-closest exit
nearest_distance = d0
nearest_exit = t0
if nearest_distance==0 then return nearest_exit end
end
end
end
end
end
end
end
return nearest_exit
end
minetest.register_chatcommand("dumpportalkeys", {
description = S("Dump all portal keys"),
privs = { debug = true },
func = function(name, param)
keys = {}
for k, _ in pairs(exits) do
keys[#keys+1] = k
end
output = string.format("Nether portal exits keys: %s", table.concat(keys, ", "))
return true, output
end,
})
local function dump_key(key)
output = string.format("[%d] => ", tonumber(key))
for _,p in pairs(exits[tonumber(key)]) do
output = output .. minetest.pos_to_string(p) .. " "
end
output = output .. "\n"
return output
end
minetest.register_chatcommand("dumpportalexits", {
description = S("Dump coordinates of registered nether portal exits"),
privs = { debug = true },
params = "[key]",
func = function(name, param)
local key = param
if not key or key == "" then
output = "Nether portal exit locations (all dimensions):\n"
else
output = string.format("Nether portal exit locations for key [%s] (all dimensions):\n", key)
end
if not key or key == "" then
local count = 0
for k, _ in pairs(exits) do
count = count + 1
if count>100 then
output = output .. "The list exceeds 100 keys, truncated. Try /dumpportalkeys, then /dumpportalexits KEY"
break
end
output = output .. dump_key(k)
end
else
-- key specified, no limits
if not exits[tonumber(key)] then
output = output .. string.format("No exits in key [%s]\n", key)
else
dump_key(key)
end
end
return true, output
end,
})
-- Map coordinates between dimensions. Distances in X and Z are scaled by NETHER_COMPRESSION.
-- Distances in Y are mapped directly. Caller should check the return value, this function
-- returns nil if there is no valid mapping - for example if the target would be out of world.
local function get_target(p)
if not p or not p.y or not p.x or not p.z then
return
end
local _, dim = mcl_worlds.y_to_layer(p.y)
local x,y,z
if dim=="nether" then
-- traveling to OW
x = floor(p.x * NETHER_COMPRESSION)
z = floor(p.z * NETHER_COMPRESSION)
if x>=LIM_MAX or x<=LIM_MIN or z>=LIM_MAX or z<=LIM_MIN then
-- Traveling out of bounds is forbidden.
return
end
y = max(min(p.y, N_Y_MAX), N_Y_MIN)
y = y - N_Y_MIN
y = O_Y_MIN + y
y = max(min(y, O_Y_MAX), O_Y_MIN)
else
-- traveling to the nether
x = floor(p.x / NETHER_COMPRESSION)
z = floor(p.z / NETHER_COMPRESSION)
if x>=LIM_MAX or x<=LIM_MIN or z>=LIM_MAX or z<=LIM_MIN then
-- Traveling out of bounds is forbidden.
return
end
y = max(min(p.y, O_Y_MAX), O_Y_MIN)
y = y - O_Y_MIN
y = N_Y_MIN + y
y = max(min(y, N_Y_MAX), N_Y_MIN)
end
return vector.new(x,y,z)
end
-- Destroy a nether portal. Connected portal nodes are searched and removed
-- using 'bulk_set_node'. This function is called from 'after_destruct' of
-- nether portal nodes. The flag 'destroying_portal' is used to avoid this
-- function being called recursively through callbacks in 'bulk_set_node'.
-- To maintain portal integrity, it is permitted to destroy protected portal
-- blocks if the portal structure is only partly protected, and the player
-- destroys the part that is sticking out.
local destroying_portal = false
local function destroy_nether_portal(pos, node)
if destroying_portal then
return
end
destroying_portal = true
local orientation = node.param2
local checked_tab = { [minetest.hash_node_position(pos)] = true }
local nodes = { pos }
local function check_remove(pos)
local h = minetest.hash_node_position(pos)
if checked_tab[h] then
return
end
local node = minetest.get_node(pos)
if node and node.name == PORTAL and (orientation == nil or node.param2 == orientation) then
table.insert(nodes, pos)
checked_tab[h] = true
end
end
local i = 1
while i <= #nodes do
pos = nodes[i]
if orientation == 0 then
check_remove({x = pos.x - 1, y = pos.y, z = pos.z})
check_remove({x = pos.x + 1, y = pos.y, z = pos.z})
else
check_remove({x = pos.x, y = pos.y, z = pos.z - 1})
check_remove({x = pos.x, y = pos.y, z = pos.z + 1})
end
check_remove({x = pos.x, y = pos.y - 1, z = pos.z})
check_remove({x = pos.x, y = pos.y + 1, z = pos.z})
remove_exits({pos})
i = i + 1
end
minetest.bulk_set_node(nodes, { name = "air" })
destroying_portal = false
end
local on_rotate
if minetest.get_modpath("screwdriver") then
-- Presumably because it messes with the placement of exits.
on_rotate = screwdriver.disallow
end
minetest.register_node(PORTAL, {
description = S("Nether Portal"),
_doc_items_longdesc = S("A Nether portal teleports creatures and objects to the hot and dangerous Nether dimension (and back!). Enter at your own risk!"),
_doc_items_usagehelp = S("Stand in the portal for a moment to activate the teleportation. Entering a Nether portal for the first time will also create a new portal in the other dimension. If a Nether portal has been built in the Nether, it will lead to the Overworld. A Nether portal is destroyed if the any of the obsidian which surrounds it is destroyed, or if it was caught in an explosion."),
tiles = {
"blank.png",
"blank.png",
"blank.png",
"blank.png",
{
name = "mcl_portals_portal.png",
animation = {
type = "vertical_frames",
aspect_w = 16,
aspect_h = 16,
length = 1.25,
},
},
{
name = "mcl_portals_portal.png",
animation = {
type = "vertical_frames",
aspect_w = 16,
aspect_h = 16,
length = 1.25,
},
},
},
drawtype = "nodebox",
paramtype = "light",
paramtype2 = "facedir",
sunlight_propagates = true,
use_texture_alpha = minetest.features.use_texture_alpha_string_modes and "blend" or true,
walkable = false,
buildable_to = false,
is_ground_content = false,
drop = "",
light_source = 11,
post_effect_color = {a = 180, r = 51, g = 7, b = 89},
node_box = {
type = "fixed",
fixed = {
{-0.5, -0.5, -0.1, 0.5, 0.5, 0.1},
},
},
groups = { creative_breakable = 1, portal = 1, not_in_creative_inventory = 1 },
sounds = mcl_sounds.node_sound_glass_defaults(),
after_destruct = destroy_nether_portal,
on_rotate = on_rotate,
_mcl_hardness = -1,
_mcl_blast_resistance = 0,
})
local function build_and_light_frame(x1, y1, z1, x2, y2, z2, name)
local orientation = 0
if x1 == x2 then
orientation = 1
end
local pos = {}
for x = x1 - 1 + orientation, x2 + 1 - orientation do
pos.x = x
for z = z1 - orientation, z2 + orientation do
pos.z = z
for y = y1 - 1, y2 + 1 do
pos.y = y
local frame = (x < x1) or (x > x2) or (y < y1) or (y > y2) or (z < z1) or (z > z2)
if frame then
set_node(pos, {name = OBSIDIAN})
else
set_node(pos, {name = PORTAL, param2 = orientation})
add_exits({
{x=pos.x, y=pos.y-1, z=pos.z}
})
end
end
end
end
end
-- Create a portal, where cube_pos1 is the "bottom-left" coordinate of the
-- W_MINxH_MIN cube to contain the portal - i.e. the coordinate with the
-- smallest X, Y and Z. We will be placing the portal in the middle-ish, so
-- that block +1,+1,+1 is guaranteed to be a portal block. The build area
-- includes frame.
-- Small obsidian platform will be created on either side of the exit if there
-- is no nodes there, or nodes underneath (one step down is permitted).
-- Orientation 0 is portal alongside X axis, 1 alongside Z.
function build_nether_portal(cube_pos1, width, height, orientation, name, clear_before_build)
local width, height, orientation = width or W_MIN, height or H_MIN, orientation or random(0, 1)
local width_inner = width-2
local height_inner = height-2
local cube_pos2 = add(cube_pos1, vector.new(width-1, height-1, width-1))
if is_area_protected(cube_pos1, cube_pos2, name) then
if name then
minetest.chat_send_player(name, "Unable to build portal, area is protected.")
end
return
end
-- Calculate "bottom-left" position of the PORTAL blocks.
-- Offset Y because we don't want it in the floor.
-- Offset X and Z to fit the frame and place the portal in the middle-ish.
local pos = add(cube_pos1, vector.new(1,1,1))
if clear_before_build then
local clear1, clear2
if orientation == 0 then
clear1 = vector.new(
cube_pos1.x,
cube_pos1.y + 1, -- do not delete floor
cube_pos1.z
)
clear2 = vector.new(
cube_pos1.x + width - 1,
cube_pos1.y + height - 2,
cube_pos1.z + 2 -- both sides of the entrance, so player has somewhere to step.
)
else
clear1 = vector.new(
cube_pos1.x,
cube_pos1.y + 1, -- do not delete floor
cube_pos1.z
)
clear2 = vector.new(
cube_pos1.x + 2, -- both sides of the entrance, so player has somewhere to step.
cube_pos1.y + height - 2,
cube_pos1.z + width - 1
)
end
log("verbose", "Clearing between "..pos_to_string(clear1).." and "..pos_to_string(clear2))
local airs = {}
for x = clear1.x, clear2.x do
for y = clear1.y, clear2.y do
for z = clear1.z, clear2.z do
airs[#airs+1] = vector.new(x,y,z)
end
end
end
minetest.bulk_set_node(airs, {name="air"})
end
build_and_light_frame(pos.x, pos.y, pos.z, pos.x + (1 - orientation) * (width_inner - 1), pos.y + height_inner - 1, pos.z + orientation * (width_inner - 1), name)
-- Build obsidian platform:
for x = pos.x - orientation, pos.x + orientation + (width_inner - 1) * (1 - orientation), 1 + orientation do
for z = pos.z - 1 + orientation, pos.z + 1 - orientation + (width_inner - 1) * orientation, 2 - orientation do
local pp = {x = x, y = pos.y - 1, z = z}
local pp_1 = {x = x, y = pos.y - 2, z = z}
local nn = get_node(pp).name
local nn_1 = get_node(pp_1).name
if ((nn=="air" and nn_1 == "air") or not registered_nodes[nn].is_ground_content) and not is_protected(pp, name) then
set_node(pp, {name = OBSIDIAN})
end
end
end
log("verbose", "Portal generated at "..pos_to_string(pos).."!")
return pos
end
-- Spawn portal at location - spawning is not guaranteed if target area is protected.
function mcl_portals.spawn_nether_portal(pos, rot, pr, name)
if not pos then return end
local o = 0
if rot then
if rot == "270" or rot=="90" then
o = 1
elseif rot == "random" then
o = random(0,1)
end
end
return build_nether_portal(pos, W_MIN, H_MIN, o, name, true)
end
-- Useful for testing - it lets player point to a block and create the portal in
-- the exact spot using the block as floor, with correct orientation (facing the
-- player). Alternatively, portals can be created at exact locations and
-- orientations (which spawn structure doesn't support).
minetest.register_chatcommand("spawnportal", {
description = S("Spawn a new nether portal at pointed thing, or at [x],[y],[z]. "
.."The portal will either face the player, or use the passed [orientation]. "
.."Orientation 0 means alongside X axis."),
privs = { debug = true },
params = "[x] [y] [z] [orientation]",
func = function(name, param)
local params = {}
for p in param:gmatch("-?[0-9]+") do table.insert(params, p) end
local exit
if #params==0 then
local player = minetest.get_player_by_name(name)
if not player then return false, "Player not found" end
local yaw = player:get_look_horizontal()
local player_rotation = yaw / (math.pi*2)
local orientation
if (player_rotation<=0.875 and player_rotation>0.625)
or (player_rotation<=0.375 and player_rotation>0.125)
then
orientation = "90"
else
orientation = "0"
end
local pointed_thing = mcl_util.get_pointed_thing(player, false)
if not pointed_thing then return false, "Not pointing to anything" end
if pointed_thing.type~="node" then return false, "Not pointing to a node" end
local pos = pointed_thing.under
-- Portal node will appear above the pointed node. The pointed node will turn into obsidian.
exit = mcl_portals.spawn_nether_portal(add(pos, vector.new(-1,0,-1)), orientation, nil, name)
elseif #params==3 or #params==4 then
pos = {
x=tonumber(params[1]),
y=tonumber(params[2]),
z=tonumber(params[3])
}
local orientation = 0
if #params==4 then
if tonumber(params[4])==1 then orientation = "90" else orientation = "0" end
end
-- Portal will be placed so that the first registered exit is at requested location.
exit = mcl_portals.spawn_nether_portal(add(pos, vector.new(-1,-1,-1)), orientation, nil, name)
else
return false, "Invalid parameters. Pass either zero, three or four"
end
if exit then
return true, "Spawned!"
else
return false, "Unable to spawn portal, area could be protected"
end
end,
})
-- Teleportation cooloff for some seconds, to prevent back-and-forth teleportation
local function stop_teleport_cooloff(o)
cooloff[o] = nil
chatter[o] = nil
end
local function teleport_cooloff(obj)
cooloff[obj] = true
if obj:is_player() then
minetest.after(PLAYER_COOLOFF, stop_teleport_cooloff, obj)
else
minetest.after(MOB_COOLOFF, stop_teleport_cooloff, obj)
end
end
local function finalize_teleport(obj, exit)
if not obj or not exit or not exit.x or not exit.y or not exit.z then return end
local objpos = obj:get_pos()
if not objpos then return end
local is_player = obj:is_player()
local name
if is_player then
name = obj:get_player_name()
end
local _, dim = mcl_worlds.y_to_layer(exit.y)
-- If player stands, player is at ca. something+0.5 which might cause precision problems, so we used ceil for objpos.y
objpos = {x = floor(objpos.x+0.5), y = ceil(objpos.y), z = floor(objpos.z+0.5)}
if get_node(objpos).name ~= PORTAL then
log("action", "Entity no longer standing in portal")
return
end
-- In case something went wrong, make sure the origin portal is always added as a viable exit.
-- In the ideal case, this is not needed, as the exits are added upon ignition. Before Jan 2024 this
-- was broken, so there will be a lot of half-added portals out there!
add_exits({objpos})
-- Enable teleportation cooloff for some seconds, to prevent back-and-forth teleportation
teleport_cooloff(obj)
obj:set_pos(exit)
local lua_entity = obj:get_luaentity()
if is_player then
mcl_worlds.dimension_change(obj, dim)
minetest.sound_play("mcl_portals_teleport", {pos=exit, gain=0.5, max_hear_distance = 16}, true)
log("action", "Player "..name.." teleported to portal at "..pos_to_string(exit)..".")
if dim == "nether" then
awards.unlock(obj:get_player_name(), "mcl:theNether")
end
elseif lua_entity then
log("verbose", string.format(
"Entity %s teleported to portal at %s",
lua_entity.name,
pos_to_string(exit)
))
end
end
local function is_origin_queued(origin)
local key = pos_to_string(origin)
if origin_queue[key] then
return true
else
return false
end
end
local function is_entity_queued()
for _, q in pairs(origin_queue) do
if q[obj] then
return true
end
end
return false
end
local function origin_enqueue(obj, origin)
local key = pos_to_string(origin)
log("verbose", string.format("Queueing entity for origin %s", key))
local q = origin_queue[key] or {}
if not q[obj] then
q[obj] = true
origin_queue[key] = q
end
end
-- Flush the queue of entities waiting for specific origin.
-- Pass nil/false exit to purge the queue without teleporting.
local function origin_flush(origin, exit)
local key = pos_to_string(origin)
local count_teleported = 0
local count_removed = 0
if origin_queue[key] then
for obj, value in pairs(origin_queue[key]) do
if value and exit then
finalize_teleport(obj, exit)
count_teleported = count_teleported + 1
else
count_removed = count_removed + 1
end
end
origin_queue[key] = nil
end
log("verbose", string.format(
"Finished flushing entities waiting for origin %s: removed %d, proceeded to teleport %d",
key,
count_removed,
count_teleported
))
end
local function find_build_limits(pos, target_dim)
-- Find node extremes of the pos's chunk.
-- According to what people said in minetest discord and couple of docs, mapgen
-- works on entire chunks, so we need to limit the search to chunk boundary.
-- The goal is to emerge at most two chunks.
local chunk_limit1 = add(
mul(
mcl_vars.pos_to_chunk(pos),
mcl_vars.chunk_size_in_nodes
),
vector.new(
mcl_vars.central_chunk_offset_in_nodes,
mcl_vars.central_chunk_offset_in_nodes,
mcl_vars.central_chunk_offset_in_nodes
)
)
local chunk_limit2 = add(
chunk_limit1,
vector.new(
mcl_vars.chunk_size_in_nodes - 1,
mcl_vars.chunk_size_in_nodes - 1,
mcl_vars.chunk_size_in_nodes - 1
)
)
-- Limit search area by using search distances. There is no Y build limit.
local build_limit1 = add(
pos,
-- minus 1 to account for the pos block being included.
-- plus 1 to account for the portal block offset (ignore frame)
vector.new(-BUILD_DISTANCE_XZ-1+1, 0, -BUILD_DISTANCE_XZ-1+1)
)
local build_limit2 = add(
pos,
-- plus 1 to account for the portal block offset (ignore frame)
-- minus potential portal width, so that the generated portal doesn't "stick out"
vector.new(BUILD_DISTANCE_XZ+1-(W_MIN-1), 0, BUILD_DISTANCE_XZ+1-(W_MIN-1))
)
-- Start with chunk limits
pos1 = vector.new(chunk_limit1.x, chunk_limit1.y, chunk_limit1.z)
pos2 = vector.new(chunk_limit2.x, chunk_limit2.y, chunk_limit2.z)
-- Make sure the portal is not built beyond chunk boundary
-- (we will be searching for the node with lowest X, Y and Z)
pos2.x = pos2.x-(W_MIN-1)
pos2.y = pos2.y-(H_MIN-1)
pos2.z = pos2.z-(W_MIN-1)
-- Avoid negative volumes
if pos2.x < pos1.x then pos2.x = pos1.x end
if pos2.y < pos1.y then pos2.y = pos1.y end
if pos2.z < pos1.z then pos2.z = pos1.z end
-- Apply build distances.
pos1 = {
x = max(pos1.x, build_limit1.x),
y = pos1.y,
z = max(pos1.z, build_limit1.z)
}
pos2 = {
x = min(pos2.x, build_limit2.x),
y = pos2.y,
z = min(pos2.z, build_limit2.z)
}
-- Apply dimension-specific distances, so that player does not end up in void or in lava.
local limit1, limit2 = limits[target_dim].pmin, limits[target_dim].pmax
pos1 = {
x = max(pos1.x, limit1.x),
y = max(pos1.y, limit1.y),
z = max(pos1.z, limit1.z)
}
pos2 = {
x = min(pos2.x, limit2.x),
y = min(pos2.y, limit2.y),
z = min(pos2.z, limit2.z)
}
local diff = add(pos2, mul(pos1, -1))
local area = diff.x * diff.z
local msg = string.format(
"Portal build area between %s-%s, a %dx%dx%d cuboid with floor area of %d nodes. "
.."Chunk limit was at [%s,%s]. "
.."Ideal build area was at [(%d,*,%d),(%d,*,%d)].",
pos_to_string(pos1),
pos_to_string(pos2),
diff.x,
diff.y,
diff.z,
area,
pos_to_string(chunk_limit1),
pos_to_string(chunk_limit2),
build_limit1.x,
build_limit1.z,
build_limit2.x,
build_limit2.z
)
log("verbose", msg)
return pos1, pos2
end
local function get_lava_level(pos, pos1, pos2)
if pos.y > -1000 then
return mcl_vars.mg_lava_overworld_max
end
return mcl_vars.mg_lava_nether_max
end
local function search_for_build_location(blockpos, action, calls_remaining, param)
if calls_remaining and calls_remaining > 0 then return end
local target, pos1, pos2, name, obj = param.target, param.pos1, param.pos2, param.name or "", param.obj
local chunk = mcl_vars.get_chunk_number(target)
local pos0, distance
local most_airy_count, most_airy_pos, most_airy_distance = param.most_airy_count, param.most_airy_pos, param.most_airy_distance
local lava = get_lava_level(target, pos1, pos2)
-- Portal might still exist in the area even though nothing was found in the table.
-- This could be due to bugs, or old worlds (portals added before the exits table).
-- Handle gracefully by searching and adding exits as appropriate.
local exit
local portals = find_nodes_in_area(pos1, pos2, {PORTAL})
if portals and #portals>0 then
for _, p in pairs(portals) do
-- This will only add legitimate exits that are not on the list,
-- and will only save if there was any changes.
add_exits({p})
end
if param.target_dim=="nether" then
exit = find_exit(target, SEARCH_DISTANCE_NETHER, SEARCH_DISTANCE_NETHER)
else
exit = find_exit(target, SEARCH_DISTANCE_OVERWORLD, SEARCH_DISTANCE_OVERWORLD)
end
end
if exit then
log("verbose", "Using a newly found exit at "..pos_to_string(exit))
origin_flush(param.origin, exit)
chunk_building[chunk] = false
return
end
-- No suitable portal was found, look for a suitable space and build a new one in the emerged blocks.
local nodes = find_nodes_in_area_under_air(pos1, pos2, {"group:building_block"})
-- Sort by distance so that we are checking the nearest nodes first.
-- This can speed up the search considerably if there is space around the ideal X and Z.
table.sort(nodes, function(a,b)
local da = dist(param.ideal_target, add(a, vector.new(1,1,1)))
local db = dist(param.ideal_target, add(b, vector.new(1,1,1)))
return da<db
end)
if nodes then
local nc = #nodes
if nc > 0 then
for i=1,nc do
local node = nodes[i]
local portal_node = add(node, vector.new(1,1,1)) -- Skip the frame
local node1 = add(node, vector.new(0,1,0)) -- Floor can be solid
local node2 = add(node, vector.new(W_MIN-1,H_MIN-1,W_MIN-1))
local nodes2 = find_nodes_in_area(node1, node2, {"air"})
if nodes2 then
local nc2 = #nodes2
if not is_area_protected(node, node2, name) and node.y > lava then
local distance0 = dist(param.ideal_target, portal_node)
if nc2 >= (W_MIN*(H_MIN-1)*W_MIN) - ACCEPTABLE_PORTAL_REPLACES then
-- We have sorted the candidates by distance, this is the best location.
distance = distance0
pos0 = {x=node.x, y=node.y, z=node.z}
log("verbose", "Found acceptable location at "..pos_to_string(pos0)..", distance "..distance0..", air nodes "..nc2)
break
elseif not most_airy_pos or nc2>most_airy_count then
-- Remember the cube with the most amount of air as a fallback.
most_airy_count = nc2
most_airy_distance = distance0
most_airy_pos = {x=node.x, y=node.y, z=node.z}
log("verbose", "Found fallback location at "..pos_to_string(most_airy_pos)..", distance "..distance0..", air nodes "..nc2)
elseif most_airy_pos and nc2==most_airy_count and distance0<most_airy_distance then
-- Use distance as a tiebreaker.
most_airy_distance = distance0
most_airy_pos = {x=node.x, y=node.y, z=node.z}
log("verbose", "Found fallback location at "..pos_to_string(most_airy_pos)..", distance "..distance0..", air nodes "..nc2)
end
end
end
end
end
end
if pos0 then
log("verbose", "Building portal at "..pos_to_string(pos0)..", distance "..distance)
local exit = build_nether_portal(pos0, W_MIN, H_MIN, random(0,1), name)
origin_flush(param.origin, exit)
chunk_building[chunk] = false
return
end
-- Look in chunks above or below, depending on which side has more
-- space. Since our Y map distance is quite short due to the flatness of
-- the non-lava nether, falling back like this should cover entire Y range.
if param.chunk_counter==1 then
if limits[param.target_dim].pmax.y-target.y > target.y-limits[param.target_dim].pmin.y then
-- Look up
direction = 1
log("verbose", "No space found, emerging one chunk above")
else
-- Look down
direction = -1
log("verbose", "No space found, emerging one chunk below")
end
local new_target = {x=target.x, y=target.y + direction * mcl_vars.chunk_size_in_nodes, z=target.z}
pos1, pos2 = find_build_limits(new_target, param.target_dim)
local diff = add(pos2, mul(pos1, -1))
-- Only emerge if there is sufficient headroom to actually fit entire portal.
if diff.y+1>=H_MIN then
local new_chunk = mcl_vars.get_chunk_number(new_target)
if chunk_building[new_chunk] then
log("verbose", string.format("Secondary chunk %s is currently busy, backing off", new_chunk))
origin_flush(param.origin, nil)
return
end
chunk_building[new_chunk] = true
minetest.emerge_area(pos1, pos2, search_for_build_location, {
origin=param.origin,
target = new_target,
target_dim = target_dim,
ideal_target = param.ideal_target,
pos1 = pos1,
pos2 = pos2,
name=name,
obj=obj,
chunk_counter=param.chunk_counter+1,
most_airy_count=most_airy_count,
most_airy_pos=most_airy_pos,
most_airy_distance=most_airy_distance
})
chunk_building[chunk] = false
return
end
end
-- Fall back to the most airy position in previous chunk, in this chunk,
-- or if all else fails to ideal position. This could replace a lot of blocks.
local fallback = param.ideal_target
if most_airy_pos then
log(
"verbose",
string.format(
"Falling back to the most airy position at %s, distance %d",
pos_to_string(most_airy_pos),
most_airy_distance
)
)
fallback = most_airy_pos
end
if fallback.y <= lava then
fallback.y = lava + 1
end
log("verbose", "Forcing portal at "..pos_to_string(fallback)..", lava at "..lava)
local exit = build_nether_portal(fallback, W_MIN, H_MIN, random(0,1), name, true)
origin_flush(param.origin, exit)
chunk_building[chunk] = false
end
local function create_portal(origin, target, target_dim, name, obj)
local chunk = mcl_vars.get_chunk_number(target)
if chunk_building[chunk] then
log("verbose", string.format("Primary chunk %s is currently busy, backing off", chunk))
origin_flush(origin, nil)
return
end
chunk_building[chunk] = true
local pos1, pos2 = find_build_limits(target, target_dim)
minetest.emerge_area(pos1, pos2, search_for_build_location, {
origin = origin,
target = target,
target_dim = target_dim,
ideal_target = vector.new(target.x, target.y, target.z), -- copy
pos1 = pos1,
pos2 = pos2,
name=name,
obj=obj,
chunk_counter=1
})
end
local function available_for_nether_portal(p)
-- No need to check for protected - non-owner can't ignite blocks anyway.
local nn = get_node(p).name
local obsidian = nn == OBSIDIAN
if nn ~= "air" and minetest.get_item_group(nn, "fire") ~= 1 then
return false, obsidian
end
return true, obsidian
end
local function check_and_light_shape(pos, orientation)
local stack = {{x = pos.x, y = pos.y, z = pos.z}}
local node_list = {}
local index_list = {}
local node_counter = 0
-- Search most low node from the left (pos1) and most right node from the top (pos2)
local pos1 = {x = pos.x, y = pos.y, z = pos.z}
local pos2 = {x = pos.x, y = pos.y, z = pos.z}
local kx, ky, kz = pos.x - 1999, pos.y - 1999, pos.z - 1999
while #stack > 0 do
local i = #stack
local x, y, z = stack[i].x, stack[i].y, stack[i].z
local k = (x-kx)*16000000 + (y-ky)*4000 + z-kz
if index_list[k] then
stack[i] = nil -- Already checked, skip it
else
local good, obsidian = available_for_nether_portal(stack[i])
if obsidian then
stack[i] = nil
else
if (not good) or (node_counter >= N_MAX) then
return false
end
node_counter = node_counter + 1
node_list[node_counter] = {x = x, y = y, z = z}
index_list[k] = true
stack[i].y = y - 1
stack[i + 1] = {x = x, y = y + 1, z = z}
if orientation == 0 then
stack[i + 2] = {x = x - 1, y = y, z = z}
stack[i + 3] = {x = x + 1, y = y, z = z}
else
stack[i + 2] = {x = x, y = y, z = z - 1}
stack[i + 3] = {x = x, y = y, z = z + 1}
end
if (y < pos1.y) or (y == pos1.y and (x < pos1.x or z < pos1.z)) then
pos1 = {x = x, y = y, z = z}
end
if (x > pos2.x or z > pos2.z) or (x == pos2.x and z == pos2.z and y > pos2.y) then
pos2 = {x = x, y = y, z = z}
end
end
end
end
if node_counter < N_MIN then
return false
end
-- Limit rectangles width and height
if abs(pos2.x - pos1.x + pos2.z - pos1.z) + 3 > W_MAX or abs(pos2.y - pos1.y) + 3 > H_MAX then
return false
end
-- Light the portal
for i = 1, node_counter do
minetest.set_node(node_list[i], {name = PORTAL, param2 = orientation})
end
-- Register valid portal exits (each portal has at least two!)
-- Before Jan 2024, there was a bug that did not register all exits upon ignition.
-- This means portals lit before that time will only become live as people use them
-- (and only specific portal blocks).
for i = 1, node_counter do
-- Improvement possible: we are only interested in the bottom
-- blocks as exits, but here all ignited blocks are passed in.
-- This can cause a lot of failed validations on very large
-- portals that we know can be skipped.
add_exits({node_list[i]})
end
return true
end
-- Attempts to light a Nether portal at pos
-- Pos can be any of the inner part.
-- The frame MUST be filled only with air or any fire, which will be replaced with Nether portal blocks.
-- If no Nether portal can be lit, nothing happens.
-- Returns true if portal created
function mcl_portals.light_nether_portal(pos)
-- Only allow to make portals in Overworld and Nether
local dim = mcl_worlds.pos_to_dimension(pos)
if dim ~= "overworld" and dim ~= "nether" then
return false
end
if not get_target(pos) then
-- Prevent ignition of portals that would lead to out of bounds positions.
log("verbose", string.format(
"No target found for position %s - portal would lead to invalid exit",
pos_to_string(pos)
))
return
end
local orientation = random(0, 1)
for orientation_iteration = 1, 2 do
if check_and_light_shape(pos, orientation) then
return true
end
orientation = 1 - orientation
end
return false
end
local function check_portal_then_teleport(obj, origin, exit)
-- Check we are not sending the player on a one-way trip.
minetest.emerge_area(exit, exit, function (blockpos, action, calls_remaining, param)
if calls_remaining and calls_remaining > 0 then return end
if get_node(exit).name ~= PORTAL then
-- Bogus exit! Break the teleportation so we don't strand the player.
-- The process will begin again after cooloff through the ABM, and might either
-- find another exit, or build a new portal. This will manifest to the
-- player as a teleportation that takes longer than usual.
log("warning", "removing bogus portal exit encountered at "..pos_to_string(exit)..", exit no longer exists")
remove_exits({exit})
-- Also remove from structure storage, otherwise ABM will try the same bad exit again.
local objpos = obj:get_pos()
delete_portal_pos({x = floor(objpos.x+0.5), y = ceil(objpos.y), z = floor(objpos.z+0.5)})
origin_flush(origin, nil)
return
end
origin_flush(origin, exit)
end)
end
-- Teleport function
local function teleport_no_delay(obj, portal_pos)
local is_player = obj:is_player()
if (not is_player and not obj:get_luaentity()) or cooloff[obj] then return end
local objpos = obj:get_pos()
if not objpos then return end
local _, current_dim = mcl_worlds.y_to_layer(objpos.y)
local target_dim = dimension_to_teleport[current_dim]
-- If player stands, player is at ca. something+0.5 which might cause precision problems, so we used ceil for objpos.y
origin = {x = floor(objpos.x+0.5), y = ceil(objpos.y), z = floor(objpos.z+0.5)}
if get_node(origin).name ~= PORTAL then return end
local target = get_target(origin)
if not target then
log("verbose", string.format(
"No target found for position %s - no valid exit found",
pos_to_string(origin)
))
return
end
local name
if is_player then
name = obj:get_player_name()
end
if is_entity_queued(obj) then
-- Let's not allow one entity to generate a lot of work by just moving around in a portal.
log("verbose", "Entity already queued")
return
end
log("verbose", string.format("Target calculated as %s", pos_to_string(target)))
local already_queued = is_origin_queued(origin)
origin_enqueue(obj, origin)
if already_queued then
-- Origin is already being processed, so wait in queue for the result.
log("verbose", string.format("Origin %s already queued", pos_to_string(origin)))
return
end
local exit
local saved_portal_position = get_portal_pos(origin)
if saved_portal_position then
-- Before Jan 2024, portal exits were sticky - they were stored
-- in nodes. If such a position is found, look for the exit
-- there, so that the players don't get any surprises.
-- Sticky exit can be removed by destroying and rebuilding the portal.
log("verbose", "Using block-saved portal exit: "..pos_to_string(saved_portal_position)..".")
exit = find_exit(saved_portal_position, 10, 10)
end
if not exit then
-- Search for nearest suitable exit in the lookup table.
if target_dim=="nether" then
exit = find_exit(target, SEARCH_DISTANCE_NETHER, SEARCH_DISTANCE_NETHER)
else
exit = find_exit(target, SEARCH_DISTANCE_OVERWORLD, SEARCH_DISTANCE_OVERWORLD)
end
end
if exit then
log("verbose", "Exit found at "..pos_to_string(exit).." for target "..pos_to_string(target).." traveling from "..pos_to_string(origin))
check_portal_then_teleport(obj, origin, exit)
else
create_portal(origin, target, target_dim, name, obj)
end
end
local function prevent_portal_chatter(obj)
local time_us = get_us_time()
local ch = chatter[obj] or 0
chatter[obj] = time_us
minetest.after(TOUCH_CHATTER_TIME, function(o)
if o and chatter[o] and get_us_time() - chatter[o] >= CHATTER_US then
chatter[o] = nil
end
end, obj)
return time_us - ch > CHATTER_US
end
local function animation(player, playername)
local ch = chatter[player] or 0
if cooloff[player] or get_us_time() - ch < CHATTER_US then
local pos = player:get_pos()
if not pos then
return
end
minetest.add_particlespawner({
amount = 1,
minpos = {x = pos.x - 0.1, y = pos.y + 1.4, z = pos.z - 0.1},
maxpos = {x = pos.x + 0.1, y = pos.y + 1.6, z = pos.z + 0.1},
minvel = 0,
maxvel = 0,
minacc = 0,
maxacc = 0,
minexptime = 0.1,
maxexptime = 0.2,
minsize = 5,
maxsize = 15,
collisiondetection = false,
texture = "mcl_particles_nether_portal_t.png",
playername = playername,
})
minetest.after(0.3, animation, player, playername)
end
end
local function teleport(obj, portal_pos)
local name = ""
if obj:is_player() then
name = obj:get_player_name()
animation(obj, name)
end
if cooloff[obj] then return end
local delay = math.max(0, nether_portal_survival_delay[1] - 1)
if minetest.is_creative_enabled(name) then
delay = math.max(0, nether_portal_creative_delay[1] - 1)
end
if delay == 0 then
teleport_no_delay(obj, portal_pos)
else
minetest.after(delay, teleport_no_delay, obj, portal_pos)
end
end
minetest.register_abm({
label = "Nether portal teleportation and particles",
nodenames = {PORTAL},
interval = 1,
chance = 1,
action = function(pos, node)
local o = node.param2 -- orientation
local d = random(0, 1) -- direction
local time = random() * 1.9 + 0.5
local velocity, acceleration
if o == 1 then
velocity = {x = random() * 0.7 + 0.3, y = random() - 0.5, z = random() - 0.5}
acceleration = {x = random() * 1.1 + 0.3, y = random() - 0.5, z = random() - 0.5}
else
velocity = {x = random() - 0.5, y = random() - 0.5, z = random() * 0.7 + 0.3}
acceleration = {x = random() - 0.5, y = random() - 0.5, z = random() * 1.1 + 0.3}
end
local distance = add(mul(velocity, time), mul(acceleration, time * time / 2))
if d == 1 then
if o == 1 then
distance.x = -distance.x
velocity.x = -velocity.x
acceleration.x = -acceleration.x
else
distance.z = -distance.z
velocity.z = -velocity.z
acceleration.z = -acceleration.z
end
end
distance = sub(pos, distance)
for _, obj in pairs(minetest.get_objects_inside_radius(pos, 15)) do
if obj:is_player() then
minetest.add_particlespawner({
amount = PARTICLES + 1,
minpos = distance,
maxpos = distance,
minvel = velocity,
maxvel = velocity,
minacc = acceleration,
maxacc = acceleration,
minexptime = time,
maxexptime = time,
minsize = 0.3,
maxsize = 1.8,
collisiondetection = false,
texture = "mcl_particles_nether_portal.png",
playername = obj:get_player_name(),
})
end
end
for _, obj in pairs(minetest.get_objects_inside_radius(pos, 1)) do
-- Teleport players, mobs, boats etc.
local lua_entity = obj:get_luaentity()
if (obj:is_player() or lua_entity) and prevent_portal_chatter(obj) then
teleport(obj, pos)
end
end
end,
})
--[[ ITEM OVERRIDES ]]
local longdesc = registered_nodes[OBSIDIAN]._doc_items_longdesc
longdesc = longdesc .. "\n" .. S("Obsidian is also used as the frame of Nether portals.")
local usagehelp = S("To open a Nether portal, place an upright frame of obsidian with a width of at least 4 blocks and a height of 5 blocks, leaving only air in the center. After placing this frame, light a fire in the obsidian frame. Nether portals only work in the Overworld and the Nether.")
minetest.override_item(OBSIDIAN, {
_doc_items_longdesc = longdesc,
_doc_items_usagehelp = usagehelp,
on_destruct = function(pos, node)
-- Permit extinguishing of protected portals if the frame is
-- sticking out of the protected area to maintain integrity.
local function check_remove(pos, orientation)
local node = get_node(pos)
if node and node.name == PORTAL then
minetest.remove_node(pos)
end
end
-- check each of 6 sides of it and destroy every portal
check_remove({x = pos.x - 1, y = pos.y, z = pos.z})
check_remove({x = pos.x + 1, y = pos.y, z = pos.z})
check_remove({x = pos.x, y = pos.y, z = pos.z - 1})
check_remove({x = pos.x, y = pos.y, z = pos.z + 1})
check_remove({x = pos.x, y = pos.y - 1, z = pos.z})
check_remove({x = pos.x, y = pos.y + 1, z = pos.z})
end,
_on_ignite = function(user, pointed_thing)
local x, y, z = pointed_thing.under.x, pointed_thing.under.y, pointed_thing.under.z
-- Check empty spaces around obsidian and light all frames found.
-- Permit igniting of portals that are partly protected to maintain integrity.
local portals_placed =
mcl_portals.light_nether_portal({x = x - 1, y = y, z = z}) or mcl_portals.light_nether_portal({x = x + 1, y = y, z = z}) or
mcl_portals.light_nether_portal({x = x, y = y - 1, z = z}) or mcl_portals.light_nether_portal({x = x, y = y + 1, z = z}) or
mcl_portals.light_nether_portal({x = x, y = y, z = z - 1}) or mcl_portals.light_nether_portal({x = x, y = y, z = z + 1})
if portals_placed then
log("verbose", "Nether portal activated at "..pos_to_string({x=x,y=y,z=z})..".")
if minetest.get_modpath("doc") then
doc.mark_entry_as_revealed(user:get_player_name(), "nodes", PORTAL)
-- Achievement for finishing a Nether portal TO the Nether
local dim = mcl_worlds.pos_to_dimension({x=x, y=y, z=z})
if minetest.get_modpath("awards") and dim ~= "nether" and user:is_player() then
awards.unlock(user:get_player_name(), "mcl:buildNetherPortal")
end
end
return true
else
return false
end
end,
})
mcl_structures.register_structure("nether_portal",{
nospawn = true,
filenames = {
modpath.."/schematics/mcl_portals_nether_portal.mts"
}
})
mcl_structures.register_structure("nether_portal_open",{
nospawn = true,
filenames = {
modpath.."/schematics/mcl_portals_nether_portal_open.mts"
},
after_place = function(pos,def,pr,blockseed)
-- The mts is the smallest portal (two wide) and places the first PORTAL block
-- above the location of the caller (y+1). The second one is either at x+1 or z+1.
local portals = find_nodes_in_area(add(pos, vector.new(0,1,0)), add(pos, vector.new(1,1,1)), {PORTAL})
if portals and #portals>0 then
for _, p in pairs(portals) do
add_exits({p})
end
end
end
})