Helper functions for other mods and an obsidian portal bugfix (#15)

* The obsidian portal is the first way for a player to reach the nether,
  so if damage is enabled and `nether.trap_players` is disabled,
  the obsidian portal should teleport the player instead of killing him/her
  since after death he/she respawns in the overworld and not in the nether.
  A small change in `obsi_teleport_player` fixes this problem.
  Fixes #16.
* There exist other mods which need to check if players are in the nether and
  teleport them to/from the nether, so we add four helper functions:
  `nether.is_player_in_nether`, `nether.is_player_trapped_in_nether`.
  `nether.external_nether_teleport` and `nether.registry_update`.
  Fixes #14.
* Add the `in_hell`, `trapped_in_hell` and `update_hells_registry` chat
  commands, which can be used for debugging
* `players_in_nether` has no meaningful effect if `nether.trap_players` is
  disabled, so we rename it to `players_trapped_in_nether`.
* Add `nether.bottom`, which approximately corresponds to the lower end of the
  nether
This commit is contained in:
Deathwing777 2024-02-29 12:05:03 -08:00 committed by GitHub
parent 02d14a8970
commit 1c033a048f
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 144 additions and 14 deletions

View File

@ -52,6 +52,9 @@ nether.start = f_h_max+100
-- Height of the nether (bottom of the nether is nether_middle - NETHER_HEIGHT) -- Height of the nether (bottom of the nether is nether_middle - NETHER_HEIGHT)
local NETHER_HEIGHT = 30 local NETHER_HEIGHT = 30
-- bottom height of the nether.
nether.bottom = nether_middle - NETHER_HEIGHT - 100
-- Maximum amount of randomness in the map generation -- Maximum amount of randomness in the map generation
local NETHER_RANDOM = 2 local NETHER_RANDOM = 2

View File

@ -11,7 +11,7 @@ minetest.after(5, function()
end) end)
local save_path = minetest.get_worldpath() .. "/nether_players" local save_path = minetest.get_worldpath() .. "/nether_players"
local players_in_nether = {} local players_trapped_in_nether = {}
-- Load the list of players which are trapped in the nether -- Load the list of players which are trapped in the nether
-- (or would be trapped if nether.trap_players was true) -- (or would be trapped if nether.trap_players was true)
@ -23,7 +23,7 @@ do
if contents then if contents then
local playernames = string.split(contents, " ") local playernames = string.split(contents, " ")
for i = 1,#playernames do for i = 1,#playernames do
players_in_nether[playernames[i]] = true players_trapped_in_nether[playernames[i]] = true
end end
end end
end end
@ -31,7 +31,7 @@ end
local function save_nether_players() local function save_nether_players()
local playernames,n = {},1 local playernames,n = {},1
for name in pairs(players_in_nether) do for name in pairs(players_trapped_in_nether) do
playernames[n] = name playernames[n] = name
n = n+1 n = n+1
end end
@ -41,6 +41,17 @@ local function save_nether_players()
io.close(f) io.close(f)
end end
-- Nether aware mods will need to know if a player is in the nether.
function nether.is_player_in_nether(player)
local pos = player:get_pos()
return (pos.y < nether.start) and (pos.y >= nether.bottom)
end
-- For testing nether trap state tracking.
function nether.is_player_trapped_in_nether(player)
return players_trapped_in_nether[player:get_player_name()]
end
local update_background local update_background
if nether.trap_players then if nether.trap_players then
function update_background(player, down) function update_background(player, down)
@ -54,6 +65,36 @@ else
function update_background()end function update_background()end
end end
-- Nether aware mods may have other means of moving players between the Nether
-- and Overworld, and if so, they should tell us about it so we can keep track
-- of the player state.
function nether.external_nether_teleport(player, pos)
if not nether.trap_players then
player:set_pos(pos)
return
end
local destination_in_nether = (pos.y < nether.start) and (pos.y >= nether.bottom)
update_background(player, destination_in_nether)
local pname = player:get_player_name()
players_trapped_in_nether[pname] = destination_in_nether or nil
player:set_pos(pos)
end
-- Has the player dug their way out of the nether?
-- Has nether.trap_players been disabled?
function nether.registry_update(player)
local pos = player:get_pos()
local in_nether = (pos.y < nether.start) and (pos.y >= nether.bottom)
local pname = player:get_player_name()
if nether.trap_players then
players_trapped_in_nether[pname] = in_nether or nil
update_background(player, in_nether)
elseif players_trapped_in_nether[pname] then
players_trapped_in_nether[pname] = nil
update_background(player, false)
end
end
-- returns nodename if area is generated, else calls generation function -- returns nodename if area is generated, else calls generation function
local function generated_or_generate(pos) local function generated_or_generate(pos)
local node = minetest.get_node_or_nil(pos) local node = minetest.get_node_or_nil(pos)
@ -79,11 +120,12 @@ end
-- used for obsidian portal -- used for obsidian portal
local function obsidian_teleport(player, pname, target) local function obsidian_teleport(player, pname, target)
minetest.chat_send_player(pname, "For any reason you arrived here. Type " .. minetest.chat_send_player(pname, "For some reason you arrived here. Type " ..
"/nether_help to find out things like craft recipes.") "/nether_help to find out things like craft recipes.")
players_in_nether[pname] = true players_trapped_in_nether[pname] = true
save_nether_players() save_nether_players()
update_background(player, true) update_background(player, true)
if target then if target then
player:set_pos(target) player:set_pos(target)
else else
@ -94,14 +136,14 @@ end
-- teleports players to nether or helps it -- teleports players to nether or helps it
local function player_to_nether(player, pos) local function player_to_nether(player, pos)
local pname = player:get_player_name() local pname = player:get_player_name()
players_in_nether[pname] = true players_trapped_in_nether[pname] = true
save_nether_players() save_nether_players()
update_background(player, true) update_background(player, true)
if pos then if pos then
player:set_pos(pos) player:set_pos(pos)
return return
end end
minetest.chat_send_player(pname, "For any reason you arrived here. " .. minetest.chat_send_player(pname, "For some reason you arrived here. " ..
"Type /nether_help to find out things like craft recipes.") "Type /nether_help to find out things like craft recipes.")
if nether.trap_players then if nether.trap_players then
player:set_hp(0) player:set_hp(0)
@ -113,8 +155,8 @@ end
local function player_from_nether(player, pos) local function player_from_nether(player, pos)
local pname = player:get_player_name() local pname = player:get_player_name()
if players_in_nether[pname] then if players_trapped_in_nether[pname] then
players_in_nether[pname] = nil players_trapped_in_nether[pname] = nil
save_nether_players() save_nether_players()
end end
update_background(player, false) update_background(player, false)
@ -176,13 +218,98 @@ minetest.register_chatcommand("from_hell", {
end end
}) })
-- Useful for debugging Nether player state tracking. Written by Deathwing777
minetest.register_chatcommand("in_hell", {
params = "[<player_name>]",
description = "Is the player in hell?",
func = function(name, pname)
if not minetest.check_player_privs(name, {nether=true}) then
return false,
"You need the nether priv to execute this chatcommand."
end
if not player_exists(pname) then
pname = name
end
local player = minetest.get_player_by_name(pname)
if not player then
return false, "Something went wrong."
end
local status = pname.." is in the "
if nether.is_player_in_nether(player) then
status = status.."NETHER!"
else
status = status.."OVERWORLD!"
end
return true, status
end
})
-- Useful for debugging Nether player state tracking. Written by Deathwing777
minetest.register_chatcommand("trapped_in_hell", {
params = "[<player_name>]",
description = "Is the player trapped in hell?",
func = function(name, pname)
if not minetest.check_player_privs(name, {nether=true}) then
return false,
"You need the nether priv to execute this chatcommand."
end
if not player_exists(pname) then
pname = name
end
local player = minetest.get_player_by_name(pname)
if not player then
return false, "Something went wrong."
end
local status = pname
if nether.is_player_trapped_in_nether(player) then
status = status.." is TRAPPED in nether!"
else
status = status.." is NOT trapped in nether!"
end
return true, status
end
})
-- Useful for debugging Nether player state tracking. Written by Deathwing777
minetest.register_chatcommand("update_hells_registry", {
params = "[<player_name>]",
description = "Update player state if they got to or from the nether in another way.",
func = function(name, pname)
if not minetest.check_player_privs(name, {nether=true}) then
return false,
"You need the nether priv to execute this chatcommand."
end
if not player_exists(pname) then
pname = name
end
local player = minetest.get_player_by_name(pname)
if not player then
return false, "Something went wrong."
end
nether.registry_update(player)
local status = pname
if nether.is_player_trapped_in_nether(player) then
status = status.." is TRAPPED in nether!"
else
status = status.." is NOT trapped in nether!"
end
return true, status
end
})
-- Disallow teleportation and change spawn positions if the nether traps players -- Disallow teleportation and change spawn positions if the nether traps players
if nether.trap_players then if nether.trap_players then
-- randomly set player position when he/she dies in nether -- randomly set player position when he/she dies in nether
minetest.register_on_respawnplayer(function(player) minetest.register_on_respawnplayer(function(player)
local pname = player:get_player_name() local pname = player:get_player_name()
if not players_in_nether[pname] then if not players_trapped_in_nether[pname] then
return return
end end
local target = get_player_died_target(player) local target = get_player_died_target(player)
@ -197,14 +324,14 @@ if nether.trap_players then
return true return true
end) end)
-- override set_pos etc. to disallow player teleportion by e.g. travelnet -- override set_pos etc, to disallow player teleportion by e.g. travelnet
local function can_teleport(player, pos) local function can_teleport(player, pos)
if not player:is_player() then if not player:is_player() then
-- the same metatable is used for entities -- the same metatable is used for entities
return true return true
end end
local pname = player:get_player_name() local pname = player:get_player_name()
local in_nether = players_in_nether[pname] == true local in_nether = players_trapped_in_nether[pname] == true
-- test if the target is valid -- test if the target is valid
if pos.y < nether.start then if pos.y < nether.start then
@ -299,7 +426,7 @@ local particledef = {
-- teleports player to neter (obsidian portal) -- teleports player to neter (obsidian portal)
local function obsi_teleport_player(player, pos, target) local function obsi_teleport_player(player, pos, target)
local pname = player:get_player_name() local pname = player:get_player_name()
if players_in_nether[pname] then if players_trapped_in_nether[pname] then
return return
end end
@ -310,7 +437,7 @@ local function obsi_teleport_player(player, pos, target)
end end
local has_teleported local has_teleported
if damage_enabled then if (damage_enabled and nether.trap_players) then
obsidian_teleport(player, pname) obsidian_teleport(player, pname)
has_teleported = true has_teleported = true
elseif not mclike_portal then elseif not mclike_portal then