mirror of
https://git.minetest.land/MineClone2/MineClone2.git
synced 2025-01-10 10:37:29 +01:00
311 lines
12 KiB
Lua
311 lines
12 KiB
Lua
--[[
|
|
-------------------------------------------------------------------------------
|
|
-- build schematic, replace material, rotation
|
|
-------------------------------------------------------------------------------
|
|
function mcl_villages.build_schematic(vm, data, va, pos, building, replace_wall, name)
|
|
-- get building node material for better integration to surrounding
|
|
local platform_material = mcl_vars.get_node(pos)
|
|
if not platform_material or (platform_material.name == "air" or platform_material.name == "ignore") then
|
|
return
|
|
end
|
|
platform_material = platform_material.name
|
|
-- pick random material
|
|
local material = wallmaterial[math.random(1,#wallmaterial)]
|
|
-- schematic conversion to lua
|
|
local schem_lua = minetest.serialize_schematic(building,
|
|
"lua",
|
|
{lua_use_comments = false, lua_num_indent_spaces = 0}).." return schematic"
|
|
-- replace material
|
|
if replace_wall == "y" then
|
|
schem_lua = schem_lua:gsub("mcl_core:cobble", material)
|
|
end
|
|
schem_lua = schem_lua:gsub("mcl_core:dirt_with_grass",
|
|
platform_material)
|
|
|
|
-- Disable special junglewood for now.
|
|
-- special material for spawning npcs
|
|
-- schem_lua = schem_lua:gsub("mcl_core:junglewood",
|
|
-- "settlements:junglewood")
|
|
--
|
|
|
|
-- format schematic string
|
|
local schematic = loadstring(schem_lua)()
|
|
-- build foundation for the building an make room above
|
|
local width = schematic["size"]["x"]
|
|
local depth = schematic["size"]["z"]
|
|
local height = schematic["size"]["y"]
|
|
local possible_rotations = {"0", "90", "180", "270"}
|
|
local rotation = possible_rotations[ math.random( #possible_rotations ) ]
|
|
mcl_villages.foundation(
|
|
pos,
|
|
width,
|
|
depth,
|
|
height,
|
|
rotation)
|
|
vm:set_data(data)
|
|
-- place schematic
|
|
|
|
minetest.place_schematic_on_vmanip(
|
|
vm,
|
|
pos,
|
|
schematic,
|
|
rotation,
|
|
nil,
|
|
true)
|
|
vm:write_to_map(true)
|
|
end]]
|
|
-------------------------------------------------------------------------------
|
|
-- initialize settlement_info
|
|
-------------------------------------------------------------------------------
|
|
function mcl_villages.initialize_settlement_info(pr)
|
|
local count_buildings = {}
|
|
|
|
-- count_buildings table reset
|
|
for k,v in pairs(mcl_villages.schematic_table) do
|
|
count_buildings[v["name"]] = 0
|
|
end
|
|
|
|
-- randomize number of buildings
|
|
local number_of_buildings = pr:next(10, 25)
|
|
local number_built = 0
|
|
mcl_villages.debug("Village ".. number_of_buildings)
|
|
|
|
return count_buildings, number_of_buildings, number_built
|
|
end
|
|
|
|
-------------------------------------------------------------------------------
|
|
-- check ground for a single building
|
|
-------------------------------------------------------------------------------
|
|
local function try_place_building(pos_surface, building_all_info, rotation, settlement_info, pr)
|
|
local fwidth, fdepth = building_all_info["hwidth"], building_all_info["hdepth"]
|
|
if rotation == "90" or rotation == "270" then fwidth, fdepth = fdepth, fwidth end
|
|
local fheight = building_all_info["hheight"]
|
|
-- use building centers for better placement
|
|
pos_surface.x = pos_surface.x - math.ceil(fwidth / 2)
|
|
pos_surface.z = pos_surface.z - math.ceil(fdepth / 2)
|
|
-- to find the y position, also check the corners
|
|
local ys = {pos_surface.y}
|
|
local pos_c
|
|
pos_c = mcl_villages.find_surface_down(vector.new(pos_surface.x-1, pos_surface.y+fheight, pos_surface.z-1))
|
|
if pos_c then table.insert(ys, pos_c.y) end
|
|
pos_c = mcl_villages.find_surface_down(vector.new(pos_surface.x+fwidth+2, pos_surface.y+fheight, pos_surface.z-1))
|
|
if pos_c then table.insert(ys, pos_c.y) end
|
|
pos_c = mcl_villages.find_surface_down(vector.new(pos_surface.x-1, pos_surface.y+fheight, pos_surface.z+fdepth+2))
|
|
if pos_c then table.insert(ys, pos_c.y) end
|
|
pos_c = mcl_villages.find_surface_down(vector.new(pos_surface.x+fwidth+2, pos_surface.y+fheight, pos_surface.z+fdepth+2))
|
|
if pos_c then table.insert(ys, pos_c.y) end
|
|
table.sort(ys)
|
|
-- well supported base, not too uneven?
|
|
if #ys < 5 or ys[#ys]-ys[1] > fheight + 3 then return nil end
|
|
pos_surface.y = ys[math.ceil(#ys/2)]
|
|
-- check distance to other buildings
|
|
if mcl_villages.check_distance(settlement_info, pos_surface, building_all_info["hsize"]) then
|
|
return pos_surface
|
|
end
|
|
return nil
|
|
end
|
|
-------------------------------------------------------------------------------
|
|
-- fill settlement_info
|
|
--------------------------------------------------------------------------------
|
|
function mcl_villages.create_site_plan(minp, maxp, pr)
|
|
local center = vector.new(math.floor((minp.x+maxp.x)/2),maxp.y,math.floor((minp.z+maxp.z)/2))
|
|
minetest.log("action", "sudo make me a village at: " .. minetest.pos_to_string(center))
|
|
local possible_rotations = {"0", "90", "180", "270"}
|
|
local center_surface
|
|
|
|
local count_buildings, number_of_buildings, number_built = mcl_villages.initialize_settlement_info(pr)
|
|
local settlement_info = {}
|
|
-- now some buildings around in a circle, radius = size of town center
|
|
local x, y, z, r = center.x, maxp.y, center.z, 0
|
|
-- draw j circles around center and increase radius by math.random(2,5)
|
|
for j = 1,10 do
|
|
for angle = 0, math.pi*2, 0.262 do -- 24 attempts on a circle
|
|
local pos1 = vector.new(math.floor(x + r * math.cos(angle) + 0.5), y, math.floor(z - r * math.sin(angle) + 0.5))
|
|
local pos_surface, surface_material = mcl_villages.find_surface(pos1, false)
|
|
if pos_surface then
|
|
local randomized_schematic_table = mcl_villages.shuffle(mcl_villages.schematic_table, pr)
|
|
if #settlement_info == 0 then randomized_schematic_table = { mcl_villages.schematic_table[1] } end -- place town bell first
|
|
-- pick schematic
|
|
local size = #randomized_schematic_table
|
|
for i = 1, #randomized_schematic_table do
|
|
local building_all_info = randomized_schematic_table[i]
|
|
-- already enough buildings of that type?
|
|
if count_buildings[building_all_info["name"]] < building_all_info["max_num"]*number_of_buildings then
|
|
local rotation = possible_rotations[pr:next(1, #possible_rotations)]
|
|
local pos = try_place_building(pos_surface, building_all_info, rotation, settlement_info, pr)
|
|
if pos then
|
|
if #settlement_info == 0 then -- town bell
|
|
center_surface, y = pos, pos.y + max_height_difference
|
|
end
|
|
-- limit height differences to town center
|
|
if math.abs(pos.y - center_surface.y) > max_height_difference * 0.7 then
|
|
break -- other buildings likely will not fit either
|
|
end
|
|
count_buildings[building_all_info["name"]] = count_buildings[building_all_info["name"]] +1
|
|
number_built = number_built + 1
|
|
|
|
pos.y = pos.y + (building_all_info["yadjust"] or 0)
|
|
table.insert(settlement_info, {
|
|
pos = pos,
|
|
name = building_all_info["name"],
|
|
hsize = building_all_info["hsize"],
|
|
rotat = rotation,
|
|
surface_mat = surface_material
|
|
})
|
|
-- minetest.log("action", "Placing "..building_all_info["name"].." at "..minetest.pos_to_string(pos))
|
|
break
|
|
end
|
|
end
|
|
end
|
|
if number_of_buildings == number_built then
|
|
break
|
|
end
|
|
end
|
|
if r == 0 then break end -- no angles in the very center
|
|
end
|
|
if number_built >= number_of_buildings then
|
|
break
|
|
end
|
|
r = r + pr:next(2,5)
|
|
if r > 35 then break end -- avoid touching neighboring blocks
|
|
end
|
|
mcl_villages.debug("really ".. number_built)
|
|
if number_built <= 8 then
|
|
minetest.log("action", "Bad village location, could only place "..number_built.." buildings.")
|
|
return
|
|
end
|
|
minetest.log("action", "Village completed at " .. minetest.pos_to_string(center))
|
|
minetest.log("Village completed at " .. minetest.pos_to_string(center)) -- for debugging only
|
|
return settlement_info
|
|
end
|
|
-------------------------------------------------------------------------------
|
|
-- evaluate settlement_info and place schematics
|
|
-------------------------------------------------------------------------------
|
|
-- Initialize node
|
|
local function construct_node(p1, p2, name)
|
|
local r = minetest.registered_nodes[name]
|
|
if r then
|
|
if r.on_construct then
|
|
local nodes = minetest.find_nodes_in_area(p1, p2, name)
|
|
for p=1, #nodes do
|
|
local pos = nodes[p]
|
|
r.on_construct(pos)
|
|
end
|
|
return nodes
|
|
end
|
|
minetest.log("warning", "[mcl_villages] No on_construct defined for node name " .. name)
|
|
return
|
|
end
|
|
minetest.log("warning", "[mcl_villages] Attempt to 'construct' inexistant nodes: " .. name)
|
|
end
|
|
|
|
local function spawn_iron_golem(pos)
|
|
--minetest.log("action", "Attempt to spawn iron golem.")
|
|
local p = minetest.find_node_near(pos,50,"mcl_core:grass_path")
|
|
if p then
|
|
p.y = p.y + 1
|
|
local l=minetest.add_entity(p,"mobs_mc:iron_golem"):get_luaentity()
|
|
if l then
|
|
l._home = p
|
|
end
|
|
end
|
|
end
|
|
|
|
local function spawn_villagers(minp,maxp)
|
|
--minetest.log("action", "Attempt to spawn villagers.")
|
|
local beds=minetest.find_nodes_in_area(vector.offset(minp,-20,-20,-20),vector.offset(maxp,20,20,20),{"mcl_beds:bed_red_bottom"})
|
|
for _,bed in pairs(beds) do
|
|
local m = minetest.get_meta(bed)
|
|
if m:get_string("villager") == "" then
|
|
local v=minetest.add_entity(bed,"mobs_mc:villager")
|
|
if v then
|
|
local l=v:get_luaentity()
|
|
l._bed = bed
|
|
m:set_string("villager",l._id)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function fix_village_water(minp,maxp)
|
|
local palettenodes = minetest.find_nodes_in_area(vector.offset(minp,-20,-20,-20),vector.offset(maxp,20,20,20), "group:water_palette")
|
|
for _, palettenodepos in pairs(palettenodes) do
|
|
local palettenode = minetest.get_node(palettenodepos)
|
|
minetest.set_node(palettenodepos, {name = palettenode.name})
|
|
end
|
|
end
|
|
|
|
local function init_nodes(p1, p2, size, rotation, pr)
|
|
construct_node(p1, p2, "mcl_itemframes:item_frame")
|
|
construct_node(p1, p2, "mcl_furnaces:furnace")
|
|
construct_node(p1, p2, "mcl_anvils:anvil")
|
|
|
|
construct_node(p1, p2, "mcl_smoker:smoker")
|
|
construct_node(p1, p2, "mcl_barrels:barrel_closed")
|
|
construct_node(p1, p2, "mcl_blast_furnace:blast_furnace")
|
|
construct_node(p1, p2, "mcl_brewing:stand_000")
|
|
local nodes = construct_node(p1, p2, "mcl_chests:chest")
|
|
if nodes and #nodes > 0 then
|
|
for p=1, #nodes do
|
|
local pos = nodes[p]
|
|
mcl_villages.fill_chest(pos, pr)
|
|
end
|
|
end
|
|
end
|
|
|
|
function mcl_villages.place_schematics(settlement_info, pr)
|
|
local building_all_info
|
|
local lvm = VoxelManip()
|
|
|
|
for i, built_house in ipairs(settlement_info) do
|
|
local is_last = i == #settlement_info
|
|
|
|
for j, schem in ipairs(mcl_villages.schematic_table) do
|
|
if settlement_info[i]["name"] == schem["name"] then
|
|
building_all_info = schem
|
|
break
|
|
end
|
|
end
|
|
|
|
local pos = settlement_info[i]["pos"]
|
|
local rotation = settlement_info[i]["rotat"]
|
|
-- get building node material for better integration to surrounding
|
|
local surface_material = settlement_info[i]["surface_mat"] or "mcl_core:stone"
|
|
local platform_material = surface_material
|
|
local schem_lua = building_all_info["schem_lua"]
|
|
if not schem_lua then
|
|
schem_lua = minetest.serialize_schematic(building_all_info["mts"], "lua", { lua_use_comments = false, lua_num_indent_spaces = 0 }) .. " return schematic"
|
|
building_all_info["schem_lua"] = schem_lua
|
|
end
|
|
schem_lua = schem_lua:gsub('"mcl_core:dirt"', '"'..platform_material..'"')
|
|
schem_lua = schem_lua:gsub('"mcl_core:dirt_with_grass"', '"'..surface_material..'"')
|
|
local schematic = loadstring(mcl_villages.substitute_materials(pos, schem_lua, pr))()
|
|
|
|
local is_belltower = building_all_info["name"] == "belltower"
|
|
|
|
-- already built the foundation for the building and made room above
|
|
local sx, sy, sz = schematic.size.x, schematic.size.y, schematic.size.z
|
|
local p2 = vector.new(pos.x+sx-1,pos.y+sy-1,pos.z+sz-1)
|
|
lvm:read_from_map(pos, p2)
|
|
minetest.place_schematic_on_vmanip(
|
|
lvm,
|
|
pos,
|
|
schematic,
|
|
rotation,
|
|
nil,
|
|
true,
|
|
{ place_center_x = false, place_center_y = false, place_center_z = false }
|
|
)
|
|
lvm:write_to_map(true) -- FIXME: postpone
|
|
if rotation == "90" or rotation == "270" then sx, sz = sz, sx end
|
|
init_nodes(pos, p2, schematic.size, rotation, pr)
|
|
|
|
if is_belltower then
|
|
spawn_iron_golem(pos)
|
|
else
|
|
spawn_villagers(pos,p2)
|
|
fix_village_water(pos,p2)
|
|
end
|
|
end
|
|
end
|