251 lines
6.0 KiB
Lua
Raw Normal View History

2019-08-23 10:01:41 +02:00
local S = minetest.get_translator("findbiome")
2019-08-27 06:42:14 +02:00
local mod_biomeinfo = minetest.get_modpath("biomeinfo") ~= nil
2019-08-27 06:42:14 +02:00
local mg_name = minetest.get_mapgen_setting("mg_name")
-- Calculate the maximum playable limit
local mapgen_limit = tonumber(minetest.get_mapgen_setting("mapgen_limit"))
local chunksize = tonumber(minetest.get_mapgen_setting("chunksize"))
local playable_limit = math.max(mapgen_limit - (chunksize + 1) * 16, 0)
2019-08-23 10:01:41 +02:00
-- Parameters
-------------
-- Resolution of search grid in nodes.
local res = 64
-- Number of points checked in the square search grid (edge * edge).
local checks = 128 * 128
-- End of parameters
--------------------
-- Direction table
local dirs = {
{x = 0, y = 0, z = 1},
{x = -1, y = 0, z = 0},
{x = 0, y = 0, z = -1},
{x = 1, y = 0, z = 0},
}
-- Returns true if pos is within the world boundaries
local function is_in_world(pos)
2019-08-27 06:49:50 +02:00
return not (math.abs(pos.x) > playable_limit or math.abs(pos.y) > playable_limit or math.abs(pos.z) > playable_limit)
2019-08-27 06:42:14 +02:00
end
-- Checks if pos is within the biome's boundaries. If it isn't, places pos inside the boundaries.
local function adjust_pos_to_biome_limits(pos, biome_id)
local bpos = table.copy(pos)
local biome_name = minetest.get_biome_name(biome_id)
local biome = minetest.registered_biomes[biome_name]
if not biome then
minetest.log("error", "[findbiome] adjust_pos_to_biome_limits non-existing biome!")
return bpos
end
local axes = {"y", "x", "z"}
local out_of_bounds = false
for a=1, #axes do
local ax = axes[a]
local min, max
if biome[ax.."_min"] then
min = biome[ax.."_min"]
else
min = -playable_limit
end
if biome[ax.."_max"] then
max = biome[ax.."_max"]
else
max = playable_limit
end
min = tonumber(min)
max = tonumber(max)
if bpos[ax] < min then
out_of_bounds = true
bpos[ax] = min
end
if bpos[ax] > max then
out_of_bounds = true
bpos[ax] = max
end
end
return bpos, out_of_bounds
end
2019-08-27 06:42:52 +02:00
local function find_biome(pos, biomes)
2019-08-23 10:01:41 +02:00
pos = vector.round(pos)
-- Pos: Starting point for biome checks. This also sets the y co-ordinate for all
-- points checked, so the suitable biomes must be active at this y.
-- Initial variables
local edge_len = 1
local edge_dist = 0
local dir_step = 0
local dir_ind = 1
local success = false
local spawn_pos
2019-08-27 06:49:50 +02:00
local biome_ids
2019-08-23 10:01:41 +02:00
-- Get next position on square search spiral
local function next_pos()
if edge_dist == edge_len then
edge_dist = 0
dir_ind = dir_ind + 1
if dir_ind == 5 then
dir_ind = 1
end
dir_step = dir_step + 1
edge_len = math.floor(dir_step / 2) + 1
end
local dir = dirs[dir_ind]
local move = vector.multiply(dir, res)
edge_dist = edge_dist + 1
return vector.add(pos, move)
end
-- Position search
local function search()
for iter = 1, checks do
local biome_data = minetest.get_biome_data(pos)
-- Sometimes biome_data is nil
local biome = biome_data and biome_data.biome
for id_ind = 1, #biome_ids do
local biome_id = biome_ids[id_ind]
local spos = table.copy(pos)
2019-08-23 10:01:41 +02:00
if biome == biome_id then
local spawn_y = minetest.get_spawn_level(spos.x, spos.z)
2019-08-23 10:01:41 +02:00
if spawn_y then
spawn_pos = {x = spos.x, y = spawn_y, z = spos.z}
if is_in_world(spawn_pos) then
2019-08-27 06:42:14 +02:00
return true
end
2019-08-23 10:01:41 +02:00
end
end
end
pos = next_pos()
end
return false
end
2019-08-26 21:03:31 +02:00
local function search_v6()
if not mod_biomeinfo then return
false
end
2019-08-26 21:03:31 +02:00
for iter = 1, checks do
local found_biome = biomeinfo.get_v6_biome(pos)
for i = 1, #biomes do
local searched_biome = biomes[i]
if found_biome == searched_biome then
local spawn_y = minetest.get_spawn_level(pos.x, pos.z)
if spawn_y then
spawn_pos = {x = pos.x, y = spawn_y, z = pos.z}
if is_in_world(spawn_pos) then
2019-08-27 06:42:14 +02:00
return true
end
2019-08-26 21:03:31 +02:00
end
end
end
2019-08-23 10:01:41 +02:00
2019-08-26 21:03:31 +02:00
pos = next_pos()
end
return false
end
if mg_name == "v6" then
success = search_v6()
else
-- Table of suitable biomes
2019-08-27 06:49:50 +02:00
biome_ids = {}
2019-08-26 21:03:31 +02:00
for i=1, #biomes do
local id = minetest.get_biome_id(biomes[i])
if not id then
return nil, false
end
table.insert(biome_ids, id)
end
success = search()
end
2019-08-23 10:01:41 +02:00
return spawn_pos, success
end
minetest.register_on_mods_loaded(function()
minetest.register_chatcommand("findbiome", {
description = S("Find and teleport to biome"),
params = S("<biome>"),
privs = { debug = true, teleport = true },
func = function(name, param)
local player = minetest.get_player_by_name(name)
if not player then
return false, S("No player.")
end
local pos = player:get_pos()
2019-08-26 21:03:31 +02:00
local invalid_biome = true
if mg_name == "v6" then
if not mod_biomeinfo then
return false, S("Not supported. The “biomeinfo” mod is required for v6 mapgen support!")
end
2019-08-26 21:03:31 +02:00
local biomes = biomeinfo.get_active_v6_biomes()
for b=1, #biomes do
if param == biomes[b] then
invalid_biome = false
break
end
end
else
local id = minetest.get_biome_id(param)
if id then
invalid_biome = false
2019-08-23 10:01:41 +02:00
end
end
2019-08-26 21:03:31 +02:00
if invalid_biome then
return false, S("Biome does not exist!")
end
2019-08-23 10:01:41 +02:00
local biome_pos, success = find_biome(pos, {param})
if success then
player:set_pos(biome_pos)
return true, S("Biome found at @1.", minetest.pos_to_string(biome_pos))
else
2019-08-26 21:03:31 +02:00
return false, S("No biome found!")
2019-08-23 10:01:41 +02:00
end
end,
})
minetest.register_chatcommand("listbiomes", {
description = S("List all biomes"),
params = "",
privs = { debug = true },
func = function(name, param)
2019-08-26 21:03:31 +02:00
local biomes
2019-08-23 10:01:41 +02:00
local b = 0
2019-08-26 21:03:31 +02:00
if mg_name == "v6" then
if not mod_biomeinfo then
return false, S("Not supported. The “biomeinfo” mod is required for v6 mapgen support!")
end
2019-08-26 21:03:31 +02:00
biomes = biomeinfo.get_active_v6_biomes()
b = #biomes
else
biomes = {}
for k,v in pairs(minetest.registered_biomes) do
table.insert(biomes, k)
b = b + 1
end
2019-08-23 10:01:41 +02:00
end
if b == 0 then
2019-08-26 21:03:31 +02:00
return true, S("No biomes.")
2019-08-23 10:01:41 +02:00
else
table.sort(biomes)
for b=1, #biomes do
minetest.chat_send_player(name, biomes[b])
end
return true
end
end,
})
end)