2014-08-05 01:24:11 +02:00
2014-10-10 01:51:12 +02:00
2014-08-05 01:24:11 +02:00
mg_villages.wseed = 0 ;
2014-08-05 17:34:50 +02:00
minetest.register_on_mapgen_init ( function ( mgparams )
mg_villages.wseed = math.floor ( mgparams.seed / 10000000000 )
end )
2014-08-05 01:24:11 +02:00
function mg_villages . get_bseed ( minp )
return mg_villages.wseed + math.floor ( 5 * minp.x / 47 ) + math.floor ( 873 * minp.z / 91 )
end
function mg_villages . get_bseed2 ( minp )
return mg_villages.wseed + math.floor ( 87 * minp.x / 47 ) + math.floor ( 73 * minp.z / 91 ) + math.floor ( 31 * minp.y / 12 )
end
mg_villages.inside_village = function ( x , z , village , vnoise )
return mg_villages.get_vn ( x , z , vnoise : get2d ( { x = x , y = z } ) , village ) <= 40
end
2014-08-11 00:42:48 +02:00
mg_villages.inside_village_area = function ( x , z , village , vnoise )
return mg_villages.get_vn ( x , z , vnoise : get2d ( { x = x , y = z } ) , village ) <= 80
end
2014-08-25 01:46:33 +02:00
mg_villages.get_vnoise = function ( x , z , village , vnoise ) -- PM v
return mg_villages.get_vn ( x , z , vnoise : get2d ( { x = x , y = z } ) , village )
end -- PM ^
2014-08-05 01:24:11 +02:00
mg_villages.get_vn = function ( 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
2014-10-23 16:49:37 +02:00
mg_villages.villages_in_mapchunk = function ( minp , mapchunk_size )
2014-08-05 01:24:11 +02:00
local noise1raw = minetest.get_perlin ( 12345 , 6 , 0.5 , 256 )
2014-08-19 18:32:42 +02:00
local vcr = mg_villages.VILLAGE_CHECK_RADIUS
2014-08-05 01:24:11 +02:00
local villages = { }
for xi = - vcr , vcr do
for zi = - vcr , vcr do
2014-10-23 16:49:37 +02:00
for _ , village in ipairs ( mg_villages.villages_at_point ( { x = minp.x + xi * mapchunk_size , z = minp.z + zi * mapchunk_size } , noise1raw ) ) do
2014-08-05 01:24:11 +02:00
villages [ # villages + 1 ] = village
end
end
end
return villages ;
end
2014-08-06 03:00:43 +02:00
mg_villages.node_is_ground = { } ; -- store nodes which have previously been identified as ground
mg_villages.check_if_ground = function ( ci )
2014-08-26 20:41:29 +02:00
-- pre-generate a list of no-ground-nodes for caching
if ( # mg_villages.node_is_ground < 1 ) then
2014-08-28 04:35:42 +02:00
local no_ground_nodes = { ' air ' , ' ignore ' , ' default:sandstonebrick ' , ' default:cactus ' , ' default:wood ' , ' default:junglewood ' ,
' ethereal:mushroom_pore ' , ' ethereal:mushroom_trunk ' , ' ethereal:bamboo ' } ;
2014-08-26 20:41:29 +02:00
for _ , name in ipairs ( no_ground_nodes ) do
mg_villages.node_is_ground [ minetest.get_content_id ( name ) ] = false ;
end
end
2014-08-06 03:00:43 +02:00
if ( not ( ci ) ) then
return false ;
end
2014-08-28 04:35:42 +02:00
if ( mg_villages.node_is_ground [ ci ] ~= nil ) then
return mg_villages.node_is_ground [ ci ] ;
2014-08-06 03:00:43 +02:00
end
-- analyze the node
-- only nodes on which walking is possible may be counted as ground
local node_name = minetest.get_name_from_content_id ( ci ) ;
local def = minetest.registered_nodes [ node_name ] ;
2014-08-28 04:35:42 +02:00
-- store information about this node type for later use
if ( not ( def ) ) then
mg_villages.node_is_ground [ ci ] = false ;
2014-11-20 21:16:48 +01:00
elseif ( not ( def.walkable ) ) then
mg_villages.node_is_ground [ ci ] = false ;
elseif ( def.groups and def.groups . tree ) then
2014-08-28 04:35:42 +02:00
mg_villages.node_is_ground [ ci ] = false ;
elseif ( def.drop and def.drop == ' default:dirt ' ) then
2014-08-26 20:41:29 +02:00
mg_villages.node_is_ground [ ci ] = true ;
2014-10-06 18:46:56 +02:00
elseif ( def.walkable == true and def.is_ground_content == true and not ( def.node_box ) ) then
2014-08-28 04:35:42 +02:00
mg_villages.node_is_ground [ ci ] = true ;
else
mg_villages.node_is_ground [ ci ] = false ;
2014-08-06 03:00:43 +02:00
end
2014-08-28 04:35:42 +02:00
return mg_villages.node_is_ground [ ci ] ;
2014-08-06 03:00:43 +02:00
end
2014-08-25 17:34:16 +02:00
2014-08-27 00:01:48 +02:00
-- sets evrything at x,z and above height target_height to air;
-- the area below gets filled up in a suitable way (i.e. dirt with grss - dirt - stone)
2014-10-11 23:25:50 +02:00
mg_villages.lower_or_raise_terrain_at_point = function ( x , z , target_height , minp , maxp , vm , data , param2_data , a , cid , vh , treepos , has_artificial_snow , blend )
2014-08-27 00:01:48 +02:00
local surface_node = nil ;
2014-10-10 01:51:12 +02:00
local has_snow = has_artificial_snow ;
2014-08-27 00:01:48 +02:00
local tree = false ;
2014-09-15 00:40:17 +02:00
local jtree = false ;
2014-08-27 00:01:48 +02:00
local old_height = maxp.y ;
2014-11-20 21:16:48 +01:00
local y = maxp.y ;
2014-08-27 00:01:48 +02:00
-- search for a surface and set everything above target_height to air
while ( y > minp.y ) do
local ci = data [ a : index ( x , y , z ) ] ;
if ( ci == cid.c_snow ) then
has_snow = true ;
elseif ( ci == cid.c_tree ) then
tree = true ;
2014-09-15 05:11:28 +02:00
-- no jungletrees for branches
elseif ( ci == cid.c_jtree and data [ a : index ( x , y - 1 , z ) ] == cid.c_jtree ) then
2014-08-27 00:01:48 +02:00
jtree = true ;
elseif ( not ( surface_node ) and ci ~= cid.c_air and ci ~= cid.c_ignore and mg_villages.check_if_ground ( ci ) == true ) then
-- we have found a surface of some kind
surface_node = ci ;
old_height = y ;
2014-09-14 03:08:30 +02:00
if ( surface_node == cid.c_dirt_with_snow ) then
has_snow = true ;
end
2014-08-27 00:01:48 +02:00
end
-- make sure there is air for the village
if ( y > target_height ) then
data [ a : index ( x , y , z ) ] = cid.c_air ;
-- abort search once we've reached village ground level and found a surface node
elseif ( y <= target_height and surface_node ) then
y = minp.y - 1 ;
end
y = y - 1 ;
end
2014-11-14 00:19:01 +01:00
if ( not ( surface_node ) and old_height == maxp.y ) then
2014-11-20 21:16:48 +01:00
if ( data [ a : index ( x , minp.y , z ) ] == cid.c_air ) then
2014-11-14 00:19:01 +01:00
old_height = vh - 2 ;
elseif ( minp.y < 0 ) then
old_height = minp.y ;
end
2014-09-14 03:08:30 +02:00
end
2014-08-27 00:01:48 +02:00
if ( not ( surface_node ) or surface_node == cid.c_dirt ) then
surface_node = cid.c_dirt_with_grass ;
end
2014-09-16 00:05:53 +02:00
if ( has_snow and surface_node == cid.c_dirt_with_grass and target_height > 1 ) then
surface_node = cid.c_dirt_with_snow ;
end
2014-08-27 00:01:48 +02:00
local below_1 = cid.c_dirt ;
local below_2 = cid.c_stone ;
if ( surface_node == cid.c_desert_sand ) then
below_1 = cid.c_desert_sand ;
below_2 = cid.c_desert_stone ;
elseif ( surface_node == cid.c_sand ) then
below_1 = cid.c_sand ;
below_2 = cid.c_stone ;
2014-08-28 16:16:32 +02:00
elseif ( cid.c_ethereal_clay_read
and ( surface_node == cid.c_ethereal_clay_red
or surface_node == cid.c_ethereal_clay_orange ) ) then
below_1 = cid.c_ethereal_clay_orange ;
below_2 = cid.c_ethereal_clay_orange ;
elseif ( surface_node == cid.c_sandstone ) then
below_1 = cid.c_sandstone ;
below_2 = cid.c_sandstone ;
2014-08-27 00:01:48 +02:00
else
below_1 = cid.c_dirt ;
below_2 = cid.c_stone ;
end
2014-08-28 04:35:42 +02:00
2014-08-28 04:37:22 +02:00
-- do terrain blending; target_height has to be calculated based on old_height
2014-08-28 04:35:42 +02:00
if ( target_height == maxp.y ) then
local yblend = old_height ;
2014-10-11 23:25:50 +02:00
if blend > 0 then -- leave some cliffs unblended
2014-08-28 04:35:42 +02:00
yblend = math.floor ( vh + blend * ( old_height - vh ) )
target_height = yblend + 1 ;
else
target_height = old_height ;
end
for y = yblend , maxp.y do
if ( y <= 1 ) then
data [ a : index ( x , y , z ) ] = cid.c_water ;
else
data [ a : index ( x , y , z ) ] = cid.c_air ;
end
end
end
2014-08-27 00:01:48 +02:00
2014-09-15 05:18:39 +02:00
if ( target_height < 1 ) then
-- no trees or snow below water level
elseif ( has_snow ) then
2014-08-27 00:01:48 +02:00
data [ a : index ( x , target_height + 1 , z ) ] = cid.c_snow ;
2014-09-15 05:11:28 +02:00
elseif ( tree and not ( mg_villages.ethereal_trees ) and treepos ) then
2014-08-28 04:35:42 +02:00
data [ a : index ( x , target_height + 1 , z ) ] = cid.c_sapling
2014-09-15 05:11:28 +02:00
table.insert ( treepos , { x = x , y = target_height + 1 , z = z , typ = 0 } ) ;
elseif ( jtree and not ( mg_villages.ethereal_trees ) and treepos ) then
2014-08-28 04:35:42 +02:00
data [ a : index ( x , target_height + 1 , z ) ] = cid.c_jsapling
2014-09-15 05:11:28 +02:00
table.insert ( treepos , { x = x , y = target_height + 1 , z = z , typ = 1 } ) ;
2014-08-27 00:01:48 +02:00
end
data [ a : index ( x , target_height , z ) ] = surface_node ;
if ( target_height - 1 >= minp.y ) then
data [ a : index ( x , target_height - 1 , z ) ] = below_1 ;
end
-- not every column will get a coal block; some may get two
local coal_height1 = math.random ( minp.y , maxp.y ) ;
local coal_height2 = math.random ( minp.y , maxp.y ) ;
y = target_height - 2 ;
while ( y > minp.y and y > target_height - 40 ) do
local old_node = data [ a : index ( x , y , z ) ] ;
-- abort as soon as we hit anything other than air
if ( old_node == cid.c_air or old_node == cid.c_water ) then
-- the occasional coal makes large stone cliffs slightly less boring
if ( y == coal_height1 or y == coal_height2 ) then
data [ a : index ( x , y , z ) ] = cid.c_stone_with_coal ;
else
data [ a : index ( x , y , z ) ] = below_2 ;
end
y = y - 1 ;
else
y = minp.y - 1 ;
end
end
end
2014-08-25 01:46:33 +02:00
2014-08-05 03:20:07 +02:00
2014-08-27 00:01:48 +02:00
-- adjust the terrain level to the respective height of the village
2014-10-11 23:25:50 +02:00
mg_villages.flatten_village_area = function ( villages , minp , maxp , vm , data , param2_data , a , village_area , cid )
2014-09-15 05:11:28 +02:00
local treepos = { } ;
2014-08-05 03:20:07 +02:00
for z = minp.z , maxp.z do
for x = minp.x , maxp.x do
2014-09-15 05:11:28 +02:00
for village_nr , village in ipairs ( villages ) do
2014-10-11 23:25:50 +02:00
-- is village_nr the village that is the one that is relevant for this spot?
if ( village_area [ x ] [ z ] [ 1 ] > 0
and village_area [ x ] [ z ] [ 1 ] == village_nr
and village_area [ x ] [ z ] [ 2 ] ~= 0
and data [ a : index ( x , village.vh , z ) ] ~= cid.c_ignore ) then
local has_artificial_snow = false ;
if ( village.artificial_snow and village.artificial_snow == 1 ) then
has_artificial_snow = true ;
end
2014-08-26 20:41:29 +02:00
2014-10-11 23:25:50 +02:00
if ( village_area [ x ] [ z ] [ 2 ] > 0 ) then -- inside a village
mg_villages.lower_or_raise_terrain_at_point ( x , z , village.vh , minp , maxp , vm , data , param2_data , a , cid , village.vh ,
nil , has_artificial_snow , 0 ) ;
elseif ( mg_villages.ENABLE_TERRAIN_BLEND and village_area [ x ] [ z ] [ 2 ] < 0 ) then
mg_villages.lower_or_raise_terrain_at_point ( x , z , maxp.y , minp , maxp , vm , data , param2_data , a , cid , village.vh ,
treepos , has_artificial_snow , - 1 * village_area [ x ] [ z ] [ 2 ] ) ;
end
2014-08-25 17:34:16 +02:00
end -- PM ^
end
end
end
2014-09-15 05:11:28 +02:00
-- grow normal trees and jungletrees in those parts of the terrain where height blending occours
for _ , tree in ipairs ( treepos ) do
2014-09-23 18:08:04 +02:00
local plant_id = cid.c_jsapling ;
if ( tree.typ == 0 ) then
plant_id = cid.c_sapling ;
2014-09-15 05:11:28 +02:00
end
2014-09-23 18:08:04 +02:00
mg_villages.grow_a_tree ( { x = tree.x , y = tree.y , z = tree.z } , plant_id , minp , maxp , data , a , cid , nil ) -- no pseudorandom present
2014-09-15 05:11:28 +02:00
end
2014-08-25 17:34:16 +02:00
end
-- TODO: limit this function to the shell in order to speed things up
-- repair mapgen griefings
2014-10-11 00:54:48 +02:00
mg_villages.repair_outer_shell = function ( villages , minp , maxp , vm , data , param2_data , a , village_area , cid )
2014-08-25 17:34:16 +02:00
for z = minp.z , maxp.z do
for x = minp.x , maxp.x do
-- inside a village
if ( village_area [ x ] [ z ] [ 2 ] > 0 ) then
2014-11-20 21:16:48 +01:00
local y ;
2014-08-25 17:34:16 +02:00
local village = villages [ village_area [ x ] [ z ] [ 1 ] ] ;
-- the current node at the ground
local node = data [ a : index ( x , village.vh , z ) ] ;
-- there ought to be something - but there is air
2014-08-27 00:01:48 +02:00
if ( village and village.vh and ( node == cid.c_air or node == cid.c_water ) ) then
2014-08-25 17:34:16 +02:00
y = village.vh - 1 ;
-- search from village height downards for holes generated by cavegen and fill them up
while ( y > minp.y ) do
local ci = data [ a : index ( x , y , z ) ] ;
2014-08-27 00:01:48 +02:00
if ( ci == cid.c_desert_stone or ci == cid.c_desert_sand ) then
data [ a : index ( x , village.vh , z ) ] = cid.c_desert_sand ;
2014-08-25 17:34:16 +02:00
y = minp.y - 1 ;
2014-08-27 00:01:48 +02:00
elseif ( ci == cid.c_sand ) then
data [ a : index ( x , village.vh , z ) ] = cid.c_sand ;
2014-08-25 17:34:16 +02:00
y = minp.y - 1 ;
-- use dirt_with_grass as a fallback
2014-08-27 00:01:48 +02:00
elseif ( ci ~= cid.c_air and ci ~= cid.c_ignore and ci ~= cid.c_water and mg_villages.check_if_ground ( ci ) == true ) then
data [ a : index ( x , village.vh , z ) ] = cid.c_dirt_with_grass ;
2014-08-25 17:34:16 +02:00
y = minp.y - 1 ;
-- abort the search - there is no data available yet
2014-08-27 00:01:48 +02:00
elseif ( ci == cid.c_ignore ) then
2014-08-25 17:34:16 +02:00
y = minp.y - 1 ;
end
y = y - 1 ;
end
end
-- remove mudflow
y = village.vh + 1 ;
while ( y < village.vh + 40 and y < maxp.y ) do
local ci = data [ a : index ( x , y , z ) ] ;
2014-08-27 00:01:48 +02:00
if ( ci ~= cid.c_ignore and ( ci == cid.c_dirt or ci == cid.c_dirt_with_grass or ci == cid.c_sand or ci == cid.c_desert_sand ) ) then
data [ a : index ( x , y , z ) ] = cid.c_air ;
2014-09-16 00:05:53 +02:00
-- if there was a moresnow cover, add a snow on top of the new floor node
2014-11-20 21:16:48 +01:00
elseif ( ci ~= cid.c_ignore
2014-09-16 06:01:37 +02:00
and ( ci == cid.c_msnow_1 or ci == cid.c_msnow_2 or ci == cid.c_msnow_3 or ci == cid.c_msnow_4 or
2014-09-16 00:05:53 +02:00
ci == cid.c_msnow_5 or ci == cid.c_msnow_6 or ci == cid.c_msnow_7 or ci == cid.c_msnow_8 or
ci == cid.c_msnow_9 or ci == cid.c_msnow_10 or ci == cid.c_msnow_11 ) ) then
data [ a : index ( x , village.vh + 1 , z ) ] = cid.c_snow ;
data [ a : index ( x , village.vh , z ) ] = cid.c_dirt_with_snow ;
2014-08-25 17:34:16 +02:00
end
y = y + 1 ;
2014-08-05 03:20:07 +02:00
end
end
end
end
end
2014-08-05 01:24:11 +02:00
2014-08-25 17:34:16 +02:00
2014-08-19 04:56:44 +02:00
-- helper functions for mg_villages.place_villages_via_voxelmanip
-- this one marks the positions of buildings plus a frame around them
2014-08-24 01:03:28 +02:00
mg_villages.village_area_mark_buildings = function ( village_area , village_nr , bpos )
2014-08-19 04:56:44 +02:00
-- mark the roads and buildings and the area between buildings in the village_area table
-- 2: road
-- 3: border around a road
-- 4: building
-- 5: border around a building
for _ , pos in ipairs ( bpos ) do
local reserved_for = 4 ; -- a building will be placed here
if ( pos.btype and pos.btype == ' road ' ) then
reserved_for = 2 ; -- the building will be a road
end
-- the building + a border of 1 around it
for x = - 1 , pos.bsizex do
for z = - 1 , pos.bsizez do
local p = { x = pos.x + x , z = pos.z + z } ;
if ( not ( village_area [ p.x ] ) ) then
village_area [ p.x ] = { } ;
end
if ( x ==- 1 or z ==- 1 or x == pos.bsizex or z == pos.bsizez ) then
village_area [ p.x ] [ p.z ] = { village_nr , reserved_for + 1 } ; -- border around a building
else
village_area [ p.x ] [ p.z ] = { village_nr , reserved_for } ; -- the actual building
2014-08-06 00:12:24 +02:00
end
end
end
2014-08-19 04:56:44 +02:00
end
end
2014-08-24 01:03:28 +02:00
mg_villages.village_area_mark_dirt_roads = function ( village_area , village_nr , dirt_roads )
2014-08-19 04:56:44 +02:00
-- mark the dirt roads
-- 8: dirt road
for _ , pos in ipairs ( dirt_roads ) do
-- the building + a border of 1 around it
for x = 0 , pos.bsizex - 1 do
for z = 0 , pos.bsizez - 1 do
local p = { x = pos.x + x , z = pos.z + z } ;
if ( not ( village_area [ p.x ] ) ) then
village_area [ p.x ] = { } ;
2014-08-18 18:18:53 +02:00
end
2014-08-19 04:56:44 +02:00
village_area [ p.x ] [ p.z ] = { village_nr , 8 } ; -- the actual dirt road
2014-08-18 18:18:53 +02:00
end
end
2014-08-19 04:56:44 +02:00
end
end
2014-08-06 03:00:43 +02:00
2014-08-19 04:56:44 +02:00
mg_villages.village_area_mark_inside_village_area = function ( village_area , villages , village_noise , minp , maxp )
2014-08-06 03:00:43 +02:00
-- mark the rest ( inside_village but not part of an actual building) as well
2014-08-05 19:49:02 +02:00
for x = minp.x , maxp.x do
2014-08-06 00:12:24 +02:00
if ( not ( village_area [ x ] ) ) then
village_area [ x ] = { } ;
end
2014-08-05 19:49:02 +02:00
for z = minp.z , maxp.z do
2014-08-06 00:12:24 +02:00
if ( not ( village_area [ x ] [ z ] ) ) then
village_area [ x ] [ z ] = { 0 , 0 } ;
2014-10-23 16:49:37 +02:00
local n_rawnoise = village_noise : get2d ( { x = x , y = z } ) -- create new blended terrain
2014-08-06 00:12:24 +02:00
for village_nr , village in ipairs ( villages ) do
2014-10-11 23:25:50 +02:00
local vn = mg_villages.get_vn ( x , z , n_rawnoise , village ) ;
2014-11-08 23:01:13 +01:00
if ( village.is_single_house ) then
-- do nothing here; the village area will be specificly marked later on
2014-10-11 23:25:50 +02:00
-- the village core; this is where the houses stand (but there's no house or road at this particular spot)
2014-11-08 23:01:13 +01:00
elseif ( vn <= 40 ) then
2014-10-11 23:25:50 +02:00
village_area [ x ] [ z ] = { village_nr , 6 } ;
-- the flattened land around the village where wheat, cotton, trees or grass may be grown (depending on village type)
elseif ( vn <= 80 ) then
2014-08-06 00:12:24 +02:00
village_area [ x ] [ z ] = { village_nr , 1 } ;
2014-10-11 23:25:50 +02:00
-- terrain blending for the flattened land
elseif ( vn <= 160 and mg_villages.ENABLE_TERRAIN_BLEND ) then
if n_rawnoise > - 0.5 then -- leave some cliffs unblended
local blend = ( ( vn - 80 ) / 80 ) ^ 2 -- 0 at village edge, 1 at normal terrain
-- assign a negative value to terrain that needs to be adjusted in height
village_area [ x ] [ z ] = { village_nr , - 1 * blend } ;
else
-- no height adjustments for this terrain; the terrain is not considered to be part of the village
village_area [ x ] [ z ] = { village_nr , 0 } ;
end
2014-08-06 00:12:24 +02:00
end
2014-08-05 19:49:02 +02:00
end
end
end
end
2014-11-08 23:01:13 +01:00
-- single houses get their own form of terrain blend
local pr = PseudoRandom ( mg_villages.get_bseed ( minp ) ) ;
for village_nr , village in ipairs ( villages ) do
if ( village and village.is_single_house and village.to_add_data and village.to_add_data . bpos and # village.to_add_data . bpos >= 1 ) then
2014-11-13 23:15:57 +01:00
mg_villages.village_area_mark_single_house_area ( village_area , minp , maxp , village.to_add_data . bpos [ 1 ] , pr , village_nr , village ) ;
2014-11-08 23:01:13 +01:00
end
end
2014-08-19 04:56:44 +02:00
end
2014-08-05 19:49:02 +02:00
2014-08-08 18:08:28 +02:00
2014-08-24 01:03:28 +02:00
-- analyzes optimal height for villages which have their center inside this mapchunk
2014-08-27 00:01:48 +02:00
mg_villages.village_area_get_height = function ( village_area , villages , minp , maxp , data , param2_data , a , cid )
2014-08-18 00:10:36 +02:00
-- figuring out the height this way hardly works - because only a tiny part of the village may be contained in this chunk
2014-08-06 03:00:43 +02:00
local height_sum = { } ;
local height_count = { } ;
2014-08-24 01:03:28 +02:00
local height_statistic = { } ;
2014-08-06 03:00:43 +02:00
-- initialize the variables for counting
for village_nr , village in ipairs ( villages ) do
2014-08-24 01:03:28 +02:00
height_sum [ village_nr ] = 0 ;
height_count [ village_nr ] = 0 ;
height_statistic [ village_nr ] = { } ;
2014-08-06 03:00:43 +02:00
end
-- try to find the optimal village height by looking at the borders defined by inside_village
for x = minp.x + 1 , maxp.x - 1 do
for z = minp.z + 1 , maxp.z - 1 do
2014-08-24 01:03:28 +02:00
if ( village_area [ x ] [ z ] [ 1 ] ~= 0
and village_area [ x ] [ z ] [ 2 ] ~= 0
2014-10-11 23:25:50 +02:00
and ( village_area [ x + 1 ] [ z ] [ 2 ] <= 0
or village_area [ x - 1 ] [ z ] [ 2 ] <= 0
or village_area [ x ] [ z + 1 ] [ 2 ] <= 0
or village_area [ x ] [ z - 1 ] [ 2 ] <= 0 )
2014-08-24 01:03:28 +02:00
-- if the corners of the mapblock are inside the village area, they may count as borders here as well
2014-10-11 23:25:50 +02:00
or ( x == minp.x + 1 and village_area [ x - 1 ] [ z ] [ 1 ] >= 0 )
or ( x == maxp.x - 1 and village_area [ x + 1 ] [ z ] [ 1 ] >= 0 )
or ( z == minp.z - 1 and village_area [ x ] [ z - 1 ] [ 1 ] >= 0 )
or ( z == maxp.z + 1 and village_area [ x ] [ z + 1 ] [ 1 ] >= 0 ) ) then
2014-08-06 03:00:43 +02:00
2014-11-20 21:16:48 +01:00
local y = maxp.y ;
2014-08-06 03:00:43 +02:00
while ( y > minp.y and y >= 0 ) do
local ci = data [ a : index ( x , y , z ) ] ;
2014-10-19 21:15:25 +02:00
if ( ( ci ~= cid.c_air and ci ~= cid.c_ignore and mg_villages.check_if_ground ( ci ) == true ) or ( y == 0 ) ) then
2014-08-06 03:00:43 +02:00
local village_nr = village_area [ x ] [ z ] [ 1 ] ;
2014-08-24 01:03:28 +02:00
if ( village_nr > 0 and height_sum [ village_nr ] ) then
height_sum [ village_nr ] = height_sum [ village_nr ] + y ;
height_count [ village_nr ] = height_count [ village_nr ] + 1 ;
if ( not ( height_statistic [ village_nr ] [ y ] ) ) then
height_statistic [ village_nr ] [ y ] = 1 ;
else
height_statistic [ village_nr ] [ y ] = height_statistic [ village_nr ] [ y ] + 1 ;
end
end
2014-08-06 03:00:43 +02:00
y = minp.y - 1 ;
end
y = y - 1 ;
end
end
end
end
for village_nr , village in ipairs ( villages ) do
2014-10-26 02:30:42 +01:00
local tmin = maxp.y ;
local tmax = minp.y ;
local topt = 2 ;
for k , v in pairs ( height_statistic [ village_nr ] ) do
if ( k >= 2 and k < tmin and k >= minp.y ) then
tmin = k ;
end
if ( k <= maxp.y and k > tmax ) then
tmax = k ;
end
if ( height_statistic [ village_nr ] [ topt ]
and height_statistic [ village_nr ] [ topt ] < height_statistic [ village_nr ] [ k ] ) then
topt = k ;
end
end
--print('HEIGHT for village '..tostring( village.name )..' min:'..tostring( tmin )..' max:'..tostring(tmax)..' opt:'..tostring(topt)..' count:'..tostring( height_count[ village_nr ]));
2014-08-25 17:34:16 +02:00
if ( village.optimal_height ) then
2014-08-24 01:03:28 +02:00
-- villages above a size of 40 are *always* place at a convenient height of 1
2014-10-24 18:10:31 +02:00
elseif ( village.vs >= 40 and not ( village.is_single_house ) ) then
2014-08-27 02:11:05 +02:00
village.optimal_height = 2 ;
2014-10-24 18:10:31 +02:00
elseif ( village.vs >= 30 and not ( village.is_single_house ) ) then
2014-08-28 04:35:42 +02:00
village.optimal_height = 41 - village.vs ;
2014-10-24 18:10:31 +02:00
elseif ( village.vs >= 25 and not ( village.is_single_house ) ) then
2014-08-28 04:35:42 +02:00
village.optimal_height = 36 - village.vs ;
2014-08-24 01:03:28 +02:00
2014-10-26 02:30:42 +01:00
-- in some cases, choose that height which was counted most often
elseif ( topt and ( tmax - tmin ) > 8 and height_count [ village_nr ] > 0 ) then
2014-11-20 21:16:48 +01:00
local qmw ;
2014-10-26 02:30:42 +01:00
if ( ( tmax - topt ) > ( topt - tmin ) ) then
qmw = tmax ;
else
qmw = tmin ;
end
village.optimal_height = qmw ;
2014-08-24 01:03:28 +02:00
-- if no border height was found, there'd be no point in calculating anything;
-- also, this is done only if the village has its center inside this mapchunk
2014-08-27 02:11:05 +02:00
elseif ( height_count [ village_nr ] > 0 ) then
2014-08-24 01:03:28 +02:00
local max = 0 ;
local target = village.vh ;
local qmw = 0 ;
for k , v in pairs ( height_statistic [ village_nr ] ) do
qmw = qmw + v * ( k * k ) ;
if ( v > max ) then
target = k ;
max = v ;
end
end
if ( height_count [ village_nr ] > 5 ) then
2014-11-14 01:43:47 +01:00
qmw = math.floor ( math.sqrt ( qmw / height_count [ village_nr ] ) + 1.5 ) ; -- round the value
2014-10-19 21:15:25 +02:00
-- a height of 0 would be one below water level; so let's choose something higher;
-- as this may be an island created withhin deep ocean, it might look better if it extends a bit from said ocean
if ( qmw < 1 ) then
qmw = 2 ;
end
2014-08-24 01:03:28 +02:00
else
qmw = 0 ; -- if in doubt, a height of 0 usually works well
end
village.optimal_height = qmw ;
2014-08-06 03:00:43 +02:00
end
end
2014-08-19 04:56:44 +02:00
end
2014-08-18 00:10:36 +02:00
2014-08-24 01:03:28 +02:00
mg_villages.change_village_height = function ( village , new_height )
print ( ' CHANGING HEIGHT from ' .. tostring ( village.vh ) .. ' to ' .. tostring ( new_height ) ) ;
for _ , pos in ipairs ( village.to_add_data . bpos ) do
pos.y = new_height ;
end
for _ , pos in ipairs ( village.to_add_data . dirt_roads ) do
pos.y = new_height ;
end
village.vh = new_height ;
end
2014-11-20 21:16:48 +01:00
-- those functions from the mg mod do not have their own namespace
if ( minetest.get_modpath ( ' mg ' ) ) then
mg_villages.add_savannatree = add_savannatree ;
mg_villages.add_pinetree = add_pinetree ;
end
2014-08-24 01:03:28 +02:00
2014-09-23 18:08:04 +02:00
mg_villages.grow_a_tree = function ( pos , plant_id , minp , maxp , data , a , cid , pr )
-- a normal tree; sometimes comes with apples
2014-11-17 05:47:12 +01:00
if ( plant_id == cid.c_sapling and minetest.registered_nodes [ ' default:tree ' ] ) then
2014-11-13 22:07:45 +01:00
mg_villages.grow_tree ( data , a , pos , math.random ( 1 , 4 ) == 1 , math.random ( 1 , 100000 ) )
2014-09-23 18:08:04 +02:00
return true ;
-- a normal jungletree
2014-11-17 05:47:12 +01:00
elseif ( plant_id == cid.c_jsapling and minetest.registered_nodes [ ' default:jungletree ' ] ) then
2014-11-13 22:07:45 +01:00
mg_villages.grow_jungletree ( data , a , pos , math.random ( 1 , 100000 ) )
2014-09-23 18:08:04 +02:00
return true ;
-- a savannatree from the mg mod
2014-11-20 21:16:48 +01:00
elseif ( plant_id == cid.c_savannasapling and mg_villages.add_savannatree ) then
mg_villages.add_savannatree ( data , a , pos.x , pos.y , pos.z , minp , maxp , pr )
2014-09-23 18:08:04 +02:00
return true ;
-- a pine tree from the mg mod
2014-11-20 21:16:48 +01:00
elseif ( plant_id == cid.c_pinesapling and mg_villages.add_pinetree ) then
mg_villages.add_pinetree ( data , a , pos.x , pos.y , pos.z , minp , maxp , pr )
2014-09-23 18:08:04 +02:00
return true ;
end
return false ;
end
2014-08-19 04:56:44 +02:00
-- places trees and plants at empty spaces
2014-08-27 00:01:48 +02:00
mg_villages.village_area_fill_with_plants = function ( village_area , villages , minp , maxp , data , param2_data , a , cid )
2014-08-19 03:45:45 +02:00
-- trees which require grow functions to be called
2014-08-27 00:01:48 +02:00
cid.c_savannasapling = minetest.get_content_id ( ' mg:savannasapling ' ) ;
cid.c_pinesapling = minetest.get_content_id ( ' mg:pinesapling ' ) ;
2014-08-06 00:12:24 +02:00
-- add farmland
2014-08-27 00:01:48 +02:00
cid.c_wheat = minetest.get_content_id ( ' farming:wheat_8 ' ) ;
cid.c_cotton = minetest.get_content_id ( ' farming:cotton_8 ' ) ;
cid.c_shrub = minetest.get_content_id ( ' default:dry_shrub ' ) ;
2014-08-24 01:03:28 +02:00
-- these extra nodes are used in order to avoid abms on the huge fields around the villages
2014-08-27 00:01:48 +02:00
cid.c_soil_wet = minetest.get_content_id ( ' mg_villages:soil ' ) ; --'farming:soil_wet' );
cid.c_soil_sand = minetest.get_content_id ( ' mg_villages:desert_sand_soil ' ) ; --'farming:desert_sand_soil_wet' );
2014-08-06 03:00:43 +02:00
-- desert sand soil is only available in minetest_next
2014-08-27 00:01:48 +02:00
if ( not ( cid.c_soil_sand ) ) then
cid.c_soil_sand = cid.c_soil_wet ;
2014-08-06 03:00:43 +02:00
end
2014-08-06 00:12:24 +02:00
local c_water_source = minetest.get_content_id ( ' default:water_source ' ) ;
local c_clay = minetest.get_content_id ( ' default:clay ' ) ;
2014-08-08 18:08:28 +02:00
local c_feldweg = minetest.get_content_id ( ' cottages:feldweg ' ) ;
if ( not ( c_feldweg ) ) then
2014-08-27 00:01:48 +02:00
c_feldweg = cid.c_dirt_with_grass ;
2014-08-08 18:08:28 +02:00
end
2014-08-18 00:10:36 +02:00
2014-09-25 05:22:06 +02:00
if ( mg_villages.realtest_trees ) then
cid.c_soil_wet = minetest.get_content_id ( ' farming:soil ' ) ; -- TODO: the one from mg_villages would be better...but that one lacks textures
cid.c_soil_sand = minetest.get_content_id ( ' farming:soil ' ) ; -- TODO: the one from mg_villages would be better...but that one lacks textures
cid.c_wheat = minetest.get_content_id ( ' farming:spelt_4 ' ) ;
cid.c_cotton = minetest.get_content_id ( ' farming:flax_4 ' ) ;
-- cid.c_shrub = minetest.get_content_id( 'default:dry_shrub');
end
2014-08-19 03:45:45 +02:00
local pr = PseudoRandom ( mg_villages.get_bseed ( minp ) ) ;
2014-08-06 00:12:24 +02:00
for x = minp.x , maxp.x do
for z = minp.z , maxp.z do
2014-08-08 18:08:28 +02:00
-- turn unused land (which is either dirt or desert sand) into a field that grows wheat
2014-10-11 23:25:50 +02:00
if ( village_area [ x ] [ z ] [ 2 ] == 1
or village_area [ x ] [ z ] [ 2 ] == 6 ) then
2014-08-06 00:12:24 +02:00
2014-08-19 16:41:20 +02:00
local village_nr = village_area [ x ] [ z ] [ 1 ] ;
local village = villages [ village_nr ] ;
2014-08-19 03:45:45 +02:00
local h = village.vh ;
2014-08-06 00:12:24 +02:00
local g = data [ a : index ( x , h , z ) ] ;
2014-08-19 03:45:45 +02:00
-- choose a plant/tree with a certain chance
-- Note: There are no checks weather the tree/plant will actually grow there or not;
-- Tree type is derived from wood type used in the village
local plant_id = data [ a : index ( x , h + 1 , z ) ] ;
local on_soil = false ;
for _ , v in ipairs ( village.to_add_data . plantlist ) do
-- select the first plant that fits; if the node is not air, keep what is currently inside
2014-09-16 06:01:37 +02:00
if ( ( plant_id == cid.c_air or plant_id == cid.c_snow ) and ( ( v.p == 1 or pr : next ( 1 , v.p ) == 1 ) ) ) then
2014-08-19 03:45:45 +02:00
-- TODO: check if the plant grows on that soil
plant_id = v.id ;
-- wheat and cotton require soil
2014-08-27 00:01:48 +02:00
if ( plant_id == cid.c_wheat or plant_id == cid.c_cotton ) then
2014-08-19 03:45:45 +02:00
on_soil = true ;
end
end
end
local pos = { x = x , y = h + 1 , z = z } ;
2014-09-23 18:08:04 +02:00
if ( mg_villages.grow_a_tree ( pos , plant_id , minp , maxp , data , a , cid , pr ) ) then
-- nothing to do; the function has grown the tree already
2014-08-19 03:45:45 +02:00
2014-08-25 17:34:16 +02:00
-- grow wheat and cotton on normal wet soil (and re-plant if it had been removed by mudslide)
2014-09-16 06:01:37 +02:00
elseif ( on_soil and ( g == cid.c_dirt_with_grass or g == cid.c_soil_wet or g == cid.c_dirt_with_snow ) ) then
2014-08-07 19:50:10 +02:00
param2_data [ a : index ( x , h + 1 , z ) ] = math.random ( 1 , 179 ) ;
2014-08-19 03:45:45 +02:00
data [ a : index ( x , h + 1 , z ) ] = plant_id ;
2014-08-27 00:01:48 +02:00
data [ a : index ( x , h , z ) ] = cid.c_soil_wet ;
2014-09-16 06:01:37 +02:00
-- put a snow cover on plants where needed
2014-11-20 21:16:48 +01:00
if ( g == cid.c_dirt_with_snow and cid.c_msnow_1 ~= cid.c_ignore ) then
2014-09-16 06:01:37 +02:00
data [ a : index ( x , h + 2 , z ) ] = cid.c_msnow_1 ;
end
2014-08-25 17:34:16 +02:00
-- grow wheat and cotton on desert sand soil - or on soil previously placed (before mudslide overflew it; same as above)
2014-08-27 00:01:48 +02:00
elseif ( on_soil and ( g == cid.c_desert_sand or g == cid.c_soil_sand ) and cid.c_soil_sand and cid.c_soil_sand > 0 ) then
2014-08-07 19:50:10 +02:00
param2_data [ a : index ( x , h + 1 , z ) ] = math.random ( 1 , 179 ) ;
2014-08-19 03:45:45 +02:00
data [ a : index ( x , h + 1 , z ) ] = plant_id ;
2014-08-27 00:01:48 +02:00
data [ a : index ( x , h , z ) ] = cid.c_soil_sand ;
2014-10-11 23:25:50 +02:00
2014-08-19 03:45:45 +02:00
elseif ( on_soil ) then
if ( math.random ( 1 , 5 ) == 1 ) then
2014-08-27 00:01:48 +02:00
data [ a : index ( pos.x , pos.y , pos.z ) ] = cid.c_shrub ;
2014-08-19 03:45:45 +02:00
end
elseif ( plant_id ) then -- place the sapling or plant (moretrees uses spawn_tree)
data [ a : index ( pos.x , pos.y , pos.z ) ] = plant_id ;
2014-08-06 00:12:24 +02:00
end
end
end
end
2014-08-19 04:56:44 +02:00
end
2014-09-23 18:08:04 +02:00
2014-09-22 05:43:21 +02:00
time_elapsed = function ( t_last , msg )
2014-11-20 21:16:48 +01:00
mg_villages.t_now = minetest.get_us_time ( ) ;
print ( ' TIME ELAPSED: ' .. tostring ( mg_villages.t_now - t_last ) .. ' ' .. msg ) ;
return mg_villages.t_now ;
2014-09-22 05:43:21 +02:00
end
2014-08-19 04:56:44 +02:00
2014-12-21 04:21:11 +01:00
mg_villages.save_data = function ( )
save_restore.save_data ( ' mg_all_villages.data ' , mg_villages.all_villages ) ;
end
2014-08-19 04:56:44 +02:00
mg_villages.place_villages_via_voxelmanip = function ( villages , minp , maxp , vm , data , param2_data , a , top )
2014-09-22 05:43:21 +02:00
local t1 = minetest.get_us_time ( ) ;
2014-08-27 00:01:48 +02:00
local cid = { }
cid.c_air = minetest.get_content_id ( ' air ' ) ;
cid.c_ignore = minetest.get_content_id ( ' ignore ' ) ;
cid.c_stone = minetest.get_content_id ( ' default:stone ' ) ;
cid.c_dirt = minetest.get_content_id ( ' default:dirt ' ) ;
cid.c_snow = minetest.get_content_id ( ' default:snow ' ) ;
2014-09-14 03:08:30 +02:00
cid.c_dirt_with_snow = minetest.get_content_id ( ' default:dirt_with_snow ' ) ;
2014-08-27 00:01:48 +02:00
cid.c_dirt_with_grass = minetest.get_content_id ( ' default:dirt_with_grass ' ) ;
cid.c_desert_sand = minetest.get_content_id ( ' default:desert_sand ' ) ; -- PM v
cid.c_desert_stone = minetest.get_content_id ( ' default:desert_stone ' ) ;
cid.c_sand = minetest.get_content_id ( ' default:sand ' ) ;
cid.c_tree = minetest.get_content_id ( ' default:tree ' ) ;
cid.c_sapling = minetest.get_content_id ( ' default:sapling ' ) ;
cid.c_jtree = minetest.get_content_id ( ' default:jungletree ' ) ;
cid.c_jsapling = minetest.get_content_id ( ' default:junglesapling ' ) ;
cid.c_water = minetest.get_content_id ( ' default:water_source ' ) ; -- PM ^
cid.c_stone_with_coal = minetest.get_content_id ( ' default:stone_with_coal ' ) ;
2014-08-28 16:16:32 +02:00
cid.c_sandstone = minetest.get_content_id ( ' default:sandstone ' ) ;
2014-08-27 00:01:48 +02:00
2014-09-16 00:05:53 +02:00
cid.c_msnow_1 = minetest.get_content_id ( ' moresnow:snow_top ' ) ;
cid.c_msnow_2 = minetest.get_content_id ( ' moresnow:snow_fence_top ' ) ;
cid.c_msnow_3 = minetest.get_content_id ( ' moresnow:snow_stair_top ' ) ;
cid.c_msnow_4 = minetest.get_content_id ( ' moresnow:snow_slab_top ' ) ;
cid.c_msnow_5 = minetest.get_content_id ( ' moresnow:snow_panel_top ' ) ;
cid.c_msnow_6 = minetest.get_content_id ( ' moresnow:snow_micro_top ' ) ;
cid.c_msnow_7 = minetest.get_content_id ( ' moresnow:snow_outer_stair_top ' ) ;
cid.c_msnow_8 = minetest.get_content_id ( ' moresnow:snow_inner_stair_top ' ) ;
cid.c_msnow_9 = minetest.get_content_id ( ' moresnow:snow_ramp_top ' ) ;
cid.c_msnow_10 = minetest.get_content_id ( ' moresnow:snow_ramp_outer_top ' ) ;
cid.c_msnow_11 = minetest.get_content_id ( ' moresnow:snow_ramp_inner_top ' ) ;
2014-12-21 04:21:11 +01:00
cid.c_plotmarker = minetest.get_content_id ( ' mg_villages:plotmarker ' ) ;
2014-08-27 00:01:48 +02:00
2014-08-28 16:16:32 +02:00
if ( minetest.get_modpath ( ' ethereal ' ) ) then
cid.c_ethereal_clay_red = minetest.get_content_id ( ' bakedclay:red ' ) ;
cid.c_ethereal_clay_orange = minetest.get_content_id ( ' bakedclay:orange ' ) ;
end
2014-08-24 01:03:28 +02:00
2014-09-24 02:02:17 +02:00
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' defines ' ) ;
2014-08-19 04:56:44 +02:00
local village_noise = minetest.get_perlin ( 7635 , 3 , 0.5 , 16 ) ;
-- determine which coordinates are inside the village and which are not
local village_area = { } ;
for village_nr , village in ipairs ( villages ) do
-- generate the village structure: determine positions of buildings and roads
mg_villages.generate_village ( village , village_noise ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' generate_village ' ) ;
2014-08-19 04:56:44 +02:00
2014-11-08 23:01:13 +01:00
if ( not ( village.is_single_house ) ) then
-- only add artificial snow if the village has at least a size of 15 (else it might look too artificial)
if ( not ( village.artificial_snow ) and village.vs > 15 ) then
if ( mg_villages.artificial_snow_probability and math.random ( 1 , mg_villages.artificial_snow_probability ) == 1 ) then
village.artificial_snow = 1 ;
else
village.artificial_snow = 0 ;
end
2014-10-10 01:51:12 +02:00
end
2014-11-08 23:01:13 +01:00
-- will set village_area to N where .. is:
-- 2: a building
-- 3: border around a building
-- 4: a road
-- 5: border around a road
mg_villages.village_area_mark_buildings ( village_area , village_nr , village.to_add_data . bpos ) ;
t1 = time_elapsed ( t1 , ' mark_buildings ' ) ;
-- will set village_area to N where .. is:
-- 8: a dirt road
mg_villages.village_area_mark_dirt_roads ( village_area , village_nr , village.to_add_data . dirt_roads ) ;
t1 = time_elapsed ( t1 , ' mark_dirt_roads ' ) ;
2014-10-10 01:51:12 +02:00
end
2014-08-19 04:56:44 +02:00
end
2014-11-20 21:16:48 +01:00
local emin ;
local emax ;
2014-08-19 04:56:44 +02:00
-- if no voxelmanip data was passed on, read the data here
if ( not ( vm ) or not ( a ) or not ( data ) or not ( param2_data ) ) then
vm , emin , emax = minetest.get_mapgen_object ( " voxelmanip " )
if ( not ( vm ) ) then
return ;
end
a = VoxelArea : new {
MinEdge = { x = emin.x , y = emin.y , z = emin.z } ,
MaxEdge = { x = emax.x , y = emax.y , z = emax.z } ,
}
data = vm : get_data ( )
param2_data = vm : get_param2_data ( )
end
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' get_vmap_data ' ) ;
2014-08-19 04:56:44 +02:00
2014-08-25 17:34:16 +02:00
-- all vm manipulation functions write their content to the *entire* volume/area - including those 16 nodes that
-- extend into neighbouring mapchunks; thus, cavegen griefing and mudflow can be repaired by placing everythiing again
local tmin = emin ;
local tmax = emax ;
-- if set to true, cavegen eating through houses and mudflow on roofs will NOT be repaired
2014-08-26 20:41:29 +02:00
if ( not ( mg_villages.UNDO_CAVEGEN_AND_MUDFLOW ) ) then
2014-08-25 17:34:16 +02:00
tmin = minp ;
tmax = maxp ;
end
2014-10-10 23:54:08 +02:00
-- TODO: this needs a modified version for the individual, standalone buildings - or needs a diffrent noise function!
2014-10-11 23:25:50 +02:00
-- will set village_area to N where .. is:
-- 0: not part of any village
-- 1: flattened area around the village; plants (wheat, cotton, trees, grass, ...) may be planted here
-- 6: free/unused spot in the core area of the village where the buildings are
-- negative value: do terrain blending
2014-08-25 17:34:16 +02:00
mg_villages.village_area_mark_inside_village_area ( village_area , villages , village_noise , tmin , tmax ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' mark_inside_village_area ' ) ;
2014-08-19 04:56:44 +02:00
2014-08-24 01:03:28 +02:00
-- determine optimal height for all villages that have their center in this mapchunk; sets village.optimal_height
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' get_height ' ) ;
2014-11-14 01:43:47 +01:00
if ( mg_villages.all_villages and ( not ( mg_villages.ENABLE_VILLAGES ) or mg_villages.anz_villages > 0 ) ) then
2014-10-13 17:17:42 +02:00
mg_villages.village_area_get_height ( village_area , villages , tmin , tmax , data , param2_data , a , cid ) ;
-- the villages in the first mapchunk are set to a fixed height of 1 so that players will not end up embedded in stone
else
for _ , village in ipairs ( villages ) do
village.optimal_height = 1 ;
end
end
2014-08-24 01:03:28 +02:00
-- change height of those villages where an optimal_height could be determined
2014-09-15 05:11:28 +02:00
local village_data_updated = false ;
2014-08-24 01:03:28 +02:00
for _ , village in ipairs ( villages ) do
2014-10-11 23:34:41 +02:00
if ( village.optimal_height and village.optimal_height > 0 and village.optimal_height ~= village.vh ) then
2014-10-20 20:03:43 +02:00
-- towers are usually found on elevated places
if ( village.village_type == ' tower ' ) then
village.optimal_height = village.optimal_height + math.max ( math.floor ( village.vs / 2 ) , 2 ) ;
end
2014-08-24 01:03:28 +02:00
mg_villages.change_village_height ( village , village.optimal_height ) ;
2014-09-15 05:11:28 +02:00
village_data_updated = true ;
2014-08-24 01:03:28 +02:00
end
end
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' change_height ' ) ;
2014-08-24 01:03:28 +02:00
2014-10-11 23:25:50 +02:00
mg_villages.flatten_village_area ( villages , minp , maxp , vm , data , param2_data , a , village_area , cid ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' flatten_village_area ' ) ;
2014-08-25 17:34:16 +02:00
-- repair cavegen griefings and mudflow which may have happened in the outer shell (which is part of other mapnodes)
2014-10-11 23:25:50 +02:00
mg_villages.repair_outer_shell ( villages , tmin , tmax , vm , data , param2_data , a , village_area , cid ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' repair_outer_shell ' ) ;
2014-08-19 04:56:44 +02:00
local c_feldweg = minetest.get_content_id ( ' cottages:feldweg ' ) ;
if ( not ( c_feldweg ) ) then
c_feldweg = minetest.get_content_id ( ' default:cobble ' ) ;
end
for _ , village in ipairs ( villages ) do
2014-12-21 04:21:11 +01:00
-- the village_id will be stored in the plot markers
local village_id = tostring ( village.vx ) .. ' : ' .. tostring ( village.vz ) ;
village.to_add_data = mg_villages.place_buildings ( village , tmin , tmax , data , param2_data , a , cid , village_id ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' place_buildings ' ) ;
2014-08-19 04:56:44 +02:00
2014-10-11 00:54:48 +02:00
mg_villages.place_dirt_roads ( village , tmin , tmax , data , param2_data , a , c_feldweg ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' place_dirt_roads ' ) ;
2014-09-24 23:44:34 +02:00
-- grow trees which are part of buildings into saplings
for _ , v in ipairs ( village.to_add_data . extra_calls.trees ) do
2014-11-20 21:16:48 +01:00
mg_villages.grow_a_tree ( v , v.typ , minp , maxp , data , a , cid , nil ) ; -- TODO: supply pseudorandom value?
2014-09-24 23:44:34 +02:00
end
2014-08-19 04:56:44 +02:00
end
2014-08-06 00:12:24 +02:00
2014-08-27 00:01:48 +02:00
mg_villages.village_area_fill_with_plants ( village_area , villages , tmin , tmax , data , param2_data , a , cid ) ;
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' fill_with_plants ' ) ;
2014-08-18 00:10:36 +02:00
2014-08-05 01:24:11 +02:00
vm : set_data ( data )
vm : set_param2_data ( param2_data )
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' vm data set ' ) ;
2014-08-05 01:24:11 +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 }
)
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' vm calc lighting ' ) ;
2014-08-05 01:24:11 +02:00
vm : write_to_map ( data )
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' vm data written ' ) ;
2014-08-05 01:24:11 +02:00
2014-11-08 23:01:13 +01:00
vm : update_liquids ( )
t1 = time_elapsed ( t1 , ' vm update liquids ' ) ;
2014-09-24 23:44:34 +02:00
-- do on_construct calls AFTER the map data has been written - else i.e. realtest fences can not update themshevles
for _ , village in ipairs ( villages ) do
for k , v in pairs ( village.to_add_data . extra_calls.on_constr ) do
local node_name = minetest.get_name_from_content_id ( k ) ;
if ( minetest.registered_nodes [ node_name ] . on_construct ) then
for _ , pos in ipairs ( v ) do
minetest.registered_nodes [ node_name ] . on_construct ( pos ) ;
end
end
end
end
2014-09-28 03:56:24 +02:00
local pr = PseudoRandom ( mg_villages.get_bseed ( minp ) ) ;
for _ , village in ipairs ( villages ) do
for _ , v in ipairs ( village.to_add_data . extra_calls.chests ) do
local building_nr = village.to_add_data . bpos [ v.bpos_i ] ;
local building_typ = mg_villages.BUILDINGS [ building_nr.btype ] . scm ;
mg_villages.fill_chest_random ( v , pr , building_nr , building_typ ) ;
end
end
-- TODO: extra_calls.signs
2014-09-24 23:44:34 +02:00
2014-08-05 01:24:11 +02:00
-- initialize the pseudo random generator so that the chests will be filled in a reproducable pattern
local meta
for _ , village in ipairs ( villages ) do
-- now add those buildings which are .mts files and need to be placed by minetest.place_schematic(...)
2014-09-30 16:08:50 +02:00
-- place_schematics is no longer needed
--mg_villages.place_schematics( village.to_add_data.bpos, village.to_add_data.replacements, a, pr );
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' place_schematics ' ) ;
2014-08-05 01:24:11 +02:00
if ( not ( mg_villages.all_villages ) ) then
mg_villages.all_villages = { } ;
end
-- unique id - there can only be one village at a given pair of x,z coordinates
local village_id = tostring ( village.vx ) .. ' : ' .. tostring ( village.vz ) ;
-- the village data is saved only once per village - and not whenever part of the village is generated
if ( not ( mg_villages.all_villages [ village_id ] ) ) then
-- count how many villages we already have and assign each village a uniq number
local count = 1 ;
for _ , v in pairs ( mg_villages.all_villages ) do
count = count + 1 ;
end
2014-09-28 01:59:41 +02:00
village.extra_calls = { } ; -- do not save these values
2014-08-05 01:24:11 +02:00
village.nr = count ;
mg_villages.anz_villages = count ;
mg_villages.all_villages [ village_id ] = minetest.deserialize ( minetest.serialize ( village ) ) ;
print ( " Village No. " .. tostring ( count ) .. " of type \' " .. tostring ( village.village_type ) .. " \' of size " .. tostring ( village.vs ) .. " spawned at: x = " .. village.vx .. " , z = " .. village.vz )
2014-09-15 05:11:28 +02:00
village_data_updated = true ;
2014-08-05 01:24:11 +02:00
end
end
2014-09-15 05:11:28 +02:00
if ( village_data_updated ) then
2014-12-21 04:21:11 +01:00
mg_villages.save_data ( ) ;
2014-09-15 05:11:28 +02:00
end
2014-10-20 17:16:05 +02:00
t1 = time_elapsed ( t1 , ' save village data ' ) ;
2014-09-22 05:43:21 +02:00
2014-08-05 01:24:11 +02:00
end
2014-08-05 19:55:03 +02:00
-- the actual mapgen
-- It only does changes if there is at least one village in the area that is to be generated.
minetest.register_on_generated ( function ( minp , maxp , seed )
-- only generate village on the surface chunks
2014-08-24 01:03:28 +02:00
if ( minp.y ~= - 32 or minp.y < - 32 or minp.y > 64 ) then
2014-08-05 19:55:03 +02:00
return ;
end
2014-12-13 06:51:00 +01:00
-- this function has to be called ONCE and AFTER all village types and buildings have been added
-- (which might have been done by other mods so we can't do this earlier)
if ( not ( mg_villages.village_types ) ) then
mg_villages.init_weights ( ) ;
end
2014-10-19 20:07:40 +02:00
local villages = { } ;
-- create normal villages
if ( mg_villages.ENABLE_VILLAGES == true ) then
2014-10-23 16:49:37 +02:00
villages = mg_villages.villages_in_mapchunk ( minp , maxp.x - minp.x + 1 ) ;
2014-10-19 20:07:40 +02:00
end
2014-10-23 16:49:37 +02:00
2014-10-19 20:07:40 +02:00
-- if this mapchunk contains no part of a village, probably a lone building may be found in it
2014-10-23 16:49:37 +02:00
if ( mg_villages.INVERSE_HOUSE_DENSITY > 0 ) then
villages = mg_villages.houses_in_mapchunk ( minp , maxp.x - minp.x + 1 , villages ) ;
end
-- check if the village exists already
local v_nr = 1 ;
for v_nr , village in ipairs ( villages ) do
local village_id = tostring ( village.vx ) .. ' : ' .. tostring ( village.vz ) ;
if ( not ( village.name ) or village.name == ' ' ) then
village.name = ' unknown ' ;
end
if ( mg_villages.all_villages and mg_villages.all_villages [ village_id ] ) then
villages [ v_nr ] = mg_villages.all_villages [ village_id ] ;
end
2014-10-19 20:07:40 +02:00
end
2014-10-23 16:49:37 +02:00
2014-08-05 19:55:03 +02:00
if ( villages and # villages > 0 ) then
mg_villages.place_villages_via_voxelmanip ( villages , minp , maxp , nil , nil , nil , nil , nil ) ;
end
end )