798 lines
24 KiB
Lua
Raw Permalink Normal View History

2013-10-27 09:53:05 +01:00
mg = {}
2013-12-01 14:27:41 +01:00
local ENABLE_SNOW = false
local DEBUG = false
2013-12-01 14:27:41 +01:00
2013-09-15 17:08:41 +02:00
local DMAX = 20
local AREA_SIZE = 80
dofile(minetest.get_modpath(minetest.get_current_modname()).."/nodes.lua")
local function get_content_id(name)
name = minetest.registered_aliases[name] or name
return minetest.get_content_id(name)
end
c_air = get_content_id("air")
c_ignore = get_content_id("ignore")
c_water = get_content_id("default:water_source")
c_grass = get_content_id("default:dirt_with_grass")
c_dry_grass = get_content_id("mg:dirt_with_dry_grass")
c_dirt_snow = get_content_id("default:dirt_with_snow")
c_snow = get_content_id("default:snow")
c_sapling = get_content_id("default:sapling")
c_tree = get_content_id("default:tree")
c_leaves = get_content_id("default:leaves")
c_junglesapling = get_content_id("default:junglesapling")
c_jungletree = get_content_id("default:jungletree")
c_jungleleaves = get_content_id("default:jungleleaves")
c_savannasapling = get_content_id("mg:savannasapling")
c_savannatree = get_content_id("mg:savannatree")
c_savannaleaves = get_content_id("mg:savannaleaves")
c_pinesapling = get_content_id("mg:pinesapling")
c_pinetree = get_content_id("mg:pinetree")
c_pineleaves = get_content_id("mg:pineleaves")
c_dirt = get_content_id("default:dirt")
c_stone = get_content_id("default:stone")
c_water = get_content_id("default:water_source")
c_ice = get_content_id("default:ice")
c_sand = get_content_id("default:sand")
c_sandstone = get_content_id("default:sandstone")
c_desert_sand = get_content_id("default:desert_sand")
c_desert_stone = get_content_id("default:desert_stone")
c_snowblock = get_content_id("default:snowblock")
c_cactus = get_content_id("default:cactus")
c_grass_1 = get_content_id("default:grass_1")
c_grass_2 = get_content_id("default:grass_2")
c_grass_3 = get_content_id("default:grass_3")
c_grass_4 = get_content_id("default:grass_4")
c_grass_5 = get_content_id("default:grass_5")
c_grasses = {c_grass_1, c_grass_2, c_grass_3, c_grass_4, c_grass_5}
c_jungle_grass = get_content_id("default:junglegrass")
c_dry_shrub = get_content_id("default:dry_shrub")
c_papyrus = get_content_id("default:papyrus")
2013-09-15 17:08:41 +02:00
minetest.register_on_mapgen_init(function(mgparams)
2014-07-05 18:38:56 +02:00
minetest.set_mapgen_params({mgname = "singlenode", flags = "nolight"})
2013-09-15 17:08:41 +02:00
end)
local cache = {}
local function cliff(x, n)
return 0.2*x*x - x + n*x - n*n*x*x - 0.01 * math.abs(x*x*x) + math.abs(x)*100*n*n*n*n
end
local function get_vn(x, z, noise, village)
local vx, vz, vs = village.vx, village.vz, village.vs
return (noise - 2) * 20 +
(40 / (vs * vs)) * ((x - vx) * (x - vx) + (z - vz) * (z - vz))
end
local function get_base_surface_at_point(x, z, vnoise, villages, ni, noise1, noise2, noise3, noise4)
2013-09-20 20:36:53 +02:00
local index = 65536*x+z
if cache[index] ~= nil then return cache[index] end
cache[index] = 25*noise1[ni]+noise2[ni]*noise3[ni]/3
if noise4[ni] > 0.8 then
cache[index] = cliff(cache[index], noise4[ni]-0.8)
2013-09-20 20:36:53 +02:00
end
local s = 0
local t = 0
local noise = vnoise[ni]
for _, village in ipairs(villages) do
local vn = get_vn(x, z, noise, village)
if vn < 40 then
cache[index] = village.vh
return village.vh
elseif vn < 200 then
s = s + ((cache[index] * (vn - 40) + village.vh * (200 - vn)) / 160) / (vn - 40)
t = t + 1 / (vn - 40)
end
end
if t > 0 then
cache[index] = s / t
2013-09-20 20:36:53 +02:00
end
return cache[index]
end
local function surface_at_point(x, z, ...)
return get_base_surface_at_point(x, z, unpack({...}))
end
local SMOOTHED = AREA_SIZE+2*DMAX
local HSMOOTHED = AREA_SIZE+DMAX
local INSIDE = AREA_SIZE-DMAX
local function smooth(x, z, ...)
local s=0
local w=0
for xi=-DMAX, DMAX do
for zi=-DMAX, DMAX do
local d2=xi*xi+zi*zi
if d2<DMAX*DMAX then
local w1 = 1-d2/(DMAX*DMAX)
local w2 = 15/16*w1*w1
w = w+w2
s=s+w2*surface_at_point(x+xi, z+zi, unpack({...}))
end
end
end
return s/w
end
function inside_village(x, z, village, vnoise)
return get_vn(x, z, vnoise:get_2d({x = x, y = z}), village) <= 40
2013-09-22 15:38:07 +02:00
end
local wseed
2013-09-20 20:36:53 +02:00
minetest.register_on_mapgen_init(function(mgparams)
wseed = math.floor(mgparams.seed/10000000000)
end)
2013-09-21 15:07:43 +02:00
function get_bseed(minp)
2013-09-20 20:36:53 +02:00
return wseed + math.floor(5*minp.x/47) + math.floor(873*minp.z/91)
end
2013-09-21 15:07:43 +02:00
function get_bseed2(minp)
2013-09-21 19:58:04 +02:00
return wseed + math.floor(87*minp.x/47) + math.floor(73*minp.z/91) + math.floor(31*minp.y/12)
2013-09-20 20:36:53 +02:00
end
2013-09-22 09:28:18 +02:00
local function add_leaves(data, vi, c_leaves, c_snow)
if data[vi]==c_air or data[vi]==c_ignore or data[vi] == c_snow then
2013-09-20 20:36:53 +02:00
data[vi] = c_leaves
end
end
function add_tree(data, a, x, y, z, minp, maxp, pr)
local th = pr:next(3, 4)
2013-09-20 20:36:53 +02:00
for yy=math.max(minp.y, y), math.min(maxp.y, y+th) do
local vi = a:index(x, yy, z)
data[vi] = c_tree
end
local maxy = y+th
2013-09-20 20:36:53 +02:00
for xx=math.max(minp.x, x-1), math.min(maxp.x, x+1) do
for yy=math.max(minp.y, maxy-1), math.min(maxp.y, maxy+1) do
for zz=math.max(minp.z, z-1), math.min(maxp.z, z+1) do
add_leaves(data, a:index(xx, yy, zz), c_leaves)
end
end
end
for i=1,8 do
local xi = pr:next(x-2, x+1)
local yi = pr:next(maxy-1, maxy+1)
local zi = pr:next(z-2, z+1)
2013-09-20 20:36:53 +02:00
for xx=math.max(minp.x, xi), math.min(maxp.x, xi+1) do
for yy=math.max(minp.y, yi), math.min(maxp.y, yi+1) do
for zz=math.max(minp.z, zi), math.min(maxp.z, zi+1) do
add_leaves(data, a:index(xx, yy, zz), c_leaves)
end
end
end
end
end
function add_jungletree(data, a, x, y, z, minp, maxp, pr)
local th = pr:next(7, 11)
2013-09-20 20:36:53 +02:00
for yy=math.max(minp.y, y), math.min(maxp.y, y+th) do
local vi = a:index(x, yy, z)
data[vi] = c_jungletree
2013-09-20 20:36:53 +02:00
end
local maxy = y+th
2013-09-20 20:36:53 +02:00
for xx=math.max(minp.x, x-1), math.min(maxp.x, x+1) do
for yy=math.max(minp.y, maxy-1), math.min(maxp.y, maxy+1) do
for zz=math.max(minp.z, z-1), math.min(maxp.z, z+1) do
add_leaves(data, a:index(xx, yy, zz), c_jungleleaves)
2013-09-20 20:36:53 +02:00
end
end
end
for i=1,30 do
local xi = pr:next(x-3, x+2)
local yi = pr:next(maxy-2, maxy+1)
local zi = pr:next(z-3, z+2)
2013-09-20 20:36:53 +02:00
for xx=math.max(minp.x, xi), math.min(maxp.x, xi+1) do
for yy=math.max(minp.y, yi), math.min(maxp.y, yi+1) do
for zz=math.max(minp.z, zi), math.min(maxp.z, zi+1) do
add_leaves(data, a:index(xx, yy, zz), c_jungleleaves)
2013-09-20 20:36:53 +02:00
end
end
end
end
end
function add_savannatree(data, a, x, y, z, minp, maxp, pr)
local th = pr:next(7, 11)
2013-09-21 16:10:55 +02:00
for yy=math.max(minp.y, y), math.min(maxp.y, y+th) do
local vi = a:index(x, yy, z)
data[vi] = c_savannatree
2013-09-21 16:10:55 +02:00
end
local maxy = y+th
2013-09-21 16:10:55 +02:00
for xx=math.max(minp.x, x-1), math.min(maxp.x, x+1) do
for yy=math.max(minp.y, maxy-1), math.min(maxp.y, maxy+1) do
for zz=math.max(minp.z, z-1), math.min(maxp.z, z+1) do
add_leaves(data, a:index(xx, yy, zz), c_savannaleaves)
2013-09-21 16:10:55 +02:00
end
end
end
for i=1,20 do
local xi = pr:next(x-3, x+2)
local yi = pr:next(maxy-2, maxy)
local zi = pr:next(z-3, z+2)
2013-09-21 16:10:55 +02:00
for xx=math.max(minp.x, xi), math.min(maxp.x, xi+1) do
for yy=math.max(minp.y, yi), math.min(maxp.y, yi+1) do
for zz=math.max(minp.z, zi), math.min(maxp.z, zi+1) do
add_leaves(data, a:index(xx, yy, zz), c_savannaleaves)
2013-09-21 16:10:55 +02:00
end
end
end
end
for i=1,15 do
local xi = pr:next(x-3, x+2)
local yy = pr:next(maxy-6, maxy-5)
local zi = pr:next(z-3, z+2)
2013-09-21 16:10:55 +02:00
for xx=math.max(minp.x, xi), math.min(maxp.x, xi+1) do
for zz=math.max(minp.z, zi), math.min(maxp.z, zi+1) do
2013-09-21 16:43:02 +02:00
if minp.y<=yy and maxp.y>=yy then
add_leaves(data, a:index(xx, yy, zz), c_savannaleaves)
2013-09-21 16:43:02 +02:00
end
2013-09-21 16:10:55 +02:00
end
end
end
end
function add_savannabush(data, a, x, y, z, minp, maxp, pr)
local bh = pr:next(1, 2)
local bw = pr:next(2, 4)
2013-09-21 18:10:26 +02:00
for xx=math.max(minp.x, x-bw), math.min(maxp.x, x+bw) do
for zz=math.max(minp.z, z-bw), math.min(maxp.z, z+bw) do
for yy=math.max(minp.y, y-bh), math.min(maxp.y, y+bh) do
if pr:next(1, 100) < 95 and math.abs(xx-x) < pr:next(bh, bh+2)-math.abs(y-yy) and math.abs(zz-z) < pr:next(bh, bh+2)-math.abs(y-yy) then
add_leaves(data, a:index(xx, yy, zz), c_savannaleaves)
2013-09-21 18:10:26 +02:00
for yyy=math.max(minp.y, yy-2), yy do
add_leaves(data, a:index(xx, yyy, zz), c_savannaleaves)
2013-09-21 18:10:26 +02:00
end
end
end
end
end
2013-09-28 19:39:35 +02:00
if x<=maxp.x and x>=minp.x and y<=maxp.y and y>=minp.y and z<=maxp.z and z>=minp.z then
local vi = a:index(x, y, z)
data[vi] = c_savannatree
2013-09-28 19:39:35 +02:00
end
2013-09-21 18:10:26 +02:00
end
2013-10-30 17:25:47 +01:00
function add_pinetree(data, a, x, y, z, minp, maxp, pr, snow)
if snow == nil then snow = c_snow end
local th = pr:next(9, 13)
2013-09-22 09:21:42 +02:00
for yy=math.max(minp.y, y), math.min(maxp.y, y+th) do
local vi = a:index(x, yy, z)
data[vi] = c_pinetree
2013-09-22 09:21:42 +02:00
end
local maxy = y+th
2013-09-22 09:21:42 +02:00
for xx=math.max(minp.x, x-3), math.min(maxp.x, x+3) do
for yy=math.max(minp.y, maxy-1), math.min(maxp.y, maxy-1) do
for zz=math.max(minp.z, z-3), math.min(maxp.z, z+3) do
if pr:next(1, 100) < 80 then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
for xx=math.max(minp.x, x-2), math.min(maxp.x, x+2) do
for yy=math.max(minp.y, maxy), math.min(maxp.y, maxy) do
for zz=math.max(minp.z, z-2), math.min(maxp.z, z+2) do
if pr:next(1, 100) < 85 then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
for xx=math.max(minp.x, x-1), math.min(maxp.x, x+1) do
for yy=math.max(minp.y, maxy+1), math.min(maxp.y, maxy+1) do
for zz=math.max(minp.z, z-1), math.min(maxp.z, z+1) do
if pr:next(1, 100) < 90 then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
2013-09-22 09:31:59 +02:00
if maxy+1<=maxp.y and maxy+1>=minp.y then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(x, maxy+1, z), c_pineleaves, snow)
add_leaves(data, a:index(x, maxy+2, z), snow)
2013-09-22 09:31:59 +02:00
end
local my = 0
2013-09-22 09:21:42 +02:00
for i=1,20 do
local xi = pr:next(x-3, x+2)
local yy = pr:next(maxy-6, maxy-5)
local zi = pr:next(z-3, z+2)
2013-09-22 09:21:42 +02:00
if yy > my then
my = yy
end
for xx=math.max(minp.x, xi), math.min(maxp.x, xi+1) do
for zz=math.max(minp.z, zi), math.min(maxp.z, zi+1) do
if minp.y<=yy and maxp.y>=yy then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
for xx=math.max(minp.x, x-2), math.min(maxp.x, x+2) do
for yy=math.max(minp.y, my+1), math.min(maxp.y, my+1) do
for zz=math.max(minp.z, z-2), math.min(maxp.z, z+2) do
if pr:next(1, 100) < 85 then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
for xx=math.max(minp.x, x-1), math.min(maxp.x, x+1) do
for yy=math.max(minp.y, my+2), math.min(maxp.y, my+2) do
for zz=math.max(minp.z, z-1), math.min(maxp.z, z+1) do
if pr:next(1, 100) < 90 then
2013-10-30 17:25:47 +01:00
add_leaves(data, a:index(xx, yy, zz), c_pineleaves, snow)
add_leaves(data, a:index(xx, yy+1, zz), snow)
2013-09-22 09:21:42 +02:00
end
end
end
end
end
2013-09-22 18:14:27 +02:00
dofile(minetest.get_modpath(minetest.get_current_modname()).."/we.lua")
2013-09-22 11:11:38 +02:00
dofile(minetest.get_modpath(minetest.get_current_modname()).."/rotate.lua")
2013-09-21 15:07:43 +02:00
dofile(minetest.get_modpath(minetest.get_current_modname()).."/buildings.lua")
dofile(minetest.get_modpath(minetest.get_current_modname()).."/villages.lua")
2013-09-21 18:54:02 +02:00
dofile(minetest.get_modpath(minetest.get_current_modname()).."/ores.lua")
2013-09-21 15:07:43 +02:00
2013-11-01 14:52:40 +01:00
function get_biome_table(minp, humidity, temperature, range)
if range == nil then range = 1 end
local l = {}
2013-11-01 14:52:40 +01:00
for xi = -range, range do
for zi = -range, range do
local mnp, mxp = {x=minp.x+xi*80,z=minp.z+zi*80}, {x=minp.x+xi*80+80,z=minp.z+zi*80+80}
local pr = PseudoRandom(get_bseed(mnp))
local bxp, bzp = pr:next(mnp.x, mxp.x), pr:next(mnp.z, mxp.z)
local h, t = humidity:get_2d({x=bxp, y=bzp}), temperature:get_2d({x=bxp, y=bzp})
2013-09-21 17:21:34 +02:00
l[#l+1] = {x=bxp, z=bzp, h=h, t=t}
end
end
return l
end
local function get_distance(x1, x2, z1, z2)
return (x1-x2)*(x1-x2)+(z1-z2)*(z1-z2)
end
2013-11-01 14:52:40 +01:00
function get_nearest_biome(biome_table, x, z)
local m = math.huge
local k = 0
2013-09-21 17:21:34 +02:00
for key, bdef in ipairs(biome_table) do
local dist = get_distance(bdef.x, x, bdef.z, z)
if dist<m then
m=dist
k=key
end
end
return biome_table[k]
end
local function get_perlin_map(seed, octaves, persistance, scale, minp, maxp)
local sidelen = maxp.x - minp.x +1
local pm = minetest.get_perlin_map(
{offset=0, scale=1, spread={x=scale, y=scale, z=scale}, seed=seed, octaves=octaves, persist=persistance},
{x=sidelen, y=sidelen, z=sidelen}
)
return pm:get_2d_map_flat({x = minp.x, y = minp.z, z = 0})
end
2013-12-01 13:43:29 +01:00
local function mg_generate(minp, maxp, emin, emax, vm)
2013-09-15 17:08:41 +02:00
local a = VoxelArea:new{
MinEdge={x=emin.x, y=emin.y, z=emin.z},
MaxEdge={x=emax.x, y=emax.y, z=emax.z},
}
2013-09-21 15:07:43 +02:00
local treemin = {x=emin.x, y=minp.y, z=emin.z}
local treemax = {x=emax.x, y=maxp.y, z=emax.z}
local sidelen = maxp.x-minp.x+1
local noise1 = get_perlin_map(12345, 6, 0.5, 256, minp, maxp)
local noise2 = get_perlin_map(56789, 6, 0.5, 256, minp, maxp)
local noise3 = get_perlin_map(42, 3, 0.5, 32, minp, maxp)
local noise4 = get_perlin_map(8954, 8, 0.5, 1024, minp, maxp)
local noise1raw = minetest.get_perlin(12345, 6, 0.5, 256)
2013-12-09 15:17:43 +01:00
local vcr = VILLAGE_CHECK_RADIUS
local villages = {}
2013-12-09 15:17:43 +01:00
for xi = -vcr, vcr do
for zi = -vcr, vcr do
2014-07-09 09:12:34 +02:00
for _, village in ipairs(villages_at_point({x = minp.x + xi * 80, z = minp.z + zi * 80}, noise1raw)) do
village.to_grow = {}
villages[#villages+1] = village
2013-09-24 20:02:54 +02:00
end
2013-09-20 20:36:53 +02:00
end
end
2013-09-15 17:08:41 +02:00
2013-09-21 17:21:34 +02:00
2013-09-20 20:36:53 +02:00
local pr = PseudoRandom(get_bseed(minp))
2013-09-28 16:53:25 +02:00
local village_noise = minetest.get_perlin(7635, 3, 0.5, 16)
local village_noise_map = get_perlin_map(7635, 3, 0.5, 16, minp, maxp)
2013-09-15 17:08:41 +02:00
local noise_top_layer = get_perlin_map(654, 6, 0.5, 256, minp, maxp)
local noise_second_layer = get_perlin_map(123, 6, 0.5, 256, minp, maxp)
2013-09-15 17:08:41 +02:00
local noise_temperature_raw = minetest.get_perlin(763, 7, 0.5, 512)
local noise_humidity_raw = minetest.get_perlin(834, 7, 0.5, 512)
local noise_temperature = get_perlin_map(763, 7, 0.5, 512, minp, maxp)
local noise_humidity = get_perlin_map(834, 7, 0.5, 512, minp, maxp)
local noise_beach = get_perlin_map(452, 6, 0.5, 256, minp, maxp)
2013-09-15 17:08:41 +02:00
local biome_table = get_biome_table(minp, noise_humidity_raw, noise_temperature_raw)
2013-09-21 17:21:34 +02:00
2013-09-15 17:08:41 +02:00
local data = vm:get_data()
local param2_data = vm:get_param2_data()
2013-09-15 17:08:41 +02:00
2013-09-22 15:38:07 +02:00
local villages_to_grow = {}
local ni = 0
2013-09-15 17:08:41 +02:00
for z = minp.z, maxp.z do
for x = minp.x, maxp.x do
ni = ni + 1
local y = math.floor(surface_at_point(x, z, village_noise_map, villages, ni, noise1, noise2, noise3, noise4))
local humidity = noise_humidity[ni]
local temperature = noise_temperature[ni] - math.max(y, 0) / 50
local biome = get_nearest_biome(biome_table, x, z)
local biome_humidity = biome.h
local biome_temperature = biome.t
local liquid_top
if biome_temperature < -0.4 then
2013-09-20 20:36:53 +02:00
liquid_top = c_ice
else
liquid_top = c_water
end
local above_top, top, top_layer, second_layer
2013-09-15 17:08:41 +02:00
if y < -1 then
above_top = c_air
top = c_dirt
top_layer = c_dirt
second_layer = c_stone
elseif y < 3 and noise_beach[ni] < 0.2 then
above_top = c_air
2013-09-15 17:08:41 +02:00
top = c_sand
top_layer = c_sand
second_layer = c_sandstone
else
above_top = c_air
if biome_temperature > 0.4 then
if biome_humidity < -0.4 then
2013-09-15 17:08:41 +02:00
top = c_desert_sand
top_layer = c_desert_sand
second_layer = c_desert_stone
elseif biome_humidity < 0.4 then
2013-09-21 16:10:55 +02:00
top = c_dry_grass
2013-09-15 17:08:41 +02:00
top_layer = c_dirt
second_layer = c_stone
else
top = c_grass
top_layer = c_dirt
second_layer = c_stone
end
elseif biome_temperature < -0.4 then
2013-09-15 17:08:41 +02:00
above_top = c_snow
top = c_dirt_snow
top_layer = c_dirt
second_layer = c_stone
else
top = c_grass
top_layer = c_dirt
second_layer = c_stone
end
end
if y >= 100 then
above_top = c_air
top = c_snow
top_layer = c_snowblock
end
if y < 0 then
2013-09-21 16:43:02 +02:00
above_top = c_air
end
if y <= maxp.y and y >= minp.y then
2013-09-28 17:40:43 +02:00
local vi = a:index(x, y, z)
if y >= 0 then
data[vi] = top
else
data[vi] = top_layer
end
2013-09-15 17:08:41 +02:00
end
local add_above_top = true
for id, tree in ipairs(mg.registered_trees) do
if tree.min_humidity <= humidity and humidity <= tree.max_humidity
and tree.min_temperature <= temperature and temperature <= tree.max_temperature
and tree.min_biome_humidity <= biome_humidity and biome_humidity <= tree.max_biome_humidity
and tree.min_biome_temperature <= biome_temperature and biome_temperature <= tree.max_biome_temperature
and tree.min_height <= y + 1 and y + 1 <= tree.max_height
and ((not tree.grows_on) or tree.grows_on == top)
and pr:next(1, tree.chance) == 1 then
local in_village = false
for _, village in ipairs(villages) do
if inside_village(x, z, village, village_noise) and not tree.can_be_in_village then
village.to_grow[#village.to_grow+1] = {x = x, y = y + 1, z = z, id = id}
in_village = true
break
end
end
if not in_village then
tree.grow(data, a, x, y + 1, z, minp, maxp, pr)
end
add_above_top = false
break
2013-09-20 20:36:53 +02:00
end
2013-09-15 17:08:41 +02:00
end
if add_above_top and y + 1 <= maxp.y and y + 1 >= minp.y then
local vi = a:index(x, y + 1, z)
data[vi] = above_top
end
if y < 0 and minp.y <= 0 and maxp.y > y then
for yy = math.max(y + 1, minp.y), math.min(0, maxp.y) do
2013-09-15 17:08:41 +02:00
local vi = a:index(x, yy, z)
data[vi] = c_water
end
if maxp.y >= 0 then
2013-09-15 17:08:41 +02:00
data[a:index(x, 0, z)] = liquid_top
end
end
local tl = math.floor((noise_top_layer[ni] + 2.5) * 2)
if y - tl - 1 <= maxp.y and y - 1 >= minp.y then
for yy = math.max(y - tl - 1, minp.y), math.min(y - 1, maxp.y) do
2013-09-15 17:08:41 +02:00
local vi = a:index(x, yy, z)
data[vi] = top_layer
end
end
local sl = math.floor((noise_second_layer[ni] + 5) * 3)
if y - sl - 1 <= maxp.y and y - tl - 2 >= minp.y then
for yy = math.max(y - sl - 1, minp.y), math.min(y - tl - 2, maxp.y) do
2013-09-15 17:08:41 +02:00
local vi = a:index(x, yy, z)
data[vi] = second_layer
end
end
if y - sl - 2 >= minp.y then
for yy = minp.y, math.min(y - sl - 2, maxp.y) do
2013-09-15 17:08:41 +02:00
local vi = a:index(x, yy, z)
data[vi] = c_stone
end
end
end
end
2013-09-21 18:54:02 +02:00
2013-09-21 19:58:04 +02:00
local va = VoxelArea:new{MinEdge=minp, MaxEdge=maxp}
2013-09-21 18:54:02 +02:00
for _, ore_sheet in ipairs(mg.registered_ore_sheets) do
local sidelen = maxp.x - minp.x + 1
local np = table.copy(ore_sheet.noise_params)
np.seed = np.seed + minp.y
local pm = minetest.get_perlin_map(np, {x = sidelen, y = sidelen, z = 1})
local map = pm:get_2d_map_flat({x = minp.x, y = minp.z})
local ni = 0
local trh = ore_sheet.threshhold
2013-10-28 08:16:38 +01:00
local wherein = minetest.get_content_id(ore_sheet.wherein)
local ore = minetest.get_content_id(ore_sheet.name)
local hmin = ore_sheet.height_min
local hmax = ore_sheet.height_max
local tmin = ore_sheet.tmin
local tmax = ore_sheet.tmax
for z = minp.z, maxp.z do
for x = minp.x, maxp.x do
ni = ni + 1
local noise = map[ni]
if noise > trh then
local thickness = pr:next(tmin, tmax)
local y0 = math.floor(minp.y + (noise - trh) * 4)
for y = math.max(y0, hmin), math.min(y0 + thickness - 1, hmax) do
local vi = a:index(x, y, z)
2013-10-28 07:28:48 +01:00
if data[vi] == wherein or wherein == c_ignore then
data[vi] = ore
end
end
end
end
end
end
2013-10-27 10:18:38 +01:00
for _, ore in ipairs(mg.registered_ores) do
2013-10-28 08:16:38 +01:00
generate_vein(minetest.get_content_id(ore.name), minetest.get_content_id(ore.wherein), minp, maxp, ore.seeddiff, ore, data, a, va)
2013-10-27 09:53:05 +01:00
end
for _, village in ipairs(villages) do
village.to_add = generate_village(village, minp, maxp, data, param2_data, a, village_noise)
end
2013-09-21 18:49:24 +02:00
2013-09-15 17:08:41 +02:00
vm:set_data(data)
vm:set_param2_data(param2_data)
2013-09-21 18:49:24 +02:00
2013-09-15 17:08:41 +02:00
vm:calc_lighting(
{x = minp.x - 16, y = minp.y, z = minp.z - 16},
{x = maxp.x + 16, y = maxp.y, z = maxp.z + 16}
2013-09-15 17:08:41 +02:00
)
2013-09-21 18:49:24 +02:00
vm:write_to_map()
2013-09-21 18:49:24 +02:00
local meta
for _, village in ipairs(villages) do
for _, n in pairs(village.to_add) do
2013-09-21 19:16:04 +02:00
minetest.set_node(n.pos, n.node)
if n.meta ~= nil then
2013-09-22 18:24:50 +02:00
meta = minetest.get_meta(n.pos)
2013-09-21 19:16:04 +02:00
meta:from_table(n.meta)
2013-09-28 17:40:43 +02:00
if n.node.name == "default:chest" then
local inv = meta:get_inventory()
local items = inv:get_list("main")
for i = 1, inv:get_size("main") do
2013-09-28 17:56:13 +02:00
inv:set_stack("main", i, ItemStack(""))
end
local numitems = pr:next(3, 20)
for i = 1, numitems do
local ii = pr:next(1, #items)
local prob = items[ii]:get_count() % 2 ^ 8
local stacksz = math.floor(items[ii]:get_count() / 2 ^ 8)
2013-09-29 09:26:38 +02:00
if pr:next(0, prob) == 0 and stacksz>0 then
local stk = ItemStack({
name = items[ii]:get_name(),
count = pr:next(1, stacksz),
wear = items[ii]:get_wear(),
metadata = items[ii]:get_metadata()
})
2013-09-28 19:03:49 +02:00
local ind = pr:next(1, inv:get_size("main"))
while not inv:get_stack("main", ind):is_empty() do
2013-09-28 19:03:49 +02:00
ind = pr:next(1, inv:get_size("main"))
end
inv:set_stack("main", ind, stk)
2013-09-28 17:40:43 +02:00
end
end
end
2013-09-21 19:16:04 +02:00
end
end
end
2013-12-01 13:43:29 +01:00
end
minetest.register_on_generated(function(minp, maxp, seed)
local vm, emin, emax = minetest.get_mapgen_object("voxelmanip")
mg_generate(minp, maxp, emin, emax, vm)
2013-09-15 17:08:41 +02:00
end)
2013-10-27 09:53:05 +01:00
2013-12-01 13:43:29 +01:00
local function mg_regenerate(pos, name)
local minp = {x = 80 * math.floor((pos.x + 32) / 80) - 32,
y = 80 * math.floor((pos.y + 32) / 80) - 32,
z = 80 * math.floor((pos.z + 32) / 80) - 32}
local maxp = {x = minp.x + 79, y = minp.y + 79, z = minp.z + 79}
2013-12-01 13:43:29 +01:00
local vm = minetest.get_voxel_manip()
local emin, emax = vm:read_from_map(minp, maxp)
local data = {}
for i = 1, (maxp.x - minp.x + 1) * (maxp.y - minp.y + 1) * (maxp.z - minp.z + 1) do
2013-12-01 13:43:29 +01:00
data[i] = c_air
end
vm:set_data(data)
vm:write_to_map()
mg_generate(minp, maxp, emin, emax, vm)
minetest.chat_send_player(name, "Regenerating done, fixing lighting. This may take a while...")
-- Fix lighting
local nodes = minetest.find_nodes_in_area(minp, maxp, "air")
local nnodes = #nodes
local p = math.floor(nnodes / 5)
2013-12-01 13:43:29 +01:00
local dig_node = minetest.dig_node
for i, pos in ipairs(nodes) do
2013-12-01 13:43:29 +01:00
dig_node(pos)
if i % p == 0 then
minetest.chat_send_player(name, math.floor(i / nnodes * 100).."%")
2013-12-01 13:43:29 +01:00
end
end
minetest.chat_send_player(name, "Done")
return minp, maxp
2013-12-01 13:43:29 +01:00
end
minetest.register_chatcommand("mg_regenerate", {
privs = {server = true},
func = function(name, param)
local player = minetest.get_player_by_name(name)
if player then
local pos = player:getpos()
local minp, maxp = mg_regenerate(pos, name)
if minetest.get_modpath("biome_lib") and minp and maxp then
biome_lib.blocklist_aircheck[#biome_lib.blocklist_aircheck + 1] = {minp, maxp}
biome_lib.blocklist_no_aircheck[#biome_lib.blocklist_no_aircheck + 1] = {minp, maxp}
end
2013-12-01 13:43:29 +01:00
end
end,
})
2014-07-05 11:08:23 +02:00
local function spawnplayer(player)
2014-10-24 19:28:23 +02:00
if minetest.setting_get("static_spawnpoint") then return end
2014-07-05 11:08:23 +02:00
local noise1 = minetest.get_perlin(12345, 6, 0.5, 256)
local min_dist = math.huge
local min_pos = {x = 0, y = 3, z = 0}
for bx = -20, 20 do
for bz = -20, 20 do
local minp = {x = -32 + 80 * bx, y = -32, z = -32 + 80 * bz}
2014-07-09 13:25:19 +02:00
for _, village in ipairs(villages_at_point(minp, noise1)) do
if math.abs(village.vx) + math.abs(village.vz) < min_dist then
min_pos = {x = village.vx, y = village.vh + 2, z = village.vz}
min_dist = math.abs(village.vx) + math.abs(village.vz)
2014-07-05 11:08:23 +02:00
end
end
end
end
player:setpos(min_pos)
end
minetest.register_on_newplayer(function(player)
spawnplayer(player)
end)
minetest.register_on_respawnplayer(function(player)
spawnplayer(player)
return true
end)
2013-10-27 09:53:05 +01:00
mg.registered_ores = {}
function mg.register_ore(oredef)
if oredef.wherein == nil then
oredef.wherein = "ignore"
end
if DEBUG then
oredef.wherein = "ignore"
oredef.maxheight = 31000
end
2013-10-27 09:53:05 +01:00
mg.registered_ores[#mg.registered_ores+1] = oredef
end
mg.registered_ore_sheets = {}
function mg.register_ore_sheet(oredef)
if oredef.wherein == nil then
oredef.wherein = "ignore"
end
2013-10-28 07:28:48 +01:00
if DEBUG then
oredef.wherein = "ignore"
oredef.height_max = 31000
end
mg.registered_ore_sheets[#mg.registered_ore_sheets + 1] = oredef
end
mg.registered_trees = {}
function mg.register_tree(treedef)
if treedef.min_humidity == nil then
treedef.min_humidity = -2
end
if treedef.max_humidity == nil then
treedef.max_humidity = 2
end
if treedef.min_biome_humidity == nil then
treedef.min_biome_humidity = -2
end
if treedef.max_biome_humidity == nil then
treedef.max_biome_humidity = 2
end
if treedef.min_temperature == nil then
treedef.min_temperature = -2
end
if treedef.max_temperature == nil then
treedef.max_temperature = 2
end
if treedef.min_biome_temperature == nil then
treedef.min_biome_temperature = -2
end
if treedef.max_biome_temperature == nil then
treedef.max_biome_temperature = 2
end
mg.registered_trees[#mg.registered_trees + 1] = treedef
end
dofile(minetest.get_modpath(minetest.get_current_modname()).."/oredesc.lua")
dofile(minetest.get_modpath(minetest.get_current_modname()).."/trees.lua")
2013-11-01 14:52:40 +01:00
2013-12-01 14:27:41 +01:00
if ENABLE_SNOW then
dofile(minetest.get_modpath(minetest.get_current_modname()).."/snow.lua")
end