2021-02-21 23:15:32 +00:00
|
|
|
|
mcl_mapgen_core = {}
|
2021-03-28 18:56:51 +00:00
|
|
|
|
local registered_generators = {}
|
2021-02-21 23:15:32 +00:00
|
|
|
|
|
|
|
|
|
local lvm, nodes, param2 = 0, 0, 0
|
2021-03-28 18:56:51 +00:00
|
|
|
|
local lvm_buffer = {}
|
2021-02-21 23:15:32 +00:00
|
|
|
|
|
2022-08-30 01:39:29 +00:00
|
|
|
|
local modname = minetest.get_current_modname()
|
|
|
|
|
local modpath = minetest.get_modpath(modname)
|
|
|
|
|
|
2015-06-29 17:55:56 +00:00
|
|
|
|
--
|
|
|
|
|
-- Aliases for map generator outputs
|
|
|
|
|
--
|
|
|
|
|
|
|
|
|
|
minetest.register_alias("mapgen_air", "air")
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_stone", "mcl_core:stone")
|
|
|
|
|
minetest.register_alias("mapgen_tree", "mcl_core:tree")
|
|
|
|
|
minetest.register_alias("mapgen_leaves", "mcl_core:leaves")
|
|
|
|
|
minetest.register_alias("mapgen_jungletree", "mcl_core:jungletree")
|
|
|
|
|
minetest.register_alias("mapgen_jungleleaves", "mcl_core:jungleleaves")
|
2017-03-07 21:15:58 +00:00
|
|
|
|
minetest.register_alias("mapgen_pine_tree", "mcl_core:sprucetree")
|
|
|
|
|
minetest.register_alias("mapgen_pine_needles", "mcl_core:spruceleaves")
|
2017-01-31 22:32:56 +00:00
|
|
|
|
|
|
|
|
|
minetest.register_alias("mapgen_apple", "mcl_core:leaves")
|
|
|
|
|
minetest.register_alias("mapgen_water_source", "mcl_core:water_source")
|
|
|
|
|
minetest.register_alias("mapgen_dirt", "mcl_core:dirt")
|
|
|
|
|
minetest.register_alias("mapgen_dirt_with_grass", "mcl_core:dirt_with_grass")
|
2017-04-01 04:44:02 +00:00
|
|
|
|
minetest.register_alias("mapgen_dirt_with_snow", "mcl_core:dirt_with_grass_snow")
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_sand", "mcl_core:sand")
|
|
|
|
|
minetest.register_alias("mapgen_gravel", "mcl_core:gravel")
|
|
|
|
|
minetest.register_alias("mapgen_clay", "mcl_core:clay")
|
2017-05-19 23:27:09 +00:00
|
|
|
|
minetest.register_alias("mapgen_lava_source", "air") -- Built-in lava generator is too unpredictable, we generate lava on our own
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_cobble", "mcl_core:cobble")
|
|
|
|
|
minetest.register_alias("mapgen_mossycobble", "mcl_core:mossycobble")
|
2017-03-15 02:00:20 +00:00
|
|
|
|
minetest.register_alias("mapgen_junglegrass", "mcl_flowers:fern")
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_stone_with_coal", "mcl_core:stone_with_coal")
|
|
|
|
|
minetest.register_alias("mapgen_stone_with_iron", "mcl_core:stone_with_iron")
|
|
|
|
|
minetest.register_alias("mapgen_desert_sand", "mcl_core:sand")
|
|
|
|
|
minetest.register_alias("mapgen_desert_stone", "mcl_core:sandstone")
|
|
|
|
|
minetest.register_alias("mapgen_sandstone", "mcl_core:sandstone")
|
2017-11-30 13:16:04 +00:00
|
|
|
|
if minetest.get_modpath("mclx_core") then
|
|
|
|
|
minetest.register_alias("mapgen_river_water_source", "mclx_core:river_water_source")
|
|
|
|
|
else
|
|
|
|
|
minetest.register_alias("mapgen_river_water_source", "mcl_core:water_source")
|
|
|
|
|
end
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_snow", "mcl_core:snow")
|
|
|
|
|
minetest.register_alias("mapgen_snowblock", "mcl_core:snowblock")
|
|
|
|
|
minetest.register_alias("mapgen_ice", "mcl_core:ice")
|
2017-01-27 13:04:30 +00:00
|
|
|
|
|
2017-06-05 16:40:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_stair_cobble", "mcl_stairs:stair_cobble")
|
2017-01-31 22:32:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_sandstonebrick", "mcl_core:sandstonesmooth")
|
2017-06-05 16:40:56 +00:00
|
|
|
|
minetest.register_alias("mapgen_stair_sandstonebrick", "mcl_stairs:stair_sandstone")
|
|
|
|
|
minetest.register_alias("mapgen_stair_sandstone_block", "mcl_stairs:stair_sandstone")
|
|
|
|
|
minetest.register_alias("mapgen_stair_desert_stone", "mcl_stairs:stair_sandstone")
|
2017-01-27 13:04:30 +00:00
|
|
|
|
|
2017-06-13 11:53:34 +00:00
|
|
|
|
local mg_name = minetest.get_mapgen_setting("mg_name")
|
2019-02-09 01:42:11 +00:00
|
|
|
|
local superflat = mg_name == "flat" and minetest.get_mapgen_setting("mcl_superflat_classic") == "true"
|
2017-06-13 11:53:34 +00:00
|
|
|
|
|
2017-09-10 18:16:13 +00:00
|
|
|
|
local WITCH_HUT_HEIGHT = 3 -- Exact Y level to spawn witch huts at. This height refers to the height of the floor
|
|
|
|
|
|
2021-04-06 13:48:17 +00:00
|
|
|
|
-- End exit portal position
|
2021-04-06 18:08:20 +00:00
|
|
|
|
local END_EXIT_PORTAL_POS = vector.new(-3, -27003, -3)
|
2017-11-21 06:24:56 +00:00
|
|
|
|
|
2017-09-11 02:24:03 +00:00
|
|
|
|
-- Content IDs
|
|
|
|
|
local c_bedrock = minetest.get_content_id("mcl_core:bedrock")
|
2017-11-21 04:39:27 +00:00
|
|
|
|
local c_obsidian = minetest.get_content_id("mcl_core:obsidian")
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local c_stone = minetest.get_content_id("mcl_core:stone")
|
|
|
|
|
local c_dirt = minetest.get_content_id("mcl_core:dirt")
|
|
|
|
|
local c_dirt_with_grass = minetest.get_content_id("mcl_core:dirt_with_grass")
|
|
|
|
|
local c_dirt_with_grass_snow = minetest.get_content_id("mcl_core:dirt_with_grass_snow")
|
2022-04-18 02:28:23 +00:00
|
|
|
|
local c_reeds = minetest.get_content_id("mcl_core:reeds")
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local c_sand = minetest.get_content_id("mcl_core:sand")
|
2021-04-17 07:26:37 +00:00
|
|
|
|
--local c_sandstone = minetest.get_content_id("mcl_core:sandstone")
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local c_void = minetest.get_content_id("mcl_core:void")
|
|
|
|
|
local c_lava = minetest.get_content_id("mcl_core:lava_source")
|
|
|
|
|
local c_water = minetest.get_content_id("mcl_core:water_source")
|
|
|
|
|
local c_soul_sand = minetest.get_content_id("mcl_nether:soul_sand")
|
|
|
|
|
local c_netherrack = minetest.get_content_id("mcl_nether:netherrack")
|
|
|
|
|
local c_nether_lava = minetest.get_content_id("mcl_nether:nether_lava_source")
|
2021-04-17 07:26:37 +00:00
|
|
|
|
--local c_end_stone = minetest.get_content_id("mcl_end:end_stone")
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local c_realm_barrier = minetest.get_content_id("mcl_core:realm_barrier")
|
|
|
|
|
local c_top_snow = minetest.get_content_id("mcl_core:snow")
|
|
|
|
|
local c_snow_block = minetest.get_content_id("mcl_core:snowblock")
|
|
|
|
|
local c_clay = minetest.get_content_id("mcl_core:clay")
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local c_leaves = minetest.get_content_id("mcl_core:leaves")
|
|
|
|
|
local c_jungleleaves = minetest.get_content_id("mcl_core:jungleleaves")
|
2021-04-17 07:26:37 +00:00
|
|
|
|
--local c_jungletree = minetest.get_content_id("mcl_core:jungletree")
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local c_cocoa_1 = minetest.get_content_id("mcl_cocoas:cocoa_1")
|
|
|
|
|
local c_cocoa_2 = minetest.get_content_id("mcl_cocoas:cocoa_2")
|
|
|
|
|
local c_cocoa_3 = minetest.get_content_id("mcl_cocoas:cocoa_3")
|
|
|
|
|
local c_vine = minetest.get_content_id("mcl_core:vine")
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local c_air = minetest.CONTENT_AIR
|
|
|
|
|
|
2022-08-30 01:39:29 +00:00
|
|
|
|
dofile(modpath.."/ores.lua")
|
2015-06-29 17:55:56 +00:00
|
|
|
|
|
2020-06-15 23:46:07 +00:00
|
|
|
|
local mg_flags = minetest.settings:get_flags("mg_flags")
|
2020-06-16 00:33:51 +00:00
|
|
|
|
|
|
|
|
|
-- Inform other mods of dungeon setting for MCL2-style dungeons
|
|
|
|
|
mcl_vars.mg_dungeons = mg_flags.dungeons and not superflat
|
|
|
|
|
|
2020-06-15 23:46:07 +00:00
|
|
|
|
-- Disable builtin dungeons, we provide our own dungeons
|
|
|
|
|
mg_flags.dungeons = false
|
|
|
|
|
|
2022-08-30 01:40:12 +00:00
|
|
|
|
|
|
|
|
|
if superflat then
|
2020-06-15 23:46:07 +00:00
|
|
|
|
-- Enforce superflat-like mapgen: no caves, decor, lakes and hills
|
|
|
|
|
mg_flags.caves = false
|
|
|
|
|
mg_flags.decorations = false
|
2019-02-09 01:42:11 +00:00
|
|
|
|
minetest.set_mapgen_setting("mgflat_spflags", "nolakes,nohills", true)
|
2017-04-01 15:01:15 +00:00
|
|
|
|
end
|
|
|
|
|
|
2020-06-15 23:46:07 +00:00
|
|
|
|
local mg_flags_str = ""
|
|
|
|
|
for k,v in pairs(mg_flags) do
|
|
|
|
|
if v == false then
|
|
|
|
|
k = "no" .. k
|
|
|
|
|
end
|
|
|
|
|
mg_flags_str = mg_flags_str .. k .. ","
|
|
|
|
|
end
|
|
|
|
|
if string.len(mg_flags_str) > 0 then
|
|
|
|
|
mg_flags_str = string.sub(mg_flags_str, 1, string.len(mg_flags_str)-1)
|
|
|
|
|
end
|
|
|
|
|
minetest.set_mapgen_setting("mg_flags", mg_flags_str, true)
|
|
|
|
|
|
2017-09-10 18:35:57 +00:00
|
|
|
|
-- Helper function for converting a MC probability to MT, with
|
|
|
|
|
-- regards to MapBlocks.
|
|
|
|
|
-- Some MC generated structures are generated on per-chunk
|
|
|
|
|
-- probability.
|
|
|
|
|
-- The MC probability is 1/x per Minecraft chunk (16×16).
|
|
|
|
|
|
|
|
|
|
-- x: The MC probability is 1/x.
|
|
|
|
|
-- minp, maxp: MapBlock limits
|
|
|
|
|
-- returns: Probability (1/return_value) for a single MT mapblock
|
|
|
|
|
local function minecraft_chunk_probability(x, minp, maxp)
|
|
|
|
|
-- 256 is the MC chunk height
|
|
|
|
|
return x * (((maxp.x-minp.x+1)*(maxp.z-minp.z+1)) / 256)
|
|
|
|
|
end
|
|
|
|
|
|
2017-09-10 22:28:52 +00:00
|
|
|
|
-- Takes an index of a biomemap table (from minetest.get_mapgen_object),
|
|
|
|
|
-- minp and maxp (from an on_generated callback) and returns the real world coordinates
|
|
|
|
|
-- as X, Z.
|
|
|
|
|
-- Inverse function of xz_to_biomemap
|
2021-05-25 10:52:25 +00:00
|
|
|
|
--[[local function biomemap_to_xz(index, minp, maxp)
|
2017-09-10 22:28:52 +00:00
|
|
|
|
local xwidth = maxp.x - minp.x + 1
|
|
|
|
|
local zwidth = maxp.z - minp.z + 1
|
|
|
|
|
local x = ((index-1) % xwidth) + minp.x
|
|
|
|
|
local z = ((index-1) / zwidth) + minp.z
|
|
|
|
|
return x, z
|
2021-04-17 07:26:37 +00:00
|
|
|
|
end]]
|
2017-09-10 22:28:52 +00:00
|
|
|
|
|
|
|
|
|
-- Takes x and z coordinates and minp and maxp of a generated chunk
|
|
|
|
|
-- (in on_generated callback) and returns a biomemap index)
|
|
|
|
|
-- Inverse function of biomemap_to_xz
|
2021-05-25 10:52:25 +00:00
|
|
|
|
local function xz_to_biomemap_index(x, z, minp, maxp)
|
2017-09-10 22:28:52 +00:00
|
|
|
|
local xwidth = maxp.x - minp.x + 1
|
|
|
|
|
local zwidth = maxp.z - minp.z + 1
|
|
|
|
|
local minix = x % xwidth
|
|
|
|
|
local miniz = z % zwidth
|
|
|
|
|
|
|
|
|
|
return (minix + miniz * zwidth) + 1
|
|
|
|
|
end
|
|
|
|
|
|
2017-05-09 14:30:30 +00:00
|
|
|
|
-- Perlin noise objects
|
2017-05-27 01:19:19 +00:00
|
|
|
|
local perlin_structures
|
2017-05-27 01:50:35 +00:00
|
|
|
|
local perlin_vines, perlin_vines_fine, perlin_vines_upwards, perlin_vines_length, perlin_vines_density
|
2017-09-11 01:54:38 +00:00
|
|
|
|
local perlin_clay
|
2017-05-09 14:30:30 +00:00
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local function generate_clay(minp, maxp, blockseed, voxelmanip_data, voxelmanip_area, lvm_used)
|
2017-09-11 02:24:03 +00:00
|
|
|
|
-- TODO: Make clay generation reproducible for same seed.
|
2017-09-11 01:54:38 +00:00
|
|
|
|
if maxp.y < -5 or minp.y > 0 then
|
2017-09-12 16:09:26 +00:00
|
|
|
|
return lvm_used
|
2017-09-11 01:54:38 +00:00
|
|
|
|
end
|
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local pr = PseudoRandom(blockseed)
|
|
|
|
|
|
2017-09-11 01:54:38 +00:00
|
|
|
|
perlin_clay = perlin_clay or minetest.get_perlin({
|
|
|
|
|
offset = 0.5,
|
|
|
|
|
scale = 0.2,
|
|
|
|
|
spread = {x = 5, y = 5, z = 5},
|
|
|
|
|
seed = -316,
|
|
|
|
|
octaves = 1,
|
|
|
|
|
persist = 0.0
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
for y=math.max(minp.y, 0), math.min(maxp.y, -8), -1 do
|
2015-06-29 17:55:56 +00:00
|
|
|
|
-- Assume X and Z lengths are equal
|
|
|
|
|
local divlen = 4
|
|
|
|
|
local divs = (maxp.x-minp.x)/divlen+1;
|
2017-09-11 01:54:38 +00:00
|
|
|
|
for divx=0+1,divs-2 do
|
|
|
|
|
for divz=0+1,divs-2 do
|
|
|
|
|
-- Get position and shift it a bit randomly so the clay do not obviously appear in a grid
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local cx = minp.x + math.floor((divx+0.5)*divlen) + pr:next(-1,1)
|
|
|
|
|
local cz = minp.z + math.floor((divz+0.5)*divlen) + pr:next(-1,1)
|
2017-09-11 02:24:03 +00:00
|
|
|
|
|
|
|
|
|
local water_pos = voxelmanip_area:index(cx, y+1, cz)
|
2017-09-12 06:11:03 +00:00
|
|
|
|
local waternode = voxelmanip_data[water_pos]
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local surface_pos = voxelmanip_area:index(cx, y, cz)
|
2017-09-12 06:11:03 +00:00
|
|
|
|
local surfacenode = voxelmanip_data[surface_pos]
|
2017-09-11 02:24:03 +00:00
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local genrnd = pr:next(1, 20)
|
2019-03-06 03:38:57 +00:00
|
|
|
|
if genrnd == 1 and perlin_clay:get_3d({x=cx,y=y,z=cz}) > 0 and waternode == c_water and
|
2017-09-11 02:24:03 +00:00
|
|
|
|
(surfacenode == c_dirt or minetest.get_item_group(minetest.get_name_from_content_id(surfacenode), "sand") == 1) then
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local diamondsize = pr:next(1, 3)
|
2017-09-11 01:54:38 +00:00
|
|
|
|
for x1 = -diamondsize, diamondsize do
|
|
|
|
|
for z1 = -(diamondsize - math.abs(x1)), diamondsize - math.abs(x1) do
|
2017-09-11 02:24:03 +00:00
|
|
|
|
local ccpos = voxelmanip_area:index(cx+x1, y, cz+z1)
|
|
|
|
|
local claycandidate = voxelmanip_data[ccpos]
|
2017-09-13 03:45:27 +00:00
|
|
|
|
if voxelmanip_data[ccpos] == c_dirt or minetest.get_item_group(minetest.get_name_from_content_id(claycandidate), "sand") == 1 then
|
2017-09-11 02:24:03 +00:00
|
|
|
|
voxelmanip_data[ccpos] = c_clay
|
|
|
|
|
lvm_used = true
|
2015-06-29 17:55:56 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
2017-09-11 01:54:38 +00:00
|
|
|
|
end
|
2015-06-29 17:55:56 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2017-04-01 15:50:28 +00:00
|
|
|
|
end
|
2017-09-11 02:24:03 +00:00
|
|
|
|
return lvm_used
|
2017-09-10 23:59:20 +00:00
|
|
|
|
end
|
|
|
|
|
|
2021-04-06 13:48:17 +00:00
|
|
|
|
local function generate_end_exit_portal(pos)
|
2021-05-09 21:57:34 +00:00
|
|
|
|
local obj = minetest.add_entity(vector.add(pos, vector.new(3, 11, 3)), "mobs_mc:enderdragon")
|
|
|
|
|
if obj then
|
|
|
|
|
local dragon_entity = obj:get_luaentity()
|
|
|
|
|
dragon_entity._initial = true
|
|
|
|
|
dragon_entity._portal_pos = pos
|
|
|
|
|
else
|
|
|
|
|
minetest.log("error", "[mcl_mapgen_core] ERROR! Ender dragon doesn't want to spawn")
|
|
|
|
|
end
|
2021-04-06 13:48:17 +00:00
|
|
|
|
mcl_structures.call_struct(pos, "end_exit_portal")
|
|
|
|
|
end
|
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local function generate_structures(minp, maxp, blockseed, biomemap)
|
2017-11-21 06:24:56 +00:00
|
|
|
|
-- End exit portal
|
2022-08-30 01:40:12 +00:00
|
|
|
|
|
2017-09-10 23:59:20 +00:00
|
|
|
|
end
|
2015-06-29 17:55:56 +00:00
|
|
|
|
|
2017-09-12 23:47:24 +00:00
|
|
|
|
-- Buffers for LuaVoxelManip
|
2021-02-21 23:15:32 +00:00
|
|
|
|
-- local lvm_buffer = {}
|
|
|
|
|
-- local lvm_buffer_param2 = {}
|
2017-02-20 05:10:49 +00:00
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
-- Generate tree decorations in the bounding box. This adds:
|
|
|
|
|
-- * Cocoa at jungle trees
|
|
|
|
|
-- * Jungle tree vines
|
|
|
|
|
-- * Oak vines in swamplands
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local function generate_tree_decorations(minp, maxp, seed, data, param2_data, area, biomemap, lvm_used, pr)
|
2017-08-21 00:28:15 +00:00
|
|
|
|
if maxp.y < 0 then
|
2017-09-12 23:47:24 +00:00
|
|
|
|
return lvm_used
|
2017-08-21 00:28:15 +00:00
|
|
|
|
end
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
local oaktree, oakleaves, jungletree, jungleleaves = {}, {}, {}, {}
|
2017-09-12 22:36:16 +00:00
|
|
|
|
local swampland = minetest.get_biome_id("Swampland")
|
|
|
|
|
local swampland_shore = minetest.get_biome_id("Swampland_shore")
|
|
|
|
|
local jungle = minetest.get_biome_id("Jungle")
|
|
|
|
|
local jungle_shore = minetest.get_biome_id("Jungle_shore")
|
|
|
|
|
local jungle_m = minetest.get_biome_id("JungleM")
|
|
|
|
|
local jungle_m_shore = minetest.get_biome_id("JungleM_shore")
|
|
|
|
|
local jungle_edge = minetest.get_biome_id("JungleEdge")
|
|
|
|
|
local jungle_edge_shore = minetest.get_biome_id("JungleEdge_shore")
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local jungle_edge_m = minetest.get_biome_id("JungleEdgeM")
|
|
|
|
|
local jungle_edge_m_shore = minetest.get_biome_id("JungleEdgeM_shore")
|
2017-09-08 19:44:15 +00:00
|
|
|
|
|
2017-09-12 03:20:03 +00:00
|
|
|
|
-- Modifier for Jungle M biome: More vines and cocoas
|
|
|
|
|
local dense_vegetation = false
|
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
if biomemap then
|
|
|
|
|
-- Biome map available: Check if the required biome (jungle or swampland)
|
|
|
|
|
-- is in this mapchunk. We are only interested in trees in the correct biome.
|
|
|
|
|
-- The nodes are added if the correct biome is *anywhere* in the mapchunk.
|
|
|
|
|
-- TODO: Strictly generate vines in the correct biomes only.
|
|
|
|
|
local swamp_biome_found, jungle_biome_found = false, false
|
|
|
|
|
for b=1, #biomemap do
|
|
|
|
|
local id = biomemap[b]
|
2017-09-10 22:28:52 +00:00
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
if not swamp_biome_found and (id == swampland or id == swampland_shore) then
|
|
|
|
|
oaktree = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:tree"})
|
|
|
|
|
oakleaves = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:leaves"})
|
|
|
|
|
swamp_biome_found = true
|
2017-09-12 03:20:03 +00:00
|
|
|
|
end
|
2017-09-12 23:47:24 +00:00
|
|
|
|
if not jungle_biome_found and (id == jungle or id == jungle_shore or id == jungle_m or id == jungle_m_shore or id == jungle_edge or id == jungle_edge_shore or id == jungle_edge_m or id == jungle_edge_m_shore) then
|
2017-09-08 19:44:15 +00:00
|
|
|
|
jungletree = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:jungletree"})
|
|
|
|
|
jungleleaves = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:jungleleaves"})
|
|
|
|
|
jungle_biome_found = true
|
|
|
|
|
end
|
2017-09-12 03:20:03 +00:00
|
|
|
|
if not dense_vegetation and (id == jungle_m or id == jungle_m_shore) then
|
|
|
|
|
dense_vegetation = true
|
|
|
|
|
end
|
|
|
|
|
if swamp_biome_found and jungle_biome_found and dense_vegetation then
|
2017-09-08 19:44:15 +00:00
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
else
|
|
|
|
|
-- If there is no biome map, we just count all jungle things we can find.
|
|
|
|
|
-- Oak vines will not be generated.
|
|
|
|
|
jungletree = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:jungletree"})
|
|
|
|
|
jungleleaves = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:jungleleaves"})
|
|
|
|
|
end
|
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
local pos, treepos, dir
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-09-12 06:11:03 +00:00
|
|
|
|
local cocoachance = 40
|
2017-09-12 03:20:03 +00:00
|
|
|
|
if dense_vegetation then
|
|
|
|
|
cocoachance = 32
|
|
|
|
|
end
|
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
-- Pass 1: Generate cocoas at jungle trees
|
2017-08-21 00:28:15 +00:00
|
|
|
|
for n = 1, #jungletree do
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2019-03-22 10:20:34 +00:00
|
|
|
|
pos = table.copy(jungletree[n])
|
2017-08-21 00:28:15 +00:00
|
|
|
|
treepos = table.copy(pos)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
if minetest.find_node_near(pos, 1, {"mcl_core:jungleleaves"}) then
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
dir = pr:next(1, cocoachance)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
if dir == 1 then
|
|
|
|
|
pos.z = pos.z + 1
|
|
|
|
|
elseif dir == 2 then
|
|
|
|
|
pos.z = pos.z - 1
|
|
|
|
|
elseif dir == 3 then
|
|
|
|
|
pos.x = pos.x + 1
|
|
|
|
|
elseif dir == 4 then
|
|
|
|
|
pos.x = pos.x -1
|
|
|
|
|
end
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local p_pos = area:index(pos.x, pos.y, pos.z)
|
2017-11-18 20:39:00 +00:00
|
|
|
|
local l = minetest.get_node_light(pos)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
if dir < 5
|
2017-09-12 23:47:24 +00:00
|
|
|
|
and data[p_pos] == c_air
|
2021-05-29 14:12:33 +00:00
|
|
|
|
and l and l > 12 then
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local c = pr:next(1, 3)
|
2017-09-12 23:47:24 +00:00
|
|
|
|
if c == 1 then
|
|
|
|
|
data[p_pos] = c_cocoa_1
|
|
|
|
|
elseif c == 2 then
|
|
|
|
|
data[p_pos] = c_cocoa_2
|
|
|
|
|
else
|
|
|
|
|
data[p_pos] = c_cocoa_3
|
|
|
|
|
end
|
|
|
|
|
param2_data[p_pos] = minetest.dir_to_facedir(vector.subtract(treepos, pos))
|
|
|
|
|
lvm_used = true
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
2017-08-21 00:28:15 +00:00
|
|
|
|
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
2017-08-21 00:28:15 +00:00
|
|
|
|
end
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
-- Pass 2: Generate vines at jungle wood, jungle leaves in jungle and oak wood, oak leaves in swampland
|
2017-08-21 00:28:15 +00:00
|
|
|
|
perlin_vines = perlin_vines or minetest.get_perlin(555, 4, 0.6, 500)
|
|
|
|
|
perlin_vines_fine = perlin_vines_fine or minetest.get_perlin(43000, 3, 0.6, 1)
|
|
|
|
|
perlin_vines_length = perlin_vines_length or minetest.get_perlin(435, 4, 0.6, 75)
|
|
|
|
|
perlin_vines_upwards = perlin_vines_upwards or minetest.get_perlin(436, 3, 0.6, 10)
|
|
|
|
|
perlin_vines_density = perlin_vines_density or minetest.get_perlin(436, 3, 0.6, 500)
|
2017-09-12 03:20:03 +00:00
|
|
|
|
|
|
|
|
|
-- Extra long vines in Jungle M
|
|
|
|
|
local maxvinelength = 7
|
|
|
|
|
if dense_vegetation then
|
|
|
|
|
maxvinelength = 14
|
|
|
|
|
end
|
2017-09-08 19:44:15 +00:00
|
|
|
|
local treething
|
|
|
|
|
for i=1, 4 do
|
|
|
|
|
if i==1 then
|
|
|
|
|
treething = jungletree
|
|
|
|
|
elseif i == 2 then
|
|
|
|
|
treething = jungleleaves
|
|
|
|
|
elseif i == 3 then
|
|
|
|
|
treething = oaktree
|
|
|
|
|
elseif i == 4 then
|
|
|
|
|
treething = oakleaves
|
|
|
|
|
end
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-09-08 19:44:15 +00:00
|
|
|
|
for n = 1, #treething do
|
|
|
|
|
pos = treething[n]
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
treepos = table.copy(pos)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
local dirs = {
|
|
|
|
|
{x=1,y=0,z=0},
|
|
|
|
|
{x=-1,y=0,z=0},
|
|
|
|
|
{x=0,y=0,z=1},
|
|
|
|
|
{x=0,y=0,z=-1},
|
|
|
|
|
}
|
2017-05-27 01:19:19 +00:00
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
for d = 1, #dirs do
|
|
|
|
|
local pos = vector.add(pos, dirs[d])
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local p_pos = area:index(pos.x, pos.y, pos.z)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
|
2019-03-06 03:38:57 +00:00
|
|
|
|
local vine_threshold = math.max(0.33333, perlin_vines_density:get_2d(pos))
|
2017-09-12 23:57:03 +00:00
|
|
|
|
if dense_vegetation then
|
|
|
|
|
vine_threshold = vine_threshold * (2/3)
|
|
|
|
|
end
|
|
|
|
|
|
2019-03-06 03:38:57 +00:00
|
|
|
|
if perlin_vines:get_2d(pos) > -1.0 and perlin_vines_fine:get_3d(pos) > vine_threshold and data[p_pos] == c_air then
|
2017-08-21 00:28:15 +00:00
|
|
|
|
|
2018-01-07 15:00:21 +00:00
|
|
|
|
local rdir = {}
|
|
|
|
|
rdir.x = -dirs[d].x
|
|
|
|
|
rdir.y = dirs[d].y
|
|
|
|
|
rdir.z = -dirs[d].z
|
|
|
|
|
local param2 = minetest.dir_to_wallmounted(rdir)
|
2017-08-21 00:28:15 +00:00
|
|
|
|
|
|
|
|
|
-- Determine growth direction
|
|
|
|
|
local grow_upwards = false
|
|
|
|
|
-- Only possible on the wood, not on the leaves
|
|
|
|
|
if i == 1 then
|
2019-03-06 03:38:57 +00:00
|
|
|
|
grow_upwards = perlin_vines_upwards:get_3d(pos) > 0.8
|
2017-08-21 00:28:15 +00:00
|
|
|
|
end
|
|
|
|
|
if grow_upwards then
|
|
|
|
|
-- Grow vines up 1-4 nodes, even through jungleleaves.
|
|
|
|
|
-- This may give climbing access all the way to the top of the tree :-)
|
|
|
|
|
-- But this will be fairly rare.
|
2019-03-06 03:38:57 +00:00
|
|
|
|
local length = math.ceil(math.abs(perlin_vines_length:get_3d(pos)) * 4)
|
2017-08-21 00:28:15 +00:00
|
|
|
|
for l=0, length-1 do
|
2017-09-12 23:47:24 +00:00
|
|
|
|
local t_pos = area:index(treepos.x, treepos.y, treepos.z)
|
|
|
|
|
|
2017-09-13 03:45:27 +00:00
|
|
|
|
if (data[p_pos] == c_air or data[p_pos] == c_jungleleaves or data[p_pos] == c_leaves) and mcl_core.supports_vines(minetest.get_name_from_content_id(data[t_pos])) then
|
2017-09-12 23:47:24 +00:00
|
|
|
|
data[p_pos] = c_vine
|
|
|
|
|
param2_data[p_pos] = param2
|
|
|
|
|
lvm_used = true
|
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
else
|
|
|
|
|
break
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
2017-08-21 00:28:15 +00:00
|
|
|
|
pos.y = pos.y + 1
|
2017-09-12 23:47:24 +00:00
|
|
|
|
p_pos = area:index(pos.x, pos.y, pos.z)
|
2017-08-21 00:28:15 +00:00
|
|
|
|
treepos.y = treepos.y + 1
|
|
|
|
|
end
|
|
|
|
|
else
|
2017-09-12 03:20:03 +00:00
|
|
|
|
-- Grow vines down, length between 1 and maxvinelength
|
2019-03-06 03:38:57 +00:00
|
|
|
|
local length = math.ceil(math.abs(perlin_vines_length:get_3d(pos)) * maxvinelength)
|
2017-08-21 00:28:15 +00:00
|
|
|
|
for l=0, length-1 do
|
2017-09-12 23:47:24 +00:00
|
|
|
|
if data[p_pos] == c_air then
|
|
|
|
|
data[p_pos] = c_vine
|
|
|
|
|
param2_data[p_pos] = param2
|
|
|
|
|
lvm_used = true
|
|
|
|
|
|
2017-08-21 00:28:15 +00:00
|
|
|
|
else
|
|
|
|
|
break
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
2017-08-21 00:28:15 +00:00
|
|
|
|
pos.y = pos.y - 1
|
2017-09-12 23:47:24 +00:00
|
|
|
|
p_pos = area:index(pos.x, pos.y, pos.z)
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
2017-08-21 00:28:15 +00:00
|
|
|
|
end
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
2017-09-12 23:47:24 +00:00
|
|
|
|
|
2017-05-26 23:54:40 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
2017-09-12 23:47:24 +00:00
|
|
|
|
return lvm_used
|
2017-08-15 18:08:41 +00:00
|
|
|
|
end
|
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
minetest.register_on_generated(function(minp, maxp, blockseed)
|
2021-03-21 23:14:33 +00:00
|
|
|
|
minetest.log("action", "[mcl_mapgen_core] Generating chunk " .. minetest.pos_to_string(minp) .. " ... " .. minetest.pos_to_string(maxp))
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local p1, p2 = {x=minp.x, y=minp.y, z=minp.z}, {x=maxp.x, y=maxp.y, z=maxp.z}
|
|
|
|
|
if lvm > 0 then
|
|
|
|
|
local lvm_used, shadow = false, false
|
|
|
|
|
local lb2 = {} -- param2
|
|
|
|
|
local vm, emin, emax = minetest.get_mapgen_object("voxelmanip")
|
|
|
|
|
local e1, e2 = {x=emin.x, y=emin.y, z=emin.z}, {x=emax.x, y=emax.y, z=emax.z}
|
|
|
|
|
local data2
|
2021-03-28 18:56:51 +00:00
|
|
|
|
local data = vm:get_data(lvm_buffer)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if param2 > 0 then
|
|
|
|
|
data2 = vm:get_param2_data(lb2)
|
|
|
|
|
end
|
|
|
|
|
local area = VoxelArea:new({MinEdge=e1, MaxEdge=e2})
|
|
|
|
|
|
2022-08-29 04:11:02 +00:00
|
|
|
|
for _, rec in ipairs(registered_generators) do
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if rec.vf then
|
2021-03-28 18:56:51 +00:00
|
|
|
|
local lvm_used0, shadow0 = rec.vf(vm, data, data2, e1, e2, area, p1, p2, blockseed)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if lvm_used0 then
|
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
|
|
|
|
if shadow0 then
|
|
|
|
|
shadow = true
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if lvm_used then
|
|
|
|
|
-- Write stuff
|
|
|
|
|
vm:set_data(data)
|
|
|
|
|
if param2 > 0 then
|
|
|
|
|
vm:set_param2_data(data2)
|
|
|
|
|
end
|
|
|
|
|
vm:calc_lighting(p1, p2, shadow)
|
|
|
|
|
vm:write_to_map()
|
|
|
|
|
vm:update_liquids()
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if nodes > 0 then
|
2022-08-29 04:11:02 +00:00
|
|
|
|
for _, rec in ipairs(registered_generators) do
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if rec.nf then
|
|
|
|
|
rec.nf(p1, p2, blockseed)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2021-03-28 18:56:51 +00:00
|
|
|
|
mcl_vars.add_chunk(minp)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
end)
|
|
|
|
|
|
2021-05-27 07:34:12 +00:00
|
|
|
|
function minetest.register_on_generated(node_function)
|
2022-08-29 04:11:02 +00:00
|
|
|
|
mcl_mapgen_core.register_generator("mod_"..minetest.get_current_modname().."_"..tostring(#registered_generators+1), nil, node_function)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function mcl_mapgen_core.register_generator(id, lvm_function, node_function, priority, needs_param2)
|
|
|
|
|
if not id then return end
|
|
|
|
|
|
|
|
|
|
local priority = priority or 5000
|
|
|
|
|
|
|
|
|
|
if lvm_function then lvm = lvm + 1 end
|
2022-08-14 07:11:30 +00:00
|
|
|
|
if node_function then nodes = nodes + 1 end
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if needs_param2 then param2 = param2 + 1 end
|
|
|
|
|
|
|
|
|
|
local new_record = {
|
2022-08-29 04:11:02 +00:00
|
|
|
|
id = id,
|
2021-02-21 23:15:32 +00:00
|
|
|
|
i = priority,
|
|
|
|
|
vf = lvm_function,
|
|
|
|
|
nf = node_function,
|
|
|
|
|
needs_param2 = needs_param2,
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-29 04:11:02 +00:00
|
|
|
|
table.insert(registered_generators, new_record)
|
2021-05-29 14:12:33 +00:00
|
|
|
|
table.sort(registered_generators, function(a, b)
|
|
|
|
|
return (a.i < b.i) or ((a.i == b.i) and a.vf and (b.vf == nil))
|
|
|
|
|
end)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
function mcl_mapgen_core.unregister_generator(id)
|
2022-08-29 04:11:02 +00:00
|
|
|
|
local index
|
|
|
|
|
for i, gen in ipairs(registered_generators) do
|
|
|
|
|
if gen.id == id then
|
|
|
|
|
index = i
|
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
if not index then return end
|
|
|
|
|
local rec = registered_generators[index]
|
|
|
|
|
table.remove(registered_generators, index)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if rec.vf then lvm = lvm - 1 end
|
2021-04-17 07:26:37 +00:00
|
|
|
|
if rec.nf then nodes = nodes - 1 end
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if rec.needs_param2 then param2 = param2 - 1 end
|
2021-05-22 18:01:59 +00:00
|
|
|
|
--if rec.needs_level0 then level0 = level0 - 1 end
|
2021-02-21 23:15:32 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- Generate basic layer-based nodes: void, bedrock, realm barrier, lava seas, etc.
|
|
|
|
|
-- Also perform some basic node replacements.
|
|
|
|
|
|
|
|
|
|
local bedrock_check
|
|
|
|
|
if mcl_vars.mg_bedrock_is_rough then
|
2021-05-25 10:52:25 +00:00
|
|
|
|
function bedrock_check(pos, _, pr)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local y = pos.y
|
|
|
|
|
-- Bedrock layers with increasing levels of roughness, until a perfecly flat bedrock later at the bottom layer
|
|
|
|
|
-- This code assumes a bedrock height of 5 layers.
|
|
|
|
|
|
|
|
|
|
local diff = mcl_vars.mg_bedrock_overworld_max - y -- Overworld bedrock
|
|
|
|
|
local ndiff1 = mcl_vars.mg_bedrock_nether_bottom_max - y -- Nether bedrock, bottom
|
|
|
|
|
local ndiff2 = mcl_vars.mg_bedrock_nether_top_max - y -- Nether bedrock, ceiling
|
|
|
|
|
|
|
|
|
|
local top
|
|
|
|
|
if diff == 0 or ndiff1 == 0 or ndiff2 == 4 then
|
|
|
|
|
-- 50% bedrock chance
|
|
|
|
|
top = 2
|
|
|
|
|
elseif diff == 1 or ndiff1 == 1 or ndiff2 == 3 then
|
|
|
|
|
-- 66.666...%
|
|
|
|
|
top = 3
|
|
|
|
|
elseif diff == 2 or ndiff1 == 2 or ndiff2 == 2 then
|
|
|
|
|
-- 75%
|
|
|
|
|
top = 4
|
|
|
|
|
elseif diff == 3 or ndiff1 == 3 or ndiff2 == 1 then
|
|
|
|
|
-- 90%
|
|
|
|
|
top = 10
|
|
|
|
|
elseif diff == 4 or ndiff1 == 4 or ndiff2 == 0 then
|
|
|
|
|
-- 100%
|
|
|
|
|
return true
|
|
|
|
|
else
|
|
|
|
|
-- Not in bedrock layer
|
|
|
|
|
return false
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
return pr:next(1, top) <= top-1
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
-- Helper function to set all nodes in the layers between min and max.
|
|
|
|
|
-- content_id: Node to set
|
|
|
|
|
-- check: optional.
|
|
|
|
|
-- If content_id, node will be set only if it is equal to check.
|
|
|
|
|
-- If function(pos_to_check, content_id_at_this_pos), will set node only if returns true.
|
|
|
|
|
-- min, max: Minimum and maximum Y levels of the layers to set
|
|
|
|
|
-- minp, maxp: minp, maxp of the on_generated
|
|
|
|
|
-- lvm_used: Set to true if any node in this on_generated has been set before.
|
|
|
|
|
--
|
|
|
|
|
-- returns true if any node was set and lvm_used otherwise
|
|
|
|
|
local function set_layers(data, area, content_id, check, min, max, minp, maxp, lvm_used, pr)
|
|
|
|
|
if (maxp.y >= min and minp.y <= max) then
|
|
|
|
|
for y = math.max(min, minp.y), math.min(max, maxp.y) do
|
|
|
|
|
for x = minp.x, maxp.x do
|
|
|
|
|
for z = minp.z, maxp.z do
|
|
|
|
|
local p_pos = area:index(x, y, z)
|
|
|
|
|
if check then
|
|
|
|
|
if type(check) == "function" and check({x=x,y=y,z=z}, data[p_pos], pr) then
|
|
|
|
|
data[p_pos] = content_id
|
|
|
|
|
lvm_used = true
|
|
|
|
|
elseif check == data[p_pos] then
|
2017-09-06 06:36:01 +00:00
|
|
|
|
data[p_pos] = content_id
|
|
|
|
|
lvm_used = true
|
2017-08-15 18:08:41 +00:00
|
|
|
|
end
|
2021-02-21 23:15:32 +00:00
|
|
|
|
else
|
|
|
|
|
data[p_pos] = content_id
|
|
|
|
|
lvm_used = true
|
2017-08-15 18:08:41 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2017-09-06 06:36:01 +00:00
|
|
|
|
end
|
2021-02-21 23:15:32 +00:00
|
|
|
|
return lvm_used
|
|
|
|
|
end
|
2017-09-06 06:36:01 +00:00
|
|
|
|
|
2021-02-21 23:15:32 +00:00
|
|
|
|
-- Below the bedrock, generate air/void
|
|
|
|
|
local function basic(vm, data, data2, emin, emax, area, minp, maxp, blockseed)
|
2021-04-17 07:26:37 +00:00
|
|
|
|
local biomemap --ymin, ymax
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local lvm_used = false
|
|
|
|
|
local pr = PseudoRandom(blockseed)
|
2017-09-06 06:36:01 +00:00
|
|
|
|
|
2021-02-25 13:36:42 +00:00
|
|
|
|
-- The Void below the Nether:
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_void , nil, mcl_vars.mapgen_edge_min , mcl_vars.mg_nether_min -1, minp, maxp, lvm_used, pr)
|
2021-02-25 13:36:42 +00:00
|
|
|
|
|
|
|
|
|
-- [[ THE NETHER: mcl_vars.mg_nether_min mcl_vars.mg_nether_max ]]
|
|
|
|
|
|
|
|
|
|
-- The Air on the Nether roof, https://git.minetest.land/MineClone2/MineClone2/issues/1186
|
|
|
|
|
lvm_used = set_layers(data, area, c_air , nil, mcl_vars.mg_nether_max +1, mcl_vars.mg_nether_max + 128 , minp, maxp, lvm_used, pr)
|
|
|
|
|
-- The Void above the Nether below the End:
|
|
|
|
|
lvm_used = set_layers(data, area, c_void , nil, mcl_vars.mg_nether_max + 128 +1, mcl_vars.mg_end_min -1, minp, maxp, lvm_used, pr)
|
|
|
|
|
|
|
|
|
|
-- [[ THE END: mcl_vars.mg_end_min mcl_vars.mg_end_max ]]
|
|
|
|
|
|
|
|
|
|
-- The Void above the End below the Realm barrier:
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_void , nil, mcl_vars.mg_end_max +1, mcl_vars.mg_realm_barrier_overworld_end_min-1, minp, maxp, lvm_used, pr)
|
2017-09-06 06:36:01 +00:00
|
|
|
|
-- Realm barrier between the Overworld void and the End
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_realm_barrier, nil, mcl_vars.mg_realm_barrier_overworld_end_min , mcl_vars.mg_realm_barrier_overworld_end_max , minp, maxp, lvm_used, pr)
|
2021-02-25 13:36:42 +00:00
|
|
|
|
-- The Void above Realm barrier below the Overworld:
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_void , nil, mcl_vars.mg_realm_barrier_overworld_end_max+1, mcl_vars.mg_overworld_min -1, minp, maxp, lvm_used, pr)
|
2017-09-06 06:36:01 +00:00
|
|
|
|
|
2021-02-25 13:36:42 +00:00
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
if mg_name ~= "singlenode" then
|
|
|
|
|
-- Bedrock
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_bedrock, bedrock_check, mcl_vars.mg_bedrock_overworld_min, mcl_vars.mg_bedrock_overworld_max, minp, maxp, lvm_used, pr)
|
|
|
|
|
lvm_used = set_layers(data, area, c_bedrock, bedrock_check, mcl_vars.mg_bedrock_nether_bottom_min, mcl_vars.mg_bedrock_nether_bottom_max, minp, maxp, lvm_used, pr)
|
|
|
|
|
lvm_used = set_layers(data, area, c_bedrock, bedrock_check, mcl_vars.mg_bedrock_nether_top_min, mcl_vars.mg_bedrock_nether_top_max, minp, maxp, lvm_used, pr)
|
2017-09-06 06:36:01 +00:00
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- Flat Nether
|
|
|
|
|
if mg_name == "flat" then
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_air, nil, mcl_vars.mg_flat_nether_floor, mcl_vars.mg_flat_nether_ceiling, minp, maxp, lvm_used, pr)
|
2017-09-06 06:36:01 +00:00
|
|
|
|
end
|
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- Big lava seas by replacing air below a certain height
|
|
|
|
|
if mcl_vars.mg_lava then
|
2021-03-07 02:31:43 +00:00
|
|
|
|
lvm_used = set_layers(data, area, c_lava, c_air, mcl_vars.mg_overworld_min, mcl_vars.mg_lava_overworld_max, minp, maxp, lvm_used, pr)
|
|
|
|
|
lvm_used = set_layers(data, area, c_nether_lava, c_air, mcl_vars.mg_nether_min, mcl_vars.mg_lava_nether_max, minp, maxp, lvm_used, pr)
|
2017-11-07 04:29:39 +00:00
|
|
|
|
end
|
2017-09-06 06:36:01 +00:00
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- Clay, vines, cocoas
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = generate_clay(minp, maxp, blockseed, data, area, lvm_used)
|
2017-11-07 04:29:39 +00:00
|
|
|
|
|
2017-11-07 19:16:25 +00:00
|
|
|
|
biomemap = minetest.get_mapgen_object("biomemap")
|
2021-02-21 23:15:32 +00:00
|
|
|
|
lvm_used = generate_tree_decorations(minp, maxp, blockseed, data, data2, area, biomemap, lvm_used, pr)
|
2017-11-07 04:29:39 +00:00
|
|
|
|
|
|
|
|
|
----- Interactive block fixing section -----
|
|
|
|
|
----- The section to perform basic block overrides of the core mapgen generated world. -----
|
|
|
|
|
|
|
|
|
|
-- Snow and sand fixes. This code implements snow consistency
|
2019-02-09 04:58:40 +00:00
|
|
|
|
-- and fixes floating sand and cut plants.
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- A snowy grass block must be below a top snow or snow block at all times.
|
2021-03-07 02:31:43 +00:00
|
|
|
|
if minp.y <= mcl_vars.mg_overworld_max and maxp.y >= mcl_vars.mg_overworld_min then
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- v6 mapgen:
|
2022-08-30 01:40:12 +00:00
|
|
|
|
if mg_name ~= "v6" then
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- Non-v6 mapgens:
|
2019-12-13 14:33:13 +00:00
|
|
|
|
-- Set param2 (=color) of grass blocks.
|
|
|
|
|
-- Clear snowy grass blocks without snow above to ensure consistency.
|
|
|
|
|
local nodes = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:dirt_with_grass", "mcl_core:dirt_with_grass_snow"})
|
2021-02-21 23:15:32 +00:00
|
|
|
|
|
|
|
|
|
-- Flat area at y=0 to read biome 3 times faster than 5.3.0.get_biome_data(pos).biome: 43us vs 125us per iteration:
|
|
|
|
|
local aream = VoxelArea:new({MinEdge={x=minp.x, y=0, z=minp.z}, MaxEdge={x=maxp.x, y=0, z=maxp.z}})
|
2017-11-07 04:29:39 +00:00
|
|
|
|
for n=1, #nodes do
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local n = nodes[n]
|
|
|
|
|
local p_pos = area:index(n.x, n.y, n.z)
|
|
|
|
|
local p_pos_above = area:index(n.x, n.y+1, n.z)
|
2021-04-17 07:26:37 +00:00
|
|
|
|
--local p_pos_below = area:index(n.x, n.y-1, n.z)
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local b_pos = aream:index(n.x, 0, n.z)
|
2019-12-13 14:33:13 +00:00
|
|
|
|
local bn = minetest.get_biome_name(biomemap[b_pos])
|
|
|
|
|
if bn then
|
|
|
|
|
local biome = minetest.registered_biomes[bn]
|
2021-02-21 23:15:32 +00:00
|
|
|
|
if biome and biome._mcl_biome_type then
|
|
|
|
|
data2[p_pos] = biome._mcl_palette_index
|
|
|
|
|
lvm_used = true
|
2019-12-13 14:33:13 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
2017-11-07 04:29:39 +00:00
|
|
|
|
if data[p_pos] == c_dirt_with_grass_snow and p_pos_above and data[p_pos_above] ~= c_top_snow and data[p_pos_above] ~= c_snow_block then
|
|
|
|
|
data[p_pos] = c_dirt_with_grass
|
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
end
|
2022-05-26 05:29:28 +00:00
|
|
|
|
|
2022-04-18 02:28:23 +00:00
|
|
|
|
-- Set param2 (=color) of sugar cane
|
|
|
|
|
nodes = minetest.find_nodes_in_area(minp, maxp, {"mcl_core:reeds"})
|
|
|
|
|
for n=1, #nodes do
|
|
|
|
|
local n = nodes[n]
|
|
|
|
|
local p_pos = area:index(n.x, n.y, n.z)
|
|
|
|
|
local b_pos = aream:index(n.x, 0, n.z)
|
|
|
|
|
local bn = minetest.get_biome_name(biomemap[b_pos])
|
|
|
|
|
if bn then
|
|
|
|
|
local biome = minetest.registered_biomes[bn]
|
|
|
|
|
if biome and biome._mcl_biome_type then
|
|
|
|
|
data2[p_pos] = biome._mcl_palette_index
|
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
end
|
2017-09-07 22:38:55 +00:00
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- Nether block fixes:
|
|
|
|
|
-- * Replace water with Nether lava.
|
|
|
|
|
-- * Replace stone, sand dirt in v6 so the Nether works in v6.
|
2021-03-28 18:56:51 +00:00
|
|
|
|
elseif emin.y <= mcl_vars.mg_nether_max and emax.y >= mcl_vars.mg_nether_min then
|
2022-08-30 01:40:12 +00:00
|
|
|
|
if mg_name ~= "v6" then
|
2021-03-28 23:18:17 +00:00
|
|
|
|
local nodes = minetest.find_nodes_in_area(emin, emax, {"group:water"})
|
|
|
|
|
for _, n in pairs(nodes) do
|
|
|
|
|
data[area:index(n.x, n.y, n.z)] = c_nether_lava
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
end
|
|
|
|
|
|
2017-11-07 04:29:39 +00:00
|
|
|
|
-- End block fixes:
|
|
|
|
|
-- * Replace water with end stone or air (depending on height).
|
|
|
|
|
-- * Remove stone, sand, dirt in v6 so our End map generator works in v6.
|
2017-11-21 04:39:27 +00:00
|
|
|
|
-- * Generate spawn platform (End portal destination)
|
2021-03-07 02:31:43 +00:00
|
|
|
|
elseif minp.y <= mcl_vars.mg_end_max and maxp.y >= mcl_vars.mg_end_min then
|
2021-04-17 07:26:37 +00:00
|
|
|
|
local nodes
|
2022-08-30 01:40:12 +00:00
|
|
|
|
if mg_name ~= "v6" then
|
2021-03-28 22:43:08 +00:00
|
|
|
|
nodes = minetest.find_nodes_in_area(emin, emax, {"mcl_core:water_source"})
|
2022-08-30 01:40:12 +00:00
|
|
|
|
if #nodes > 0 then
|
|
|
|
|
lvm_used = true
|
|
|
|
|
for _,n in pairs(nodes) do
|
|
|
|
|
data[area:index(n.x, n.y, n.z)] = c_air
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
end
|
2017-11-07 04:29:39 +00:00
|
|
|
|
end
|
2017-11-21 04:39:27 +00:00
|
|
|
|
-- Obsidian spawn platform
|
|
|
|
|
if minp.y <= mcl_vars.mg_end_platform_pos.y and maxp.y >= mcl_vars.mg_end_platform_pos.y and
|
2021-02-21 23:15:32 +00:00
|
|
|
|
minp.x <= mcl_vars.mg_end_platform_pos.x and maxp.x >= mcl_vars.mg_end_platform_pos.z and
|
|
|
|
|
minp.z <= mcl_vars.mg_end_platform_pos.z and maxp.z >= mcl_vars.mg_end_platform_pos.z then
|
|
|
|
|
|
2021-04-17 07:26:37 +00:00
|
|
|
|
--local pos1 = {x = math.max(minp.x, mcl_vars.mg_end_platform_pos.x-2), y = math.max(minp.y, mcl_vars.mg_end_platform_pos.y), z = math.max(minp.z, mcl_vars.mg_end_platform_pos.z-2)}
|
|
|
|
|
--local pos2 = {x = math.min(maxp.x, mcl_vars.mg_end_platform_pos.x+2), y = math.min(maxp.y, mcl_vars.mg_end_platform_pos.y+2), z = math.min(maxp.z, mcl_vars.mg_end_platform_pos.z+2)}
|
2021-02-21 23:15:32 +00:00
|
|
|
|
|
2017-11-21 04:39:27 +00:00
|
|
|
|
for x=math.max(minp.x, mcl_vars.mg_end_platform_pos.x-2), math.min(maxp.x, mcl_vars.mg_end_platform_pos.x+2) do
|
|
|
|
|
for z=math.max(minp.z, mcl_vars.mg_end_platform_pos.z-2), math.min(maxp.z, mcl_vars.mg_end_platform_pos.z+2) do
|
|
|
|
|
for y=math.max(minp.y, mcl_vars.mg_end_platform_pos.y), math.min(maxp.y, mcl_vars.mg_end_platform_pos.y+2) do
|
|
|
|
|
local p_pos = area:index(x, y, z)
|
|
|
|
|
if y == mcl_vars.mg_end_platform_pos.y then
|
|
|
|
|
data[p_pos] = c_obsidian
|
|
|
|
|
else
|
|
|
|
|
data[p_pos] = c_air
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
2017-08-15 18:08:41 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
-- Final hackery: Set sun light level in the End.
|
|
|
|
|
-- -26912 is at a mapchunk border.
|
2021-02-21 23:15:32 +00:00
|
|
|
|
local shadow = true
|
2017-08-21 22:22:53 +00:00
|
|
|
|
if minp.y >= -26912 and maxp.y <= mcl_vars.mg_end_max then
|
|
|
|
|
vm:set_lighting({day=15, night=15})
|
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
|
|
|
|
if minp.y >= mcl_vars.mg_end_min and maxp.y <= -26911 then
|
|
|
|
|
shadow = false
|
2017-08-18 11:56:26 +00:00
|
|
|
|
lvm_used = true
|
|
|
|
|
end
|
2017-09-06 05:10:11 +00:00
|
|
|
|
|
2021-04-20 04:41:10 +00:00
|
|
|
|
return lvm_used, shadow
|
|
|
|
|
end
|
|
|
|
|
|
2022-08-30 01:40:12 +00:00
|
|
|
|
mcl_mapgen_core.register_generator("main", basic, basic_node, 1, true)
|
|
|
|
|
|
|
|
|
|
mcl_mapgen_core.register_generator("end_exit_portal",nil, function(minp, maxp, blockseed)
|
|
|
|
|
if minp.y <= END_EXIT_PORTAL_POS.y and maxp.y >= END_EXIT_PORTAL_POS.y and
|
|
|
|
|
minp.x <= END_EXIT_PORTAL_POS.x and maxp.x >= END_EXIT_PORTAL_POS.x and
|
|
|
|
|
minp.z <= END_EXIT_PORTAL_POS.z and maxp.z >= END_EXIT_PORTAL_POS.z then
|
|
|
|
|
for y=maxp.y, minp.y, -1 do
|
|
|
|
|
local p = {x=END_EXIT_PORTAL_POS.x, y=y, z=END_EXIT_PORTAL_POS.z}
|
|
|
|
|
if minetest.get_node(p).name == "mcl_end:end_stone" then
|
|
|
|
|
generate_end_exit_portal(p)
|
|
|
|
|
return
|
|
|
|
|
end
|
2022-06-22 13:56:51 +00:00
|
|
|
|
end
|
2022-08-30 01:40:12 +00:00
|
|
|
|
generate_end_exit_portal(END_EXIT_PORTAL_POS)
|
2017-11-07 04:29:39 +00:00
|
|
|
|
end
|
2022-08-30 01:40:12 +00:00
|
|
|
|
end,101,true)
|
2022-06-22 13:56:51 +00:00
|
|
|
|
|
|
|
|
|
mcl_mapgen_core.register_generator("structures",nil, function(minp, maxp, blockseed)
|
|
|
|
|
local gennotify = minetest.get_mapgen_object("gennotify")
|
2022-06-24 00:11:24 +00:00
|
|
|
|
local has_struct = {}
|
2022-08-30 01:40:12 +00:00
|
|
|
|
local has = false
|
2022-06-24 00:11:24 +00:00
|
|
|
|
local poshash = minetest.hash_node_position(minp)
|
2022-06-22 13:56:51 +00:00
|
|
|
|
for _,struct in pairs(mcl_structures.registered_structures) do
|
2022-06-24 15:54:34 +00:00
|
|
|
|
if struct.deco_id then
|
2022-07-02 01:44:13 +00:00
|
|
|
|
local pr = PseudoRandom(blockseed + 42)
|
2022-06-24 15:54:34 +00:00
|
|
|
|
for _, pos in pairs(gennotify["decoration#"..struct.deco_id] or {}) do
|
|
|
|
|
local realpos = vector.offset(pos,0,1,0)
|
|
|
|
|
minetest.remove_node(realpos)
|
|
|
|
|
if struct.chunk_probability == nil or (not has and pr:next(1,struct.chunk_probability) == 1 ) then
|
2022-07-02 01:44:13 +00:00
|
|
|
|
mcl_structures.place_structure(realpos,struct,pr,blockseed)
|
2022-06-24 15:54:34 +00:00
|
|
|
|
has=true
|
|
|
|
|
end
|
2022-06-24 00:11:24 +00:00
|
|
|
|
end
|
2022-06-22 13:56:51 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end, 100, true)
|
2022-08-30 01:40:12 +00:00
|
|
|
|
|
|
|
|
|
if mg_name == "v6" then
|
|
|
|
|
dofile(modpath.."/v6.lua")
|
|
|
|
|
end
|