329 lines
13 KiB
Lua
Raw Normal View History

2015-04-11 10:59:45 +02:00
-- Mapgen 2.0
-- Thursday April 23, 2015
2015-03-07 09:33:01 +01:00
vmg.noises = {
-- Noise 1 : Base Ground Height 2D
2015-03-09 09:04:05 +01:00
{offset = -10, scale = 50, seed = 5202, spread = {x = 1024, y = 1024, z = 1024}, octaves = 6, persist = 0.4, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 2 : Valleys (River where around zero) 2D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = -6050, spread = {x = 256, y = 256, z = 256}, octaves = 5, persist = 0.6, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 3 : Valleys Depth 2D
2015-03-09 09:04:05 +01:00
{offset = 5, scale = 4, seed = -1914, spread = {x = 512, y = 512, z = 512}, octaves = 1, persist = 1, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 4 : Valleys Profile (Higher values = Larger valleys) 2D
2015-03-14 10:34:35 +01:00
{offset = 0.6, scale = 0.5, seed = 777, spread = {x = 512, y = 512, z = 512}, octaves = 1, persist = 1, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 5 : Inter-valleys slopes 2D
2015-03-09 09:04:05 +01:00
{offset = 0.5, scale = 0.5, seed = 746, spread = {x = 128, y = 128, z = 128}, octaves = 1, persist = 1, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 6 : Inter-valleys filling 3D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = 1993, spread = {x = 256, y = 512, z = 256}, octaves = 6, persist = 0.8, lacunarity = 2},
2015-03-07 09:33:01 +01:00
-- Noise 7 : Dirt thickness 2D
2015-03-14 10:54:08 +01:00
{offset = 3, scale = 1.75, seed = 1605, spread = {x = 256, y = 256, z = 256}, octaves = 3, persist = 0.5, lacunarity = 2},
2015-03-07 09:33:01 +01:00
2015-03-14 10:34:35 +01:00
-- Noise 8 : Caves I 3D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = -4640, spread = {x = 32, y = 32, z = 32}, octaves = 4, persist = 0.5, lacunarity = 2},
2015-03-08 10:40:37 +01:00
2015-03-14 10:34:35 +01:00
-- Noise 9 : Caves II 3D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = 8804, spread = {x = 32, y = 32, z = 32}, octaves = 4, persist = 0.5, lacunarity = 2},
2015-03-08 10:40:37 +01:00
2015-03-14 10:34:35 +01:00
-- Noise 10 : Caves III 3D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = -4780, spread = {x = 32, y = 32, z = 32}, octaves = 4, persist = 0.5, lacunarity = 2},
2015-03-08 10:40:37 +01:00
2015-03-16 09:04:15 +01:00
-- Noise 11 : Caves IV and Lava I 3D
2015-03-09 09:04:05 +01:00
{offset = 0, scale = 1, seed = -9969, spread = {x = 32, y = 32, z = 32}, octaves = 4, persist = 0.5, lacunarity = 2},
2015-03-08 10:40:37 +01:00
2015-04-17 21:27:21 +02:00
-- Noise 12 : Lava II (Geologic heat) 3D
2015-03-16 09:04:15 +01:00
{offset = 0, scale = 1, seed = 3314, spread = {x = 64, y = 64, z = 64}, octaves = 4, persist = 0.5, lacunarity = 2},
2015-04-04 22:32:37 +02:00
-- Noise 13 : Clayey dirt noise 2D
{offset = 0, scale = 1, seed = 2835, spread = {x = 256, y = 256, z = 256}, octaves = 5, persist = 0.5, lacunarity = 4},
2015-04-04 22:32:37 +02:00
-- Noise 14 : Silty dirt noise 2D
{offset = 0, scale = 1, seed = 6674, spread = {x = 256, y = 256, z = 256}, octaves = 5, persist = 0.5, lacunarity = 4},
2015-04-04 22:32:37 +02:00
-- Noise 15 : Sandy dirt noise 2D
{offset = 0, scale = 1, seed = 6940, spread = {x = 256, y = 256, z = 256}, octaves = 5, persist = 0.5, lacunarity = 4},
2015-04-04 22:32:37 +02:00
2015-04-06 10:43:33 +02:00
-- Noise 16 : Beaches 2D
{offset = 2, scale = 8, seed = 2349, spread = {x = 256, y = 256, z = 256}, octaves = 3, persist = 0.5, lacunarity = 2},
2015-04-17 21:27:21 +02:00
-- Noise 17 : Temperature (not in maps) 3D
{offset = 2, scale = 1, seed = -1805, spread = {x = 768, y = 256, z = 768}, octaves = 4, persist = 0.5, lacunarity = 4},
-- Noise 18 : Humidity (not in maps) 2D
{offset = 0, scale = 1, seed = -5787, spread = {x = 243, y = 243, z = 243}, octaves = 4, persist = 0.5, lacunarity = 3},
2015-03-07 09:33:01 +01:00
}
2015-03-15 19:37:00 +01:00
function vmg.noisemap(i, minp, chulens)
local obj = minetest.get_perlin_map(vmg.noises[i], chulens)
if minp.z then
return obj:get3dMap_flat(minp)
else
return obj:get2dMap_flat(minp)
end
end
2015-03-09 09:04:05 +01:00
for i, n in ipairs(vmg.noises) do
vmg.noises[i] = vmg.define("noise_" .. i, n)
2015-03-09 09:04:05 +01:00
end
2015-03-14 10:54:08 +01:00
local average_stone_level = vmg.define("average_stone_level", 180)
2015-03-11 19:10:35 +01:00
local dirt_thickness = math.sqrt(average_stone_level) / (vmg.noises[7].offset + 0.5)
2015-03-13 20:11:48 +01:00
local river_size = vmg.define("river_size", 5) / 100
local caves_size = vmg.define("caves_size", 7) / 100
2015-03-16 09:04:15 +01:00
local lava_depth = vmg.define("lava_depth", 2000)
2015-04-11 11:23:52 +02:00
local lava_max_height = vmg.define("lava_max_height", -1)
2015-04-17 22:15:00 +02:00
local altitude_chill = vmg.define("altitude_chill", 90)
2015-03-13 20:11:48 +01:00
2015-03-14 10:47:15 +01:00
local player_max_distance = vmg.define("player_max_distance", 450)
2015-04-05 17:05:22 +02:00
local clay_threshold = vmg.define("clay_threshold", 1)
local silt_threshold = vmg.define("silt_threshold", 1)
local sand_threshold = vmg.define("sand_threshold", 0.75)
local dirt_threshold = vmg.define("dirt_threshold", 0.5)
2015-04-06 11:32:47 +02:00
local water_level = vmg.define("water_level", 1)
2015-03-07 09:33:01 +01:00
function vmg.generate(minp, maxp, seed)
local c_stone = minetest.get_content_id("default:stone")
2015-04-04 22:32:37 +02:00
local c_dirt = minetest.get_content_id("default:dirt")
2015-03-07 09:33:01 +01:00
local c_lawn = minetest.get_content_id("default:dirt_with_grass")
2015-04-04 22:32:37 +02:00
local c_dirt_clay = minetest.get_content_id("valleys_mapgen:dirt_clayey")
local c_lawn_clay = minetest.get_content_id("valleys_mapgen:dirt_clayey_with_grass")
local c_dirt_silt = minetest.get_content_id("valleys_mapgen:dirt_silty")
local c_lawn_silt = minetest.get_content_id("valleys_mapgen:dirt_silty_with_grass")
local c_dirt_sand = minetest.get_content_id("valleys_mapgen:dirt_sandy")
local c_lawn_sand = minetest.get_content_id("valleys_mapgen:dirt_sandy_with_grass")
local c_desert_sand = minetest.get_content_id("default:desert_sand")
2015-04-06 10:43:33 +02:00
local c_sand = minetest.get_content_id("default:sand")
2015-04-04 22:32:37 +02:00
local c_gravel = minetest.get_content_id("default:gravel")
local c_silt = minetest.get_content_id("valleys_mapgen:silt")
2015-04-05 16:21:51 +02:00
local c_clay = minetest.get_content_id("valleys_mapgen:red_clay")
2015-03-07 09:33:01 +01:00
local c_water = minetest.get_content_id("default:water_source")
2015-03-16 09:04:15 +01:00
local c_lava = minetest.get_content_id("default:lava_source")
2015-03-07 09:33:01 +01:00
local c_air = minetest.get_content_id("air")
local vm, emin, emax = minetest.get_mapgen_object("voxelmanip")
local data = vm:get_data()
local a = VoxelArea:new({MinEdge = emin, MaxEdge = emax})
local chulens = vector.add(vector.subtract(maxp, minp), 1)
local chulens_sup = {x = chulens.x, y = chulens.y + 6, z = chulens.z}
local minp2d = pos2d(minp)
2015-03-15 19:37:00 +01:00
local n1 = vmg.noisemap(1, minp2d, chulens)
local n2 = vmg.noisemap(2, minp2d, chulens)
local n3 = vmg.noisemap(3, minp2d, chulens)
local n4 = vmg.noisemap(4, minp2d, chulens)
local n5 = vmg.noisemap(5, minp2d, chulens)
local n6 = vmg.noisemap(6, minp, chulens_sup)
local n7 = vmg.noisemap(7, minp2d, chulens)
local n8 = vmg.noisemap(8, minp, chulens)
local n9 = vmg.noisemap(9, minp, chulens)
local n10 = vmg.noisemap(10, minp, chulens)
local n11 = vmg.noisemap(11, minp, chulens)
2015-03-16 09:04:15 +01:00
local n12 = vmg.noisemap(12, minp, chulens)
2015-04-04 22:32:37 +02:00
local n13 = vmg.noisemap(13, minp2d, chulens)
local n14 = vmg.noisemap(14, minp2d, chulens)
local n15 = vmg.noisemap(15, minp2d, chulens)
2015-04-06 10:43:33 +02:00
local n16 = vmg.noisemap(16, minp2d, chulens)
2015-03-07 09:33:01 +01:00
local i2d = 1 -- index for 2D noises
2015-03-08 10:40:37 +01:00
local i3d_a = 1 -- index for noise 6 which has a special size
local i3d_b = 1 -- index for 3D noises
2015-03-07 09:33:01 +01:00
for x = minp.x, maxp.x do -- for each east-west and bottom-top plane
for z = minp.z, maxp.z do -- for each vertical row in this plane
2015-04-06 10:43:33 +02:00
local v1, v2, v3, v4, v5, v7, v13, v14, v15, v16 = n1[i2d], n2[i2d], n3[i2d], n4[i2d], n5[i2d], n7[i2d], n13[i2d], n14[i2d], n15[i2d], n16[i2d] -- n for noise, v for value
2015-03-07 09:33:01 +01:00
v3 = v3 ^ 2 -- v3 must be > 0 and by the square there are high mountains but the median valleys depth is small.
local base_ground = v1 + v3 -- v3 is here because terrain is generally higher when valleys are deep (mountains)
2015-03-13 20:11:48 +01:00
local river = math.abs(v2) < river_size
2015-03-07 09:33:01 +01:00
local valleys = v3 * (1 - math.exp(- (v2 / v4) ^ 2)) -- use the curve of the function 1exp((x/a)²) to modelise valleys. Making "a" varying 0 < a ≤ 1 will change the shape of the valleys. v2 = x and v4 = a.
local mountain_ground = base_ground + valleys
local slopes = v5 * valleys
2015-04-04 22:32:37 +02:00
2015-03-07 09:33:01 +01:00
if river then
2015-04-06 11:32:47 +02:00
mountain_ground = math.min(math.max(base_ground - 3, water_level - 6), mountain_ground)
2015-03-07 09:33:01 +01:00
slopes = 0
end
2015-04-04 22:32:37 +02:00
local dirt = c_dirt
local lawn = c_lawn
local max = math.max(v13, v14, v15)
2015-04-05 17:05:22 +02:00
if max > dirt_threshold then
2015-04-04 22:32:37 +02:00
if v13 == max then
2015-04-05 17:05:22 +02:00
if v13 > clay_threshold then
2015-04-04 22:32:37 +02:00
dirt = c_clay
lawn = c_clay
else
dirt = c_dirt_clay
lawn = c_lawn_clay
end
elseif v14 == max then
2015-04-05 17:05:22 +02:00
if v14 > silt_threshold then
2015-04-04 22:32:37 +02:00
dirt = c_silt
lawn = c_silt
else
dirt = c_dirt_silt
lawn = c_lawn_silt
end
else
2015-04-05 17:05:22 +02:00
if v15 > sand_threshold then
2015-04-04 22:32:37 +02:00
dirt = c_desert_sand
lawn = c_desert_sand
else
dirt = c_dirt_sand
lawn = c_lawn_sand
end
end
end
2015-04-06 10:43:33 +02:00
local is_beach = v15 > 0 and v16 > 0
2015-04-06 11:32:47 +02:00
local beach = v15 * v16 + water_level
2015-04-04 22:32:37 +02:00
2015-03-07 09:33:01 +01:00
for y = minp.y, maxp.y do -- for each node in vertical row
local ivm = a:index(x, y, z)
2015-03-16 09:04:15 +01:00
local v6, v8, v9, v10, v11, v12 = n6[i3d_a], n8[i3d_b], n9[i3d_b], n10[i3d_b], n11[i3d_b], n12[i3d_b]
2015-03-13 20:11:48 +01:00
local is_cave = v8 ^ 2 + v9 ^ 2 + v10 ^ 2 + v11 ^ 2 < caves_size
2015-03-07 09:33:01 +01:00
if v6 * slopes > y - mountain_ground then -- if pos is in the ground
2015-03-08 10:40:37 +01:00
if not is_cave then
2015-03-11 19:10:35 +01:00
local above = math.ceil(
v7 + math.random() - math.sqrt(math.abs(y)) / dirt_thickness
)
2015-03-08 10:40:37 +01:00
if above <= 0 then
data[ivm] = c_stone
2015-04-06 11:32:47 +02:00
elseif y >= water_level and n6[i3d_a+80] * slopes <= y + 1 - mountain_ground and not river then
2015-04-06 10:43:33 +02:00
if is_beach and y < beach then
data[ivm] = c_sand
else
data[ivm] = lawn -- if node above is not in the ground, place lawn
end
2015-03-08 10:40:37 +01:00
elseif n6[i3d_a+above*80] * slopes <= y + above - mountain_ground then
2015-04-06 10:43:33 +02:00
if is_beach and y < beach then
data[ivm] = c_sand
else
data[ivm] = dirt
end
2015-03-08 10:40:37 +01:00
else
data[ivm] = c_stone
end
2015-04-11 11:23:52 +02:00
elseif v11 + v12 > 2 ^ (y / lava_depth) and y <= lava_max_height then
2015-03-16 09:04:15 +01:00
data[ivm] = c_lava
2015-03-07 09:33:01 +01:00
end
2015-04-06 11:32:47 +02:00
elseif y <= water_level or river and y - 2 <= mountain_ground then
2015-03-07 09:33:01 +01:00
data[ivm] = c_water
end
2015-03-08 10:40:37 +01:00
i3d_a = i3d_a + 80 -- increase i3d_a by one row
i3d_b = i3d_b + 80 -- increase i3d_b by one row
2015-03-07 09:33:01 +01:00
end
i2d = i2d + 80 -- increase i2d by one row
2015-03-08 10:40:37 +01:00
i3d_a = i3d_a + 480 -- avoid the 6 supplemental lines
2015-03-07 09:33:01 +01:00
end
i2d = i2d - 6399 -- i2d = 6401 after the first execution of this loop, it must be 2 before the second.
2015-03-08 10:40:37 +01:00
i3d_a = i3d_a - 550399 -- i3d_a = 550401 after the first execution of this loop, it must be 2 before the second.
i3d_b = i3d_b - 511999 -- i3d_b = 512001 after the first execution of this loop, it must be 2 before the second.
2015-03-07 09:33:01 +01:00
end
-- execute voxelmanip boring stuff to write to the map
vm:set_data(data)
2015-03-13 21:21:32 +01:00
minetest.generate_ores(vm, minp, maxp)
2015-03-08 10:40:37 +01:00
vm:set_lighting({day = 0, night = 0})
2015-03-07 09:33:01 +01:00
vm:calc_lighting()
vm:update_liquids()
2015-03-08 10:59:09 +01:00
vm:write_to_map()
2015-03-07 09:33:01 +01:00
end
dofile(vmg.path .. "/trees.lua")
2015-04-17 21:27:21 +02:00
function vmg.get_humidity_raw(pos)
local v13 = vmg.get_noise(pos, 13)
local v15 = vmg.get_noise(pos, 15)
local v18 = vmg.get_noise(pos, 18)
return 2 ^ (v13 - v15 + v18 * 2)
end
function vmg.get_humidity(pos)
local y = pos.y
local flatpos = pos2d(pos)
local hraw = vmg.get_humidity_raw(flatpos)
local v1 = vmg.get_noise(flatpos, 1)
local v3 = vmg.get_noise(flatpos, 3) ^ 2
local base_ground = v1 + v3
local sea_water = 0.5 ^ math.max((y - water_level) / 6, 0)
local river_water = 0.5 ^ math.max((y - base_ground) / 3, 0)
local water = sea_water + (1 - sea_water) * river_water
return hraw + water
end
function vmg.get_temperature(pos)
local v12 = vmg.get_noise(pos, 12) + 1
local v17 = vmg.get_noise(pos, 17)
local y = pos.y
if y > 0 then
2015-04-17 22:15:00 +02:00
return v17 * 0.5 ^ (y / altitude_chill)
2015-04-17 21:27:21 +02:00
else
2015-04-17 22:15:00 +02:00
return v17 * 0.5 ^ (-y / altitude_chill) + 20 * v12 * (1 - 2 ^ (y / lava_depth))
2015-04-17 21:27:21 +02:00
end
end
2015-03-07 09:33:01 +01:00
function vmg.get_noise(pos, i)
local n = vmg.noises[i]
local noise = minetest.get_perlin(n.seed, n.octaves, n.persist, 1)
if not pos.z then
return noise:get2d({x = pos.x / n.spread.x, y = pos.y / n.spread.y}) * n.scale + n.offset
else
return noise:get3d({x = pos.x / n.spread.x, y = pos.y / n.spread.y, z = pos.z / n.spread.z}) * n.scale + n.offset
end
end
local function round(n)
return math.floor(n + 0.5)
end
function vmg.get_elevation(pos)
local v1 = vmg.get_noise(pos, 1)
local v2 = vmg.get_noise(pos, 2)
local v3 = vmg.get_noise(pos, 3) ^ 2
local v4 = vmg.get_noise(pos, 4)
local v5 = vmg.get_noise(pos, 5)
local base_ground = v1 + v3
local valleys = v3 * (1 - math.exp(- (v2 / v4) ^ 2))
local mountain_ground = base_ground + valleys
local pos = pos3d(pos, round(mountain_ground))
local slopes = v5 * valleys
if vmg.get_noise(pos, 6) * slopes > pos.y - mountain_ground then
pos.y = pos.y + 1
while vmg.get_noise(pos, 6) * slopes > pos.y - mountain_ground do
pos.y = pos.y + 1
end
return pos.y
else
pos.y = pos.y - 1
while vmg.get_noise(pos, 6) * slopes <= pos.y - mountain_ground do
pos.y = pos.y - 1
end
return pos.y
end
end
function vmg.spawnplayer(player)
local angle = math.random() * math.pi * 2
2015-03-14 10:47:15 +01:00
local distance = math.random() * player_max_distance
2015-03-07 09:33:01 +01:00
local p_angle = {x = math.cos(angle), y = math.sin(angle)}
2015-03-14 10:47:15 +01:00
local pos = {x = -p_angle.x * distance, y = -p_angle.y * distance}
2015-03-07 09:33:01 +01:00
local elevation = vmg.get_elevation(pos)
2015-04-06 11:32:47 +02:00
while elevation < water_level + 2 or math.abs(vmg.get_noise(pos, 2)) < river_size do
2015-03-07 09:33:01 +01:00
pos.x = pos.x + p_angle.x
pos.y = pos.y + p_angle.y
elevation = vmg.get_elevation({x = round(pos.x), y = round(pos.y)})
end
pos = {x = round(pos.x), y = round(elevation + 1), z = round(pos.y)}
player:setpos(pos)
end