2017-05-24 10:46:40 +03:00
|
|
|
-- FIXME: Rarely, dungoens can overlap and destroy each other
|
|
|
|
|
2017-05-24 11:55:25 +03:00
|
|
|
local pr = PseudoRandom(os.time())
|
|
|
|
|
|
|
|
-- Get loot for dungeon chests
|
|
|
|
local get_loot = function()
|
|
|
|
local items = mcl_loot.get_multi_loot({
|
|
|
|
{
|
|
|
|
stacks_min = 1,
|
|
|
|
stacks_max = 3,
|
|
|
|
items = {
|
|
|
|
{ itemstring = "mobs:nametag", weight = 20 },
|
|
|
|
{ itemstring = "mcl_mobitems:saddle", weight = 20 },
|
|
|
|
{ itemstring = "mcl_jukebox:record_1", weight = 15 },
|
|
|
|
{ itemstring = "mcl_jukebox:record_4", weight = 15 },
|
|
|
|
-- TODO: Iron Horse Armor
|
|
|
|
{ itemstring = "mcl_core:iron_ingot", weight = 15 },
|
|
|
|
{ itemstring = "mcl_core:apple_gold", weight = 15 },
|
|
|
|
-- TODO: Enchanted Book
|
|
|
|
{ itemstring = "mcl_books:book", weight = 10 },
|
|
|
|
-- TODO: Gold Horse Armor
|
|
|
|
{ itemstring = "mcl_core:gold_ingot", weight = 5 },
|
|
|
|
-- TODO: Diamond Horse Armor
|
|
|
|
{ itemstring = "mcl_core:diamond", weight = 5 },
|
|
|
|
-- TODO: Enchanted Golden Apple
|
|
|
|
{ itemstring = "mcl_core:apple_gold", weight = 2 },
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
stacks_min = 1,
|
|
|
|
stacks_max = 4,
|
|
|
|
items = {
|
|
|
|
{ itemstring = "mcl_farming:wheat_item", weight = 20, amount_min = 1, amount_max = 4 },
|
|
|
|
{ itemstring = "mcl_farming:bread", weight = 20 },
|
|
|
|
{ itemstring = "mcl_core:coal_lump", weight = 15, amount_min = 1, amount_max = 4 },
|
|
|
|
{ itemstring = "mesecons:redstone", weight = 15, amount_min = 1, amount_max = 4 },
|
|
|
|
{ itemstring = "mcl_farming:beetroot_seeds", weight = 10, amount_min = 2, amount_max = 4 },
|
|
|
|
{ itemstring = "mcl_farming:melon_seeds", weight = 10, amount_min = 2, amount_max = 4 },
|
|
|
|
{ itemstring = "mcl_farming:pumpkin_seeds", weight = 10, amount_min = 2, amount_max = 4 },
|
|
|
|
{ itemstring = "mcl_core:iron_ingot", weight = 10, amount_min = 1, amount_max = 4 },
|
|
|
|
{ itemstring = "bucket:bucket_empty", weight = 10 },
|
|
|
|
{ itemstring = "mcl_core:gold_ingot", weight = 5, amount_min = 1, amount_max = 4 },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
stacks_min = 3,
|
|
|
|
stacks_max = 3,
|
|
|
|
items = {
|
|
|
|
{ itemstring = "mcl_mobitems:bone", weight = 10, amount_min = 1, amount_max = 8 },
|
|
|
|
{ itemstring = "mcl_mobitems:gunpowder", weight = 10, amount_min = 1, amount_max = 8 },
|
|
|
|
{ itemstring = "mcl_mobitems:rotten_flesh", weight = 10, amount_min = 1, amount_max = 8 },
|
|
|
|
{ itemstring = "mcl_mobitems:string", weight = 10, amount_min = 1, amount_max = 8 },
|
|
|
|
},
|
|
|
|
}}, pr)
|
|
|
|
|
|
|
|
return items
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2017-05-24 10:46:40 +03:00
|
|
|
-- Buffer for LuaVoxelManip
|
|
|
|
local lvm_buffer = {}
|
|
|
|
|
|
|
|
-- Below the bedrock, generate air/void
|
|
|
|
minetest.register_on_generated(function(minp, maxp)
|
|
|
|
if maxp.y < mcl_vars.mg_overworld_min or minp.y > mcl_vars.mg_overworld_max then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
local vm, emin, emax = minetest.get_mapgen_object("voxelmanip")
|
|
|
|
local data = vm:get_data(lvm_buffer)
|
|
|
|
local area = VoxelArea:new({MinEdge=emin, MaxEdge=emax})
|
|
|
|
local lvm_used = false
|
|
|
|
|
|
|
|
local c_air = minetest.get_content_id("air")
|
|
|
|
local c_cobble = minetest.get_content_id("mcl_core:cobble")
|
|
|
|
local c_mossycobble = minetest.get_content_id("mcl_core:mossycobble")
|
|
|
|
|
2017-05-24 11:55:25 +03:00
|
|
|
-- Remember chest positions to set metadata later
|
|
|
|
local chest_posses = {}
|
|
|
|
|
2017-05-24 10:46:40 +03:00
|
|
|
-- Calculate the number of dungeon spawn attempts
|
|
|
|
local sizevector = vector.subtract(maxp, minp)
|
|
|
|
sizevector = vector.add(sizevector, 1)
|
|
|
|
local chunksize = sizevector.x * sizevector.y * sizevector.z
|
|
|
|
|
|
|
|
-- In Minecraft, there 8 dungeon spawn attempts Minecraft chunk (16*256*16 = 65536 blocks).
|
|
|
|
-- Minetest chunks don't have this size, so scale the number accordingly.
|
|
|
|
local attempts = math.ceil(chunksize / 65536 * 8)
|
|
|
|
|
|
|
|
for a=1, attempts do
|
|
|
|
local x, y, z
|
|
|
|
local b = 7 -- buffer
|
|
|
|
x = math.random(minp.x+b, maxp.x-b)
|
|
|
|
|
|
|
|
local ymin = math.min(mcl_vars.mg_overworld_max, math.max(minp.y, mcl_vars.mg_bedrock_overworld_max) + 7)
|
|
|
|
local ymax = math.min(mcl_vars.mg_overworld_max, math.max(maxp.y, mcl_vars.mg_bedrock_overworld_max) - 4)
|
|
|
|
|
|
|
|
y = math.random(ymin, ymax)
|
|
|
|
z = math.random(minp.z+b, maxp.z-b)
|
|
|
|
|
|
|
|
local dungeonsizes = {
|
|
|
|
{ x=5, y=4, z=5},
|
|
|
|
{ x=5, y=4, z=7},
|
|
|
|
{ x=7, y=4, z=5},
|
|
|
|
{ x=7, y=4, z=7},
|
|
|
|
}
|
|
|
|
local dim = dungeonsizes[math.random(1, #dungeonsizes)]
|
|
|
|
|
|
|
|
-- Check floor and ceiling: Must be *completely* solid
|
|
|
|
local ceilingfloor_ok = true
|
|
|
|
for tx = x, x+dim.x do
|
|
|
|
for tz = z, z+dim.z do
|
|
|
|
local floor = minetest.get_name_from_content_id(data[area:index(tx, y, tz)])
|
|
|
|
local ceiling = minetest.get_name_from_content_id(data[area:index(tx, y+dim.y+1, tz)])
|
|
|
|
if (not minetest.registered_nodes[floor].walkable) or (not minetest.registered_nodes[ceiling].walkable) then
|
|
|
|
ceilingfloor_ok = false
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if not ceilingfloor_ok then break end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Check for air openings (2 stacked air at ground level) in wall positions
|
|
|
|
local openings = 0
|
|
|
|
if ceilingfloor_ok then
|
|
|
|
|
|
|
|
local walls = {
|
|
|
|
{ x, x+dim.x+1, "x", "z", z },
|
|
|
|
{ x, x+dim.x+1, "x", "z", z+dim.z+1 },
|
|
|
|
{ z+1, z+dim.z, "z", "x", x },
|
|
|
|
{ z+1, z+dim.z, "z", "x", x+dim.x+1 },
|
|
|
|
}
|
|
|
|
|
|
|
|
for w=1, #walls do
|
|
|
|
local wall = walls[w]
|
|
|
|
for iter = wall[1], wall[2] do
|
|
|
|
local pos = {}
|
|
|
|
pos[wall[3]] = iter
|
|
|
|
pos[wall[4]] = wall[5]
|
|
|
|
pos.y = y+1
|
|
|
|
|
|
|
|
local door1 = area:index(pos.x, pos.y, pos.z)
|
|
|
|
pos.y = y+2
|
|
|
|
local door2 = area:index(pos.x, pos.y, pos.z)
|
|
|
|
local doorname1 = minetest.get_name_from_content_id(data[door1])
|
|
|
|
local doorname2 = minetest.get_name_from_content_id(data[door2])
|
|
|
|
if doorname1 == "air" and doorname2 == "air" then
|
|
|
|
openings = openings + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Check conditions. If okay, start generating
|
|
|
|
|
2017-05-24 11:55:25 +03:00
|
|
|
local chestsLeft = 2
|
|
|
|
if ceilingfloor_ok and openings >= 0 and openings <= 5000 then
|
2017-05-24 10:46:40 +03:00
|
|
|
-- Ceiling and floor
|
|
|
|
local maxx, maxy, maxz = x+dim.x+1, y+dim.y+1, z+dim.z+1
|
|
|
|
for tx = x, maxx do
|
|
|
|
for tz = z, maxz do
|
|
|
|
for ty = y, maxy do
|
|
|
|
local p_pos = area:index(tx, ty, tz)
|
|
|
|
|
|
|
|
-- Floor
|
|
|
|
if ty == y then
|
|
|
|
if math.random(1,4) == 1 then
|
|
|
|
data[p_pos] = c_cobble
|
|
|
|
else
|
|
|
|
data[p_pos] = c_mossycobble
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Wall or ceiling
|
|
|
|
elseif ty == maxy or (ty > y and (tx == x or tx == maxx) or (tz == z or tz == maxz)) then
|
|
|
|
data[p_pos] = c_cobble
|
|
|
|
|
|
|
|
-- Room interiour
|
|
|
|
else
|
|
|
|
|
2017-05-24 11:55:25 +03:00
|
|
|
-- Chest
|
|
|
|
if ty == y + 1 and chestsLeft > 0 and math.random(1,6) == 1 then
|
|
|
|
chestsLeft = chestsLeft - 1
|
|
|
|
table.insert(chest_posses, {x=tx, y=ty, z=tz})
|
|
|
|
else
|
|
|
|
data[p_pos] = c_air
|
|
|
|
end
|
2017-05-24 10:46:40 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
lvm_used = true
|
|
|
|
end
|
|
|
|
|
|
|
|
if lvm_used then
|
|
|
|
vm:set_data(data)
|
|
|
|
vm:calc_lighting()
|
|
|
|
vm:update_liquids()
|
|
|
|
vm:write_to_map()
|
|
|
|
end
|
|
|
|
|
2017-05-24 11:55:25 +03:00
|
|
|
for c=1, #chest_posses do
|
|
|
|
minetest.set_node(chest_posses[c], {name="mcl_chests:chest", param2=3})
|
|
|
|
local meta = minetest.get_meta(chest_posses[c])
|
|
|
|
local inv = meta:get_inventory()
|
|
|
|
local items = get_loot()
|
|
|
|
for i=1, math.min(#items, inv:get_size("main")) do
|
|
|
|
inv:set_stack("main", i, ItemStack(items[i]))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-24 10:46:40 +03:00
|
|
|
end)
|