mirror of
https://git.minetest.land/MineClone2/MineClone2.git
synced 2025-01-07 17:17:31 +01:00
cleanups and fixes
This commit is contained in:
parent
732508ee81
commit
6c583a2e7e
@ -4,7 +4,7 @@ local modpath = minetest.get_modpath(modname)
|
|||||||
vl_structures.register_structure("fossil",{
|
vl_structures.register_structure("fossil",{
|
||||||
place_on = {"group:material_stone","group:sand"},
|
place_on = {"group:material_stone","group:sand"},
|
||||||
flags = "place_center_x, place_center_z",
|
flags = "place_center_x, place_center_z",
|
||||||
prepare = { },
|
prepare = false,
|
||||||
chunk_probability = 15, -- was 25, FIXME: needs rebalancing
|
chunk_probability = 15, -- was 25, FIXME: needs rebalancing
|
||||||
y_offset = function(pr) return pr:next(-32,-16) end,
|
y_offset = function(pr) return pr:next(-32,-16) end,
|
||||||
y_max = 15,
|
y_max = 15,
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
local modname = minetest.get_current_modname()
|
local modname = minetest.get_current_modname()
|
||||||
local modpath = minetest.get_modpath(modname)
|
local modpath = minetest.get_modpath(modname)
|
||||||
|
local water_level = minetest.get_mapgen_setting("water_level")
|
||||||
|
|
||||||
local cold_oceans = {
|
local cold_oceans = {
|
||||||
"RoofedForest_ocean",
|
"RoofedForest_ocean",
|
||||||
@ -71,14 +72,15 @@ local warm_oceans = {
|
|||||||
|
|
||||||
local cold = {
|
local cold = {
|
||||||
place_on = {"group:sand","mcl_core:gravel","mcl_core:dirt","mcl_core:clay","group:material_stone"},
|
place_on = {"group:sand","mcl_core:gravel","mcl_core:dirt","mcl_core:clay","group:material_stone"},
|
||||||
spawn_by = {"mcl_core:water_source"},
|
spawn_by = {"group:water"},
|
||||||
num_spawn_by = 2,
|
num_spawn_by = 2,
|
||||||
flags = "place_center_x, place_center_z, force_placement",
|
chunk_probability = 10, -- todo: 15?
|
||||||
y_offset = -1,
|
|
||||||
y_min = mcl_vars.mg_overworld_min,
|
|
||||||
y_max = -2,
|
|
||||||
biomes = cold_oceans,
|
biomes = cold_oceans,
|
||||||
chunk_probability = 10,
|
y_min = mcl_vars.mg_overworld_min,
|
||||||
|
y_max = water_level - 6,
|
||||||
|
y_offset = -1,
|
||||||
|
flags = "place_center_x, place_center_z, force_placement",
|
||||||
|
prepare = { foundation = -2, clear = false, mode="water" },
|
||||||
filenames = {
|
filenames = {
|
||||||
modpath.."/schematics/mcl_structures_ocean_ruins_cold_1.mts",
|
modpath.."/schematics/mcl_structures_ocean_ruins_cold_1.mts",
|
||||||
modpath.."/schematics/mcl_structures_ocean_ruins_cold_2.mts",
|
modpath.."/schematics/mcl_structures_ocean_ruins_cold_2.mts",
|
||||||
|
@ -82,7 +82,7 @@ vl_structures.register_structure("ocean_temple",{
|
|||||||
},
|
},
|
||||||
flags = "force_placement",
|
flags = "force_placement",
|
||||||
force_place = true,
|
force_place = true,
|
||||||
prepare = { tolerance = -1, clear = false, foundation = false },
|
prepare = { tolerance = 8, clear = false, foundation = 3, mode="water" },
|
||||||
biomes = ocean_biomes,
|
biomes = ocean_biomes,
|
||||||
y_max = water_level-4,
|
y_max = water_level-4,
|
||||||
y_min = mcl_vars.mg_overworld_min,
|
y_min = mcl_vars.mg_overworld_min,
|
||||||
@ -90,7 +90,7 @@ vl_structures.register_structure("ocean_temple",{
|
|||||||
modpath .. "/schematics/mcl_structures_ocean_temple.mts",
|
modpath .. "/schematics/mcl_structures_ocean_temple.mts",
|
||||||
modpath .. "/schematics/mcl_structures_ocean_temple_2.mts",
|
modpath .. "/schematics/mcl_structures_ocean_temple_2.mts",
|
||||||
},
|
},
|
||||||
y_offset = function(pr) return pr:next(-2,0) end,
|
y_offset = -1, --function(pr) return pr:next(-2,-1) end, -- fewer mobs if buried in sand
|
||||||
after_place = function(p, _, pr, p1, p2)
|
after_place = function(p, _, pr, p1, p2)
|
||||||
vl_structures.spawn_mobs("mobs_mc:guardian",spawnon,p1,p2,pr,5,true)
|
vl_structures.spawn_mobs("mobs_mc:guardian",spawnon,p1,p2,pr,5,true)
|
||||||
vl_structures.spawn_mobs("mobs_mc:guardian_elder",spawnon,p1,p2,pr,1,true)
|
vl_structures.spawn_mobs("mobs_mc:guardian_elder",spawnon,p1,p2,pr,1,true)
|
||||||
|
@ -6,7 +6,7 @@ local spawnon = {"mcl_core:stripped_oak","mcl_stairs:slab_birchwood_top"}
|
|||||||
vl_structures.register_structure("pillager_outpost",{
|
vl_structures.register_structure("pillager_outpost",{
|
||||||
place_on = {"group:grass_block","group:dirt","mcl_core:dirt_with_grass","group:sand"},
|
place_on = {"group:grass_block","group:dirt","mcl_core:dirt_with_grass","group:sand"},
|
||||||
flags = "place_center_x, place_center_z",
|
flags = "place_center_x, place_center_z",
|
||||||
prepare = { padding = 3, corners = 4, foundation = -6, clear = true },
|
prepare = { padding = 3, corners = 4, foundation = -8, clear = true },
|
||||||
y_offset = 0,
|
y_offset = 0,
|
||||||
chunk_probability = 15,
|
chunk_probability = 15,
|
||||||
y_max = mcl_vars.mg_overworld_max,
|
y_max = mcl_vars.mg_overworld_max,
|
||||||
|
@ -2,15 +2,6 @@ local modname = minetest.get_current_modname()
|
|||||||
local modpath = minetest.get_modpath(modname)
|
local modpath = minetest.get_modpath(modname)
|
||||||
local water_level = minetest.get_mapgen_setting("water_level")
|
local water_level = minetest.get_mapgen_setting("water_level")
|
||||||
|
|
||||||
--schematics by chmodsayshello
|
|
||||||
local schems = {
|
|
||||||
modpath.."/schematics/mcl_structures_shipwreck_full_damaged.mts",
|
|
||||||
modpath.."/schematics/mcl_structures_shipwreck_full_normal.mts",
|
|
||||||
modpath.."/schematics/mcl_structures_shipwreck_full_back_damaged.mts",
|
|
||||||
modpath.."/schematics/mcl_structures_shipwreck_half_front.mts",
|
|
||||||
modpath.."/schematics/mcl_structures_shipwreck_half_back.mts",
|
|
||||||
}
|
|
||||||
|
|
||||||
local ocean_biomes = {
|
local ocean_biomes = {
|
||||||
"RoofedForest_ocean",
|
"RoofedForest_ocean",
|
||||||
"JungleEdgeM_ocean",
|
"JungleEdgeM_ocean",
|
||||||
@ -74,44 +65,27 @@ local ocean_biomes = {
|
|||||||
"JungleM_ocean"
|
"JungleM_ocean"
|
||||||
}
|
}
|
||||||
|
|
||||||
local beach_biomes = {
|
|
||||||
"FlowerForest_beach",
|
|
||||||
"Forest_beach",
|
|
||||||
"StoneBeach",
|
|
||||||
"ColdTaiga_beach_water",
|
|
||||||
"Taiga_beach",
|
|
||||||
"Savanna_beach",
|
|
||||||
"Plains_beach",
|
|
||||||
"ExtremeHills_beach",
|
|
||||||
"ColdTaiga_beach",
|
|
||||||
"Swampland_shore",
|
|
||||||
"MushroomIslandShore",
|
|
||||||
"JungleM_shore",
|
|
||||||
"Jungle_shore"
|
|
||||||
}
|
|
||||||
|
|
||||||
-- FIXME: integrate treasure maps from MCLA
|
-- FIXME: integrate treasure maps from MCLA
|
||||||
|
|
||||||
vl_structures.register_structure("shipwreck",{
|
vl_structures.register_structure("shipwreck",{
|
||||||
place_on = {"group:sand","mcl_core:gravel"},
|
place_on = {"group:sand","mcl_core:gravel"},
|
||||||
spawn_by = {"group:water"},
|
spawn_by = {"group:water"},
|
||||||
num_spawn_by = 4,
|
num_spawn_by = 4,
|
||||||
noise_params = {
|
chunk_probability = 10, -- todo: 15?
|
||||||
offset = 0,
|
|
||||||
scale = 0.000022,
|
|
||||||
spread = {x = 250, y = 250, z = 250},
|
|
||||||
seed = 3,
|
|
||||||
octaves = 3,
|
|
||||||
persist = 0.001,
|
|
||||||
flags = "absvalue",
|
|
||||||
},
|
|
||||||
flags = "force_placement",
|
|
||||||
biomes = ocean_biomes,
|
biomes = ocean_biomes,
|
||||||
y_max = water_level-4,
|
|
||||||
y_min = mcl_vars.mg_overworld_min,
|
y_min = mcl_vars.mg_overworld_min,
|
||||||
prepare = { tolerance = -1, clear = false, foundation = false },
|
y_max = water_level-4,
|
||||||
filenames = schems,
|
|
||||||
y_offset = function(pr) return pr:next(-4,-2) end,
|
y_offset = function(pr) return pr:next(-4,-2) end,
|
||||||
|
flags = "place_center_x, place_center_z, force_placement",
|
||||||
|
prepare = { tolerance = -1, clear = false, foundation = -2, mode = "water" },
|
||||||
|
filenames = {
|
||||||
|
--schematics by chmodsayshello
|
||||||
|
modpath.."/schematics/mcl_structures_shipwreck_full_damaged.mts",
|
||||||
|
modpath.."/schematics/mcl_structures_shipwreck_full_normal.mts",
|
||||||
|
modpath.."/schematics/mcl_structures_shipwreck_full_back_damaged.mts",
|
||||||
|
modpath.."/schematics/mcl_structures_shipwreck_half_front.mts",
|
||||||
|
modpath.."/schematics/mcl_structures_shipwreck_half_back.mts",
|
||||||
|
},
|
||||||
loot = {
|
loot = {
|
||||||
["mcl_chests:chest_small"] = {
|
["mcl_chests:chest_small"] = {
|
||||||
{
|
{
|
||||||
|
@ -251,7 +251,7 @@ function mcl_villages.post_process_village(blockseed)
|
|||||||
local jobs, beds = {}, {}
|
local jobs, beds = {}, {}
|
||||||
|
|
||||||
local bell_pos = vector.copy(settlement_info[1].pos)
|
local bell_pos = vector.copy(settlement_info[1].pos)
|
||||||
local bell = vector.offset(bell_pos, 0, 1, 0)
|
local bell = vector.offset(bell_pos, 0, 2, 0)
|
||||||
local biome_name = minetest.get_biome_name(minetest.get_biome_data(bell_pos).biome)
|
local biome_name = minetest.get_biome_name(minetest.get_biome_data(bell_pos).biome)
|
||||||
|
|
||||||
-- Spawn Golem
|
-- Spawn Golem
|
||||||
|
@ -1,37 +1,107 @@
|
|||||||
# vl_structures
|
# vl_structures
|
||||||
|
|
||||||
Updated API for structure spawning for VoxeLibre and Mineclonia
|
Updated API for structure spawning.
|
||||||
|
|
||||||
## vl_structures.register_structure(name,structure definition,nospawn)
|
This module was developed with VoxeLibre and Mineclonia in mind, but means to be portable or at least easy to adapt to other games.
|
||||||
If nospawn is truthy the structure will not be placed by mapgen and the decoration parameters can be omitted. This is intended for secondary structures the placement of which gets triggered by the placement of other structures. It can also be used to register testing structures so they can be used with /spawnstruct.
|
|
||||||
|
|
||||||
### structure definition
|
## structure definition
|
||||||
|
|
||||||
|
Structures in this API are defined using the following table:
|
||||||
|
|
||||||
|
```
|
||||||
{
|
{
|
||||||
fill_ratio = OR noise = {},
|
name =, -- structure identifier for logging
|
||||||
biomes = {},
|
priority = 100, -- priority to make placement order more deterministic. Default 100 except for terrain features (900)
|
||||||
y_min =,
|
chunk_probability =, -- ratio that a block is chosen, 10 means 1-in-10 blocks
|
||||||
y_max =,
|
fill_ratio = nil, -- OR number of structure spawn attempts per map chunk, default is 1/(80 x 80) when chunk_probability is set
|
||||||
place_on = {},
|
noise = nil, -- OR specify noise parameters, as per minetest.register_decoration
|
||||||
spawn_by = {},
|
y_min =, -- minimum depth
|
||||||
num_spawn_by =,
|
y_max =, -- maximum depth
|
||||||
flags = (default: "place_center_x, place_center_z, force_placement")
|
biomes = {}, -- biome restriction
|
||||||
(same as decoration def)
|
place_on = {}, -- if nil, the structure will not be automatically spawned
|
||||||
y_offset =, --can be a number or a function returning a number
|
spawn_by = {}, -- nodes required nearby, as in minetest.register_decoration
|
||||||
filenames = {} OR place_func = function(pos,def,pr)
|
num_spawn_by =, -- number of nodes required nearby
|
||||||
-- filenames can be a list of any schematics accepted by mcl_structures.place_schematic / minetest.place_schematic
|
prepare =, -- configure foundation and clearing, see vl_terraforming -- ignored for place_func
|
||||||
on_place = function(pos,def,pr) end,
|
flags =, -- minetest.register_decoration placement flags, default: "place_center_x, place_center_z"
|
||||||
-- called before placement. denies placement when returning falsy.
|
y_offset =, -- vertical placement offset, can be a number or a function(pr) returning a number
|
||||||
after_place = function(pos,def,pr)
|
filenames = {}, -- table of schematic filenames
|
||||||
-- executed after successful placement
|
schematics = {}, -- OR table of preloaded schematics
|
||||||
prepare = table, -- a foundation is automatically built for the structure
|
place_func = function(pos,def,pr,blockseed)
|
||||||
loot = ,
|
-- OR a function to place a structure
|
||||||
--a table of loot tables for mcl_loot indexed by node names
|
after_place = function(pos,def,pr,pmin,pmax,size,rotation)
|
||||||
-- e.g. { ["mcl_chests:chest_small"] = {loot},... }
|
-- callback executed after successful placement
|
||||||
|
loot =, -- a table of loot tables for mcl_loot indexed by node names -- ignored for place_func, to be removed
|
||||||
|
-- e.g. { ["mcl_chests:chest_small"] = {loot},... }
|
||||||
|
terrain_feature =, -- affects placement priority and disables logging for uninteresting structures
|
||||||
|
daughters =, -- substructures to spawn, unstable API
|
||||||
}
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## vl_structures.register_structure(name,def)
|
||||||
|
|
||||||
|
Register a new structure spawn.
|
||||||
|
|
||||||
|
For extension modules, if you choose a larger (later) placement priority, this
|
||||||
|
should be less likely to change spawning of original structures and keep the
|
||||||
|
resulting maps more consistent across seeds.
|
||||||
|
|
||||||
|
## vl_structures.load_schematic(filename, name)
|
||||||
|
|
||||||
|
Load a schematic from a given file name, the name is used for error logging; otherwise it will be derived from the file name.
|
||||||
|
|
||||||
|
## vl_structures.place_structure(pos, def, pr, blockseed, rot)
|
||||||
|
|
||||||
|
Places structure defined by def at position pos, using the pseudorandom pr.
|
||||||
|
|
||||||
|
blockseed is only used by the place_func call, and unused for many simple structures.
|
||||||
|
|
||||||
|
rot is optional, it will then be chosen randomly.
|
||||||
|
|
||||||
|
This is usually called from the mapgen decoration gennotify mechanism, but can be used for substructure spawns.
|
||||||
|
|
||||||
## vl_structures.registered_structures
|
## vl_structures.registered_structures
|
||||||
|
|
||||||
Table of the registered structure defintions indexed by name.
|
Table of the registered structure defintions indexed by name.
|
||||||
|
|
||||||
## vl_structures.place_structure(pos, def, pr)
|
## vl_structures.place_schematic(pos, yoffset, schematic, rotation, def, pr)
|
||||||
Places a structure using the mapgen placement function
|
|
||||||
|
Spawn a structure as defined by "def" at the given position, yoffset, schematic, and rotation.
|
||||||
|
|
||||||
|
This is primarily meant for substructure placement where size (and hence schematic and offsets) need to be fixed before computing the position.
|
||||||
|
|
||||||
|
## vl_structures.parse_rotation(rotation, pr)
|
||||||
|
|
||||||
|
Parse a rotation value (stirngs "0", "90", "180", "270" or "random"), or choosing a random rotation.
|
||||||
|
|
||||||
|
## vl_structures.size_rotated(size, rotation)
|
||||||
|
|
||||||
|
Return the size after rotation, i.e., if rotation is 90 or 270, the x and z sizes are swapped.
|
||||||
|
|
||||||
|
## vl_structures.top_left_from_flags(pos, size, flags)
|
||||||
|
|
||||||
|
Compute the top left corner from the flags, i.e., parse place_center_x, place_center_z etc.
|
||||||
|
|
||||||
|
## vl_structures.get_extends(pos, size, yoffset, rotation, flags)
|
||||||
|
|
||||||
|
Parse rotation and flags, and return the center, minimum corner, maximum corner, and size.
|
||||||
|
|
||||||
|
## vl_structures.init_node_construct(pos)
|
||||||
|
|
||||||
|
Call on_construct callbacks for the node at the given position.
|
||||||
|
|
||||||
|
## vl_structures.construct_nodes(p1,p2,nodes)
|
||||||
|
|
||||||
|
Find all nodes of the listed types in the area and call their on_construct callbacks.
|
||||||
|
|
||||||
|
## vl_structures.fill_chests(p1,p2,loot,pr)
|
||||||
|
|
||||||
|
Fill all loot containers in the area, requires mcl_loot and likely should be moved to the loot API.
|
||||||
|
|
||||||
|
## vl_structures.spawn_mobs(mob,spawnon,p1,p2,pr,n,water)
|
||||||
|
|
||||||
|
This function spawns the desired mobs in the given area. The function should move to the mobs API.
|
||||||
|
|
||||||
|
## vl_structures.register_structure_spawn(def)
|
||||||
|
|
||||||
|
This function creates a spawn ABM for the desired mobs. The function should move to the mobs API.
|
||||||
|
|
||||||
## vl_structures.place_schematic(pos, schematic, rotation, replacements, force_placement, flags, after_placement_callback, pr, callback_param)
|
|
||||||
|
@ -1,21 +1,17 @@
|
|||||||
vl_structures.registered_structures = {}
|
vl_structures.registered_structures = {}
|
||||||
|
|
||||||
local structure_boost = tonumber(minetest.settings:get("vl_structures_boost")) or 1
|
local structure_boost = tonumber(minetest.settings:get("vl_structures_boost")) or 1
|
||||||
local worldseed = minetest.get_mapgen_setting("seed")
|
local logging = minetest.settings:get_bool("vl_structures_logging", false)
|
||||||
local RANDOM_SEED_OFFSET = 959 -- random constant that should be unique across each library
|
local disabled_structures = minetest.settings:get("vl_structures_disabled")
|
||||||
|
|
||||||
local vector_offset = vector.offset
|
|
||||||
|
|
||||||
-- FIXME: switch to vl_structures_logging?
|
|
||||||
local logging = true or minetest.settings:get_bool("mcl_logging_structures", true)
|
|
||||||
|
|
||||||
-- FIXME: switch to vl_structures_disabled?
|
|
||||||
local disabled_structures = minetest.settings:get("mcl_disabled_structures")
|
|
||||||
disabled_structures = disabled_structures and disabled_structures:split(",") or {}
|
disabled_structures = disabled_structures and disabled_structures:split(",") or {}
|
||||||
function vl_structures.is_disabled(structname)
|
function vl_structures.is_disabled(structname)
|
||||||
return table.indexof(disabled_structures,structname) ~= -1
|
return table.indexof(disabled_structures,structname) ~= -1
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local worldseed = minetest.get_mapgen_setting("seed")
|
||||||
|
local RANDOM_SEED_OFFSET = 959 -- random constant that should be unique across each library
|
||||||
|
local vector_offset = vector.offset
|
||||||
|
|
||||||
--- Trim a full path name to its last two parts as short name for logging
|
--- Trim a full path name to its last two parts as short name for logging
|
||||||
local function basename(filename)
|
local function basename(filename)
|
||||||
local fn = string.split(filename, "/")
|
local fn = string.split(filename, "/")
|
||||||
@ -101,30 +97,28 @@ function vl_structures.place_structure(pos, def, pr, blockseed, rot)
|
|||||||
end
|
end
|
||||||
-- structure has a custom place function
|
-- structure has a custom place function
|
||||||
if not def.place_func then
|
if not def.place_func then
|
||||||
minetest.log("warning","[vl_structures] no schematics and no place_func for schematic "..def.name)
|
minetest.log("warning", "[vl_structures] no schematics and no place_func for schematic "..def.name)
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
local pp = yoffset ~= 0 and vector_offset(pos, 0, yoffset, 0) or pos
|
local pp = yoffset ~= 0 and vector_offset(pos, 0, yoffset, 0) or pos
|
||||||
if def.place_func and def.prepare then
|
if def.place_func and def.prepare then
|
||||||
minetest.log("warning", "[vl_structures] needed prepare for "..def.name.." placed at "..minetest.pos_to_string(pp).." but do not have size information")
|
minetest.log("warning", "[vl_structures] needed prepare for "..def.name.." placed at "..minetest.pos_to_string(pp).." but do not have size information.")
|
||||||
end
|
end
|
||||||
if def.place_func and def.place_func(pp,def,pr,blockseed) then
|
if def.place_func and def.place_func(pp,def,pr,blockseed) then
|
||||||
if not def.after_place or (def.after_place and def.after_place(pos,def,pr,pmin,pmax,size,param.rotation)) then
|
if def.after_place and not def.after_place(pos,def,pr,pmin,pmax,size,param.rotation) then
|
||||||
if def.sidelen then
|
minetest.log("warning", "[vl_structures] after_place failed for structure "..def.name)
|
||||||
local p1, p2 = vector_offset(pos,-def.sidelen,-def.sidelen,-def.sidelen), vector.offset(pos,def.sidelen,def.sidelen,def.sidelen)
|
|
||||||
if def.loot then vl_structures.fill_chests(p1,p2,def.loot,pr) end
|
|
||||||
if def.construct_nodes then vl_structures.construct_nodes(p1,p2,def.construct_nodes) end
|
|
||||||
end
|
|
||||||
if log_enabled then
|
|
||||||
minetest.log("action","[vl_structures] "..def.name.." placed at "..minetest.pos_to_string(pp))
|
|
||||||
end
|
|
||||||
return true
|
|
||||||
else
|
|
||||||
minetest.log("warning","[vl_structures] after_place failed for schematic "..def.name)
|
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
|
if log_enabled then
|
||||||
|
minetest.log("action","[vl_structures] "..def.name.." placed at "..minetest.pos_to_string(pp))
|
||||||
|
end
|
||||||
|
return true
|
||||||
elseif log_enabled then
|
elseif log_enabled then
|
||||||
minetest.log("warning","[vl_structures] place_func failed for schematic "..def.name)
|
if def.place_func then
|
||||||
|
minetest.log("warning","[vl_structures] place_func failed for structure "..def.name)
|
||||||
|
else
|
||||||
|
minetest.log("warning","[vl_structures] do not know how to place structure "..def.name)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -139,13 +133,13 @@ function vl_structures.register_structure(name,def)
|
|||||||
def.name = name
|
def.name = name
|
||||||
vl_structures.registered_structures[name] = def
|
vl_structures.registered_structures[name] = def
|
||||||
if def.prepare and def.prepare.clear == nil and (def.prepare.clear_bottom or def.prepare.clear_top) then def.prepare.clear = true end
|
if def.prepare and def.prepare.clear == nil and (def.prepare.clear_bottom or def.prepare.clear_top) then def.prepare.clear = true end
|
||||||
if not def.noise_params and def.chunk_probability and not def.fill_ratio then
|
if not def.fill_ratio and def.chunk_probability and not def.noise_params then
|
||||||
def.fill_ratio = 1.1/80/80 -- 1 per chunk, controlled by chunk probability only
|
def.fill_ratio = 1.1/80/80 -- 1 per chunk, controlled by chunk probability only
|
||||||
end
|
end
|
||||||
def.flags = def.flags or vl_structures.DEFAULT_FLAGS
|
def.flags = def.flags or vl_structures.DEFAULT_FLAGS
|
||||||
if def.filenames then
|
if def.filenames then
|
||||||
for _, filename in ipairs(def.filenames) do
|
for _, filename in ipairs(def.filenames) do
|
||||||
if not mcl_util.file_exists(filename) then
|
if mcl_util and not mcl_util.file_exists(filename) then
|
||||||
minetest.log("warning","[vl_structures] schematic "..(name or "unknown").." is missing file "..basename(filename))
|
minetest.log("warning","[vl_structures] schematic "..(name or "unknown").." is missing file "..basename(filename))
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -153,21 +147,22 @@ function vl_structures.register_structure(name,def)
|
|||||||
end
|
end
|
||||||
if def.place_on then
|
if def.place_on then
|
||||||
minetest.register_on_mods_loaded(function()
|
minetest.register_on_mods_loaded(function()
|
||||||
def.deco = mcl_mapgen_core.register_decoration({
|
local register_decoration = mcl_mapgen_core.register_decoration or minetest.register_decoration -- optional dependency
|
||||||
|
register_decoration({
|
||||||
name = "vl_structures:"..name,
|
name = "vl_structures:"..name,
|
||||||
rank = def.rank or (def.terrain_feature and 900) or 100, -- run before regular decorations
|
rank = def.rank or (def.terrain_feature and 900) or 100, -- run before regular decorations
|
||||||
deco_type = "schematic",
|
fill_ratio = def.fill_ratio,
|
||||||
schematic = EMPTY_SCHEMATIC, -- use gennotify only
|
noise_params = def.noise_params,
|
||||||
|
y_max = def.y_max,
|
||||||
|
y_min = def.y_min,
|
||||||
|
biomes = def.biomes,
|
||||||
place_on = def.place_on,
|
place_on = def.place_on,
|
||||||
spawn_by = def.spawn_by,
|
spawn_by = def.spawn_by,
|
||||||
num_spawn_by = def.num_spawn_by,
|
num_spawn_by = def.num_spawn_by,
|
||||||
sidelen = 80, -- no def.sidelen subdivisions for now, this field was used differently before
|
sidelen = 80, -- no def.sidelen subdivisions for now, this field was used differently before
|
||||||
fill_ratio = def.fill_ratio,
|
|
||||||
noise_params = def.noise_params,
|
|
||||||
flags = def.flags,
|
flags = def.flags,
|
||||||
biomes = def.biomes,
|
deco_type = "schematic",
|
||||||
y_max = def.y_max,
|
schematic = EMPTY_SCHEMATIC, -- use gennotify only
|
||||||
y_min = def.y_min,
|
|
||||||
gen_callback = function(t,minp,maxp,blockseed)
|
gen_callback = function(t,minp,maxp,blockseed)
|
||||||
for _, pos in ipairs(t) do
|
for _, pos in ipairs(t) do
|
||||||
local pr = PcgRandom(minetest.hash_node_position(pos) + worldseed + RANDOM_SEED_OFFSET)
|
local pr = PcgRandom(minetest.hash_node_position(pos) + worldseed + RANDOM_SEED_OFFSET)
|
||||||
@ -183,6 +178,7 @@ function vl_structures.register_structure(name,def)
|
|||||||
end
|
end
|
||||||
|
|
||||||
-- To avoid a cyclic dependency, run this when modules have finished loading
|
-- To avoid a cyclic dependency, run this when modules have finished loading
|
||||||
|
-- Maybe we can eventually remove this - the end portal should likely go into the mapgen itself.
|
||||||
minetest.register_on_mods_loaded(function()
|
minetest.register_on_mods_loaded(function()
|
||||||
mcl_mapgen_core.register_generator("static structures", nil, function(minp, maxp, blockseed)
|
mcl_mapgen_core.register_generator("static structures", nil, function(minp, maxp, blockseed)
|
||||||
for _,struct in pairs(vl_structures.registered_structures) do
|
for _,struct in pairs(vl_structures.registered_structures) do
|
||||||
@ -200,37 +196,3 @@ mcl_mapgen_core.register_generator("static structures", nil, function(minp, maxp
|
|||||||
end, 100, true)
|
end, 100, true)
|
||||||
end)
|
end)
|
||||||
|
|
||||||
local structure_spawns = {}
|
|
||||||
function vl_structures.register_structure_spawn(def)
|
|
||||||
--name,y_min,y_max,spawnon,biomes,chance,interval,limit
|
|
||||||
minetest.register_abm({
|
|
||||||
label = "Spawn "..def.name,
|
|
||||||
nodenames = def.spawnon,
|
|
||||||
min_y = def.y_min or -31000,
|
|
||||||
max_y = def.y_max or 31000,
|
|
||||||
interval = def.interval or 60,
|
|
||||||
chance = def.chance or 5,
|
|
||||||
action = function(pos, node, active_object_count, active_object_count_wider)
|
|
||||||
local limit = def.limit or 7
|
|
||||||
if active_object_count_wider > limit + mob_cap_animal then return end
|
|
||||||
if active_object_count_wider > mob_cap_player then return end
|
|
||||||
local p = vector_offset(pos, 0, 1, 0)
|
|
||||||
local pname = minetest.get_node(p).name
|
|
||||||
if def.type_of_spawning == "water" then
|
|
||||||
if pname ~= "mcl_core:water_source" and pname ~= "mclx_core:river_water_source" then return end
|
|
||||||
else
|
|
||||||
if pname ~= "air" then return end
|
|
||||||
end
|
|
||||||
if minetest.get_meta(pos):get_string("spawnblock") == "" then return end
|
|
||||||
if mg_name ~= "v6" and mg_name ~= "singlenode" and def.biomes then
|
|
||||||
if table.indexof(def.biomes, minetest.get_biome_name(minetest.get_biome_data(p).biome)) == -1 then
|
|
||||||
return
|
|
||||||
end
|
|
||||||
end
|
|
||||||
local mobdef = minetest.registered_entities[def.name]
|
|
||||||
if mobdef.can_spawn and not mobdef.can_spawn(p) then return end
|
|
||||||
minetest.add_entity(p, def.name)
|
|
||||||
end,
|
|
||||||
})
|
|
||||||
end
|
|
||||||
|
|
||||||
|
@ -1,22 +1,14 @@
|
|||||||
local DEFAULT_FLAGS = vl_structures.DEFAULT_FLAGS
|
|
||||||
local DEFAULT_PREPARE = vl_structures.DEFAULT_PREPARE
|
|
||||||
|
|
||||||
local vector_offset = vector.offset
|
local vector_offset = vector.offset
|
||||||
local floor = math.floor
|
local floor = math.floor
|
||||||
|
|
||||||
-- FIXME: switch to vl_structures_logging?
|
local logging = minetest.settings:get_bool("vl_structures_logging", false)
|
||||||
local logging = true or minetest.settings:get_bool("mcl_logging_structures", true)
|
|
||||||
|
|
||||||
local mg_name = minetest.get_mapgen_setting("mg_name")
|
local mg_name = minetest.get_mapgen_setting("mg_name")
|
||||||
|
|
||||||
-- parse the prepare parameter
|
-- parse the prepare parameter
|
||||||
local function parse_prepare(prepare)
|
local function parse_prepare(prepare)
|
||||||
if prepare == nil or prepare == true then return DEFAULT_PREPARE end
|
if prepare == nil or prepare == true then return vl_structures.DEFAULT_PREPARE end
|
||||||
if prepare == false then return {} end
|
if prepare == false then return {} end
|
||||||
if prepare.foundation == true then
|
if prepare.foundation == true then prepare.foundation = vl_structures.DEFAULT_PREPARE.foundation end
|
||||||
prepare = table.copy(prepare)
|
|
||||||
prepare.foundation = DEFAULT_PREPARE.foundation
|
|
||||||
end
|
|
||||||
return prepare
|
return prepare
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -68,17 +60,17 @@ local function emerge_schematics(blockpos, action, calls_remaining, param)
|
|||||||
local prepare_start = os.clock()
|
local prepare_start = os.clock()
|
||||||
-- Get materials from biome (TODO: make this a function + table?):
|
-- Get materials from biome (TODO: make this a function + table?):
|
||||||
local b = mg_name ~= "v6" and minetest.registered_biomes[minetest.get_biome_name(minetest.get_biome_data(pos).biome)]
|
local b = mg_name ~= "v6" and minetest.registered_biomes[minetest.get_biome_name(minetest.get_biome_data(pos).biome)]
|
||||||
local node_top = b and b.node_top and { name = b.node_top } or surface_mat or { name = "mcl_core:dirt_with_grass" }
|
local node_top = b and b.node_top and { name = b.node_top } or surface_mat or vl_structures.DEFAULT_SURFACE
|
||||||
local node_filler = { name = b and b.node_filler or "mcl_core:dirt" }
|
local node_filler = b and b.node_filler and { name = b.node_filler } or vl_structures.DEFAULT_FILLER
|
||||||
local node_stone = { name = b and b.node_stone or "mcl_core:stone" }
|
local node_stone = b and b.node_stone and { name = b.node_stone } or vl_structures.DEFAULT_STONE
|
||||||
local node_dust = b and b.node_dust and { name = b.node_dust } or nil
|
local node_dust = b and b.node_dust and { name = b.node_dust } or vl_structures.DEFAULT_DUST
|
||||||
if node_top.name == "mcl_core:dirt_with_grass" and b then node_top.param2 = b._mcl_grass_palette_index end
|
if node_top.name == "mcl_core:dirt_with_grass" and b then node_top.param2 = b._mcl_grass_palette_index end
|
||||||
|
|
||||||
-- Step 2a: clear overhead area
|
-- Step 2a: clear overhead area
|
||||||
local corners, padding, depth = prepare.corners or 1, prepare.padding or 1, (type(prepare.foundation) == "number" and prepare.foundation) or -4
|
local corners, padding = prepare.corners or 1, prepare.padding or 1
|
||||||
local gp = vector_offset(pmin, -padding, -yoffset, -padding) -- base level
|
local gp = vector_offset(pmin, -padding, -yoffset, -padding) -- base level
|
||||||
if prepare.clear then
|
if prepare.clear then
|
||||||
local yoff, ymax = prepare.clear_bottom or 0, size.y + yoffset + (prepare.clear_top or DEFAULT_PREPARE.clear_top)
|
local yoff, ymax = prepare.clear_bottom or 0, size.y + yoffset + (prepare.clear_top or vl_structures.DEFAULT_PREPARE.clear_top)
|
||||||
if prepare.clear_bottom == "top" or prepare.clear_bottom == "above" then yoff = size.y + yoffset end
|
if prepare.clear_bottom == "top" or prepare.clear_bottom == "above" then yoff = size.y + yoffset end
|
||||||
--minetest.log("action", "[vl_structures] clearing air "..minetest.pos_to_string(gp)..": ".. (size.x + padding * 2)..","..ymax..","..(size.z + padding * 2))
|
--minetest.log("action", "[vl_structures] clearing air "..minetest.pos_to_string(gp)..": ".. (size.x + padding * 2)..","..ymax..","..(size.z + padding * 2))
|
||||||
vl_terraforming.clearance_vm(vm, gp.x, gp.y + yoff, gp.z,
|
vl_terraforming.clearance_vm(vm, gp.x, gp.y + yoff, gp.z,
|
||||||
@ -91,7 +83,7 @@ local function emerge_schematics(blockpos, action, calls_remaining, param)
|
|||||||
if ddp and ddp.clear then
|
if ddp and ddp.clear then
|
||||||
local dsize = vl_structures.size_rotated(ds.size, dr) -- FIXME: rotation of parent
|
local dsize = vl_structures.size_rotated(ds.size, dr) -- FIXME: rotation of parent
|
||||||
local corners, padding, yoffset = ddp.corners or 1, ddp.padding or 1, ddp.yoffset or 0
|
local corners, padding, yoffset = ddp.corners or 1, ddp.padding or 1, ddp.yoffset or 0
|
||||||
local yoff, ymax = ddp.clear_bottom or 0, dsize.y + yoffset + (ddp.clear_top or DEFAULT_PREPARE.clear_top)
|
local yoff, ymax = ddp.clear_bottom or 0, dsize.y + yoffset + (ddp.clear_top or vl_structures.DEFAULT_PREPARE.clear_top)
|
||||||
if ddp.clear_bottom == "top" or ddp.clear_bottom == "above" then yoff = dsize.y + yoffset end
|
if ddp.clear_bottom == "top" or ddp.clear_bottom == "above" then yoff = dsize.y + yoffset end
|
||||||
local gp = vector_offset(pos, dd.pos.x - floor((dsize.x-1)*0.5) - padding,
|
local gp = vector_offset(pos, dd.pos.x - floor((dsize.x-1)*0.5) - padding,
|
||||||
dd.pos.y,
|
dd.pos.y,
|
||||||
@ -110,7 +102,7 @@ local function emerge_schematics(blockpos, action, calls_remaining, param)
|
|||||||
-- Step 2b: baseplate underneath
|
-- Step 2b: baseplate underneath
|
||||||
if prepare.foundation then
|
if prepare.foundation then
|
||||||
-- minetest.log("action", "[vl_structures] fill foundation "..minetest.pos_to_string(gp).." with "..tostring(node_top.name).." "..tostring(node_filler.name))
|
-- minetest.log("action", "[vl_structures] fill foundation "..minetest.pos_to_string(gp).." with "..tostring(node_top.name).." "..tostring(node_filler.name))
|
||||||
local depth = (type(prepare.foundation) == "number" and prepare.foundation) or DEFAULT_PREPARE.foundation
|
local depth = (type(prepare.foundation) == "number" and prepare.foundation) or vl_structures.DEFAULT_PREPARE.foundation
|
||||||
vl_terraforming.foundation_vm(vm, gp.x, gp.y - 1, gp.z,
|
vl_terraforming.foundation_vm(vm, gp.x, gp.y - 1, gp.z,
|
||||||
size.x + padding * 2, depth, size.z + padding * 2,
|
size.x + padding * 2, depth, size.z + padding * 2,
|
||||||
corners, node_top, node_filler, node_stone, node_dust, pr)
|
corners, node_top, node_filler, node_stone, node_dust, pr)
|
||||||
@ -121,7 +113,7 @@ local function emerge_schematics(blockpos, action, calls_remaining, param)
|
|||||||
if ddp and ddp.foundation then
|
if ddp and ddp.foundation then
|
||||||
local dsize = vl_structures.size_rotated(ds.size, dr) -- FIXME: rotation of parent
|
local dsize = vl_structures.size_rotated(ds.size, dr) -- FIXME: rotation of parent
|
||||||
local corners, padding, yoffset = ddp.corners or 1, ddp.padding or 1, ddp.yoffset or 0
|
local corners, padding, yoffset = ddp.corners or 1, ddp.padding or 1, ddp.yoffset or 0
|
||||||
local depth = (type(ddp.foundation) == "number" and ddp.foundation) or DEFAULT_PREPARE.foundation
|
local depth = (type(ddp.foundation) == "number" and ddp.foundation) or vl_structures.DEFAULT_PREPARE.foundation
|
||||||
local gp = vector_offset(pos, dd.pos.x - floor((dsize.x-1)*0.5) - padding,
|
local gp = vector_offset(pos, dd.pos.x - floor((dsize.x-1)*0.5) - padding,
|
||||||
dd.pos.y + (yoffset or 0),
|
dd.pos.y + (yoffset or 0),
|
||||||
dd.pos.z - floor((dsize.z-1)*0.5) - padding)
|
dd.pos.z - floor((dsize.z-1)*0.5) - padding)
|
||||||
@ -158,16 +150,16 @@ vl_structures.place_schematic = function(pos, yoffset, schematic, rotation, def,
|
|||||||
if schematic and not schematic.size then schematic = vl_structures.load_schematic(schematic) end -- legacy
|
if schematic and not schematic.size then schematic = vl_structures.load_schematic(schematic) end -- legacy
|
||||||
local rotation = vl_structures.parse_rotation(rotation, pr)
|
local rotation = vl_structures.parse_rotation(rotation, pr)
|
||||||
local prepare = parse_prepare(def.prepare)
|
local prepare = parse_prepare(def.prepare)
|
||||||
local ppos, pmin, pmax, size = vl_structures.get_extends(pos, schematic.size, yoffset, rotation, def.flags or DEFAULT_FLAGS)
|
local ppos, pmin, pmax, size = vl_structures.get_extends(pos, schematic.size, yoffset, rotation, def.flags or vl_structures.DEFAULT_FLAGS)
|
||||||
-- area to emerge. Add some margin to allow for finding better suitable ground etc.
|
-- area to emerge. Add some margin to allow for finding better suitable ground etc.
|
||||||
local tolerance = prepare.tolerance or DEFAULT_PREPARE.tolerance -- may be negative to disable foundations
|
local tolerance = prepare.tolerance or vl_structures.DEFAULT_PREPARE.tolerance -- may be negative to disable foundations
|
||||||
if type(tolerance) ~= "number" then tolerance = 10 end -- extra height for emerge only, min/max/liquid_surface
|
if type(tolerance) ~= "number" then tolerance = 10 end -- extra height for emerge only, min/max/liquid_surface
|
||||||
local emin, emax = vector_offset(pmin, 0, -math.max(tolerance, 0), 0), vector.offset(pmax, 0, math.max(tolerance, 0), 0)
|
local emin, emax = vector_offset(pmin, 0, -math.max(tolerance, 0), 0), vector.offset(pmax, 0, math.max(tolerance, 0), 0)
|
||||||
-- if we need to generate a foundation, we need to emerge a larger area:
|
-- if we need to generate a foundation, we need to emerge a larger area:
|
||||||
if prepare.foundation or prepare.clear then -- these functions need some extra margins. Must match mcl_foundations!
|
if prepare.foundation or prepare.clear then -- these functions need some extra margins. Must match vl_terraforming!
|
||||||
local padding = (prepare.padding or 0) + 3
|
local padding = (prepare.padding or 0) + 3
|
||||||
local depth = prepare.foundation and ((type(prepare.foundation) == "number" and prepare.foundation or DEFAULT_PREPARE.foundation) - 3) or 0 -- minimum depth
|
local depth = prepare.foundation and ((type(prepare.foundation) == "number" and prepare.foundation or vl_structures.DEFAULT_PREPARE.foundation) - 3) or 0 -- minimum depth
|
||||||
local height = prepare.clear and ((prepare.clear_top or DEFAULT_PREPARE.clear_top)*1.5+0.5*(size.y+yoffset)+2) or 0 -- headroom
|
local height = prepare.clear and ((prepare.clear_top or vl_structures.DEFAULT_PREPARE.clear_top)*1.5+0.5*(size.y+yoffset)+2) or 0 -- headroom
|
||||||
emin = vector_offset(emin, -padding, depth, -padding)
|
emin = vector_offset(emin, -padding, depth, -padding)
|
||||||
emax = vector_offset(emax, padding, height, padding)
|
emax = vector_offset(emax, padding, height, padding)
|
||||||
end
|
end
|
||||||
|
@ -1,13 +1,18 @@
|
|||||||
local modname = minetest.get_current_modname()
|
local modname = minetest.get_current_modname()
|
||||||
local S = minetest.get_translator(modname)
|
|
||||||
local modpath = minetest.get_modpath(modname)
|
|
||||||
|
|
||||||
vl_structures = {}
|
vl_structures = {}
|
||||||
|
|
||||||
|
---- Customization parameters for different games
|
||||||
-- see vl_terraforming for documentation
|
-- see vl_terraforming for documentation
|
||||||
vl_structures.DEFAULT_PREPARE = { tolerance = 10, foundation = -3, clear = false, clear_bottom = 0, clear_top = 4, padding = 1, corners = 1 }
|
vl_structures.DEFAULT_PREPARE = { tolerance = 10, foundation = -3, clear = false, clear_bottom = 0, clear_top = 4, padding = 1, corners = 1 }
|
||||||
vl_structures.DEFAULT_FLAGS = "place_center_x,place_center_z"
|
vl_structures.DEFAULT_FLAGS = "place_center_x,place_center_z"
|
||||||
|
|
||||||
|
-- fallback types
|
||||||
|
vl_structures.DEFAULT_SURFACE = { name = "mcl_core:dirt_with_grass" }
|
||||||
|
vl_structures.DEFAULT_FILLER = { name = "mcl_core:dirt" }
|
||||||
|
vl_structures.DEFAULT_STONE = { name = "mcl_core:stone" }
|
||||||
|
vl_structures.DEFAULT_DUST = nil
|
||||||
|
|
||||||
|
local modpath = minetest.get_modpath(modname)
|
||||||
dofile(modpath.."/util.lua")
|
dofile(modpath.."/util.lua")
|
||||||
dofile(modpath.."/emerge.lua")
|
dofile(modpath.."/emerge.lua")
|
||||||
dofile(modpath.."/api.lua")
|
dofile(modpath.."/api.lua")
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
name = vl_structures
|
name = vl_structures
|
||||||
author = kno10
|
author = kno10
|
||||||
description = Structures API for VoxeLibre and Mineclonia
|
description = Structure spawning API for VoxeLibre
|
||||||
depends = mcl_init, mcl_util, mcl_loot, vl_terraforming
|
depends = vl_terraforming
|
||||||
|
optional_depends = mcl_util, mcl_loot
|
||||||
|
@ -1,10 +1,54 @@
|
|||||||
-- todo: move this mostly to the mcl_mobs module?
|
-- TODO: move this to the mcl_mobs module?
|
||||||
local mob_cap_player = tonumber(minetest.settings:get("mcl_mob_cap_player")) or 75
|
local mob_cap_player = tonumber(minetest.settings:get("mcl_mob_cap_player")) or 75
|
||||||
local mob_cap_animal = tonumber(minetest.settings:get("mcl_mob_cap_animal")) or 10
|
local mob_cap_animal = tonumber(minetest.settings:get("mcl_mob_cap_animal")) or 10
|
||||||
local peaceful = minetest.settings:get_bool("only_peaceful_mobs", false)
|
local peaceful = minetest.settings:get_bool("only_peaceful_mobs", false)
|
||||||
local mg_name = minetest.get_mapgen_setting("mg_name")
|
local mg_name = minetest.get_mapgen_setting("mg_name")
|
||||||
local vector_offset = vector.offset
|
local vector_offset = vector.offset
|
||||||
|
|
||||||
|
-- check if a node is an air node
|
||||||
|
local function is_air(node)
|
||||||
|
return node == "air"
|
||||||
|
-- todo: or: not walkable and not liquid?
|
||||||
|
end
|
||||||
|
-- check if a node is a water node
|
||||||
|
local function is_water(node)
|
||||||
|
return minetest.get_item_group(node.name, "water") ~= 0
|
||||||
|
-- return node.name == "mcl_core:water_source" or node.name ~= "mclx_core:river_water_source"
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Spawn mobs for a structure
|
||||||
|
-- @param mob string: mob to spawn
|
||||||
|
-- @param spawnon string or table: nodes to spawn on
|
||||||
|
-- @param p1 vector: Lowest coordinates of range
|
||||||
|
-- @param p2 vector: Highest coordinates of range
|
||||||
|
-- @param pr PseudoRandom: random generator
|
||||||
|
-- @param n number: Number of mobs to spawn
|
||||||
|
-- @param water boolean: Spawn water mobs
|
||||||
|
function vl_structures.spawn_mobs(mob,spawnon,p1,p2,pr,n,water)
|
||||||
|
n = n or 1
|
||||||
|
local sp = {}
|
||||||
|
if water then
|
||||||
|
local nn = minetest.find_nodes_in_area(p1,p2,spawnon)
|
||||||
|
for k,v in pairs(nn) do
|
||||||
|
if is_water(minetest.get_node(vector_offset(v,0,1,0))) then
|
||||||
|
table.insert(sp,v)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
sp = minetest.find_nodes_in_area_under_air(p1,p2,spawnon)
|
||||||
|
end
|
||||||
|
table.shuffle(sp)
|
||||||
|
local count = 0
|
||||||
|
local mob_def = minetest.registered_entities[mob]
|
||||||
|
local enabled = (not peaceful) or (mob_def and mob_spawn_class ~= "hostile")
|
||||||
|
for _, node in pairs(sp) do
|
||||||
|
if enabled and count < n and minetest.add_entity(vector_offset(node, 0, 0.5, 0), mob) then
|
||||||
|
count = count + 1
|
||||||
|
end
|
||||||
|
minetest.get_meta(node):set_string("spawnblock", "yes") -- note: also in peaceful mode!
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
local structure_spawns = {}
|
local structure_spawns = {}
|
||||||
--- Structure spawns via ABM
|
--- Structure spawns via ABM
|
||||||
-- @param def table: containing
|
-- @param def table: containing
|
||||||
@ -30,12 +74,8 @@ function vl_structures.register_structure_spawn(def)
|
|||||||
if active_object_count_wider > limit + mob_cap_animal then return end
|
if active_object_count_wider > limit + mob_cap_animal then return end
|
||||||
if active_object_count_wider > mob_cap_player then return end
|
if active_object_count_wider > mob_cap_player then return end
|
||||||
local p = vector_offset(pos, 0, 1, 0)
|
local p = vector_offset(pos, 0, 1, 0)
|
||||||
local pname = minetest.get_node(p).name
|
local pnode = minetest.get_node(p)
|
||||||
if def.type_of_spawning == "water" then
|
if not (def.type_of_spawning == "water" and is_water or is_air)(pnode) then return end
|
||||||
if pname ~= "mcl_core:water_source" and pname ~= "mclx_core:river_water_source" then return end
|
|
||||||
else
|
|
||||||
if pname ~= "air" then return end -- FIXME: allow everything non-walkable, non-water, non-lava?
|
|
||||||
end
|
|
||||||
if minetest.get_meta(pos):get_string("spawnblock") == "" then return end
|
if minetest.get_meta(pos):get_string("spawnblock") == "" then return end
|
||||||
if mg_name ~= "v6" and mg_name ~= "singlenode" and def.biomes then
|
if mg_name ~= "v6" and mg_name ~= "singlenode" and def.biomes then
|
||||||
if table.indexof(def.biomes, minetest.get_biome_name(minetest.get_biome_data(p).biome)) == -1 then
|
if table.indexof(def.biomes, minetest.get_biome_name(minetest.get_biome_data(p).biome)) == -1 then
|
||||||
@ -49,36 +89,3 @@ function vl_structures.register_structure_spawn(def)
|
|||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
--- Spawn mobs for a structure
|
|
||||||
-- @param mob string: mob to spawn
|
|
||||||
-- @param spawnon string or table: nodes to spawn on
|
|
||||||
-- @param p1 vector: Lowest coordinates of range
|
|
||||||
-- @param p2 vector: Highest coordinates of range
|
|
||||||
-- @param pr PseudoRandom: random generator
|
|
||||||
-- @param n number: Number of mobs to spawn
|
|
||||||
-- @param water boolean: Spawn water mobs
|
|
||||||
function vl_structures.spawn_mobs(mob,spawnon,p1,p2,pr,n,water)
|
|
||||||
n = n or 1
|
|
||||||
local sp = {}
|
|
||||||
if water then
|
|
||||||
local nn = minetest.find_nodes_in_area(p1,p2,spawnon)
|
|
||||||
for k,v in pairs(nn) do
|
|
||||||
if minetest.get_item_group(minetest.get_node(vector_offset(v,0,1,0)).name,"water") > 0 then
|
|
||||||
table.insert(sp,v)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
else
|
|
||||||
sp = minetest.find_nodes_in_area_under_air(p1,p2,spawnon)
|
|
||||||
end
|
|
||||||
table.shuffle(sp)
|
|
||||||
local count = 0
|
|
||||||
local mob_def = minetest.registered_entities[mob]
|
|
||||||
local enabled = (not peaceful) or (mob_def and mob_spawn_class ~= "hostile")
|
|
||||||
for _, node in pairs(sp) do
|
|
||||||
if enabled and count < n and minetest.add_entity(vector_offset(node, 0, 0.5, 0), mob) then
|
|
||||||
count = count + 1
|
|
||||||
end
|
|
||||||
minetest.get_meta(node):set_string("spawnblock", "yes") -- note: also in peaceful mode!
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
|
@ -83,6 +83,7 @@ end
|
|||||||
-- @param loot table: Loot table
|
-- @param loot table: Loot table
|
||||||
-- @param pr PseudoRandom: random generator
|
-- @param pr PseudoRandom: random generator
|
||||||
function vl_structures.fill_chests(p1,p2,loot,pr)
|
function vl_structures.fill_chests(p1,p2,loot,pr)
|
||||||
|
if not mcl_loot then return end -- optional dependency
|
||||||
for it,lt in pairs(loot) do
|
for it,lt in pairs(loot) do
|
||||||
local nodes = minetest.find_nodes_in_area(p1, p2, it)
|
local nodes = minetest.find_nodes_in_area(p1, p2, it)
|
||||||
for _,p in pairs(nodes) do
|
for _,p in pairs(nodes) do
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
# vl_terraforming
|
# vl_terraforming
|
||||||
Terraforming module for VoxeLibre and MineClonia
|
|
||||||
|
Terraforming module built with VoxeLibre and MineClonia in mind, but also useful for other games.
|
||||||
|
|
||||||
This module provides the following key functionalities:
|
This module provides the following key functionalities:
|
||||||
|
|
||||||
@ -8,7 +9,6 @@ This module provides the following key functionalities:
|
|||||||
- build a baseplate for a building
|
- build a baseplate for a building
|
||||||
- clear the area above a building
|
- clear the area above a building
|
||||||
|
|
||||||
|
|
||||||
## Rounded corners support
|
## Rounded corners support
|
||||||
|
|
||||||
To get nicer looking baseplates, the code supports rounded corners.
|
To get nicer looking baseplates, the code supports rounded corners.
|
||||||
@ -55,7 +55,7 @@ One of these values may be "extreme", and tolerance specifies the maximum height
|
|||||||
|
|
||||||
The (rounded) median of these values is used, unless tolerance is set to "min" or "max".
|
The (rounded) median of these values is used, unless tolerance is set to "min" or "max".
|
||||||
|
|
||||||
The "mode" can be set to "solid" (default), "liquid" (liquid surfaces only), "under_air" (both liquid and solid surfaces).
|
The "mode" can be set to "solid" (default), "liquid" (liquid surfaces only), "under_air" (both liquid and solid surfaces), "under_water" (solid below water).
|
||||||
|
|
||||||
|
|
||||||
## vl_terraforming.foundation_vm(vm, px, py, pz, sx, sy, sz, corners, surface_mat, platform_mat, stone_mat, dust_mat, pr)
|
## vl_terraforming.foundation_vm(vm, px, py, pz, sx, sy, sz, corners, surface_mat, platform_mat, stone_mat, dust_mat, pr)
|
||||||
@ -94,6 +94,6 @@ pr is a PcgRandom random generator
|
|||||||
|
|
||||||
- [ ] add an API that works on VM buffers
|
- [ ] add an API that works on VM buffers
|
||||||
- [ ] add an API version working on the non-VM API
|
- [ ] add an API version working on the non-VM API
|
||||||
- [ ] benchmark if VM is actually faster than not using VM (5.9 has some optimizations not in VM)
|
- [ ] benchmark if VM is actually faster than not using VM (5.9 has some optimizations not yet in VM)
|
||||||
- [ ] improve tree removal
|
- [ ] improve tree removal
|
||||||
|
|
||||||
|
@ -102,7 +102,7 @@ function vl_terraforming.foundation_vm(vm, px, py, pz, sx, sy, sz, corners, surf
|
|||||||
end
|
end
|
||||||
-- construct additional baseplate below, also try to make it interesting
|
-- construct additional baseplate below, also try to make it interesting
|
||||||
for yi = py-2,py-20,-1 do
|
for yi = py-2,py-20,-1 do
|
||||||
local dy2 = max(0,py-2-yi)^2*0.05
|
local dy2 = max(0,py-2-yi)^2*0.10
|
||||||
local active = false
|
local active = false
|
||||||
for xi = px-1,px+sx do
|
for xi = px-1,px+sx do
|
||||||
local dx22 = max(abs(cx-xi)-1.49,0)^2*wx2
|
local dx22 = max(abs(cx-xi)-1.49,0)^2*wx2
|
||||||
|
@ -15,7 +15,7 @@ function vl_terraforming.find_ground_vm(vm, pos)
|
|||||||
local cur = vm:get_node_at(pos)
|
local cur = vm:get_node_at(pos)
|
||||||
if cur.name == "ignore" then
|
if cur.name == "ignore" then
|
||||||
local e1, e2 = vm:get_emerged_area()
|
local e1, e2 = vm:get_emerged_area()
|
||||||
minetest.log("warning","find_ground with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
minetest.log("warning", "find_ground with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
||||||
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -64,7 +64,7 @@ function vl_terraforming.find_under_air_vm(vm, pos)
|
|||||||
local cur = vm:get_node_at(pos)
|
local cur = vm:get_node_at(pos)
|
||||||
if cur.name == "ignore" then
|
if cur.name == "ignore" then
|
||||||
local e1, e2 = vm:get_emerged_area()
|
local e1, e2 = vm:get_emerged_area()
|
||||||
minetest.log("warning","find_under_air with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
minetest.log("warning", "find_under_air with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
||||||
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -112,7 +112,7 @@ function vl_terraforming.find_liquid_surface_vm(vm, pos)
|
|||||||
local cur = vm:get_node_at(pos)
|
local cur = vm:get_node_at(pos)
|
||||||
if cur.name == "ignore" then
|
if cur.name == "ignore" then
|
||||||
local e1, e2 = vm:get_emerged_area()
|
local e1, e2 = vm:get_emerged_area()
|
||||||
minetest.log("warning","find_liquid_surface with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
minetest.log("warning", "find_liquid_surface with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
||||||
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
@ -154,6 +154,59 @@ function vl_terraforming.find_liquid_surface_vm(vm, pos)
|
|||||||
end
|
end
|
||||||
local find_liquid_surface_vm = vl_terraforming.find_liquid_surface_vm
|
local find_liquid_surface_vm = vl_terraforming.find_liquid_surface_vm
|
||||||
|
|
||||||
|
--- Find under water surface for a given position
|
||||||
|
-- @param vm VoxelManip: buffer
|
||||||
|
-- @param pos vector: Start position
|
||||||
|
-- @return position and material of surface
|
||||||
|
function vl_terraforming.find_under_water_surface_vm(vm, pos)
|
||||||
|
if not pos then return nil, nil end
|
||||||
|
pos = vector_copy(pos)
|
||||||
|
local cur = vm:get_node_at(pos)
|
||||||
|
if cur.name == "ignore" then
|
||||||
|
local e1, e2 = vm:get_emerged_area()
|
||||||
|
minetest.log("warning", "find_under_water_surface with invalid position (outside of emerged area?) at "..minetest.pos_to_string(pos)
|
||||||
|
..": "..tostring(cur and cur.name).." area: "..minetest.pos_to_string(e1).." "..minetest.pos_to_string(e2))
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
if is_solid_not_tree(cur) then -- find up
|
||||||
|
local prev = cur
|
||||||
|
while true do
|
||||||
|
pos.y = pos.y + 1
|
||||||
|
local cur = vm:get_node_at(pos)
|
||||||
|
if not cur or cur.name == "ignore" then
|
||||||
|
-- minetest.log("action", "No ground, "..tostring(cur and cur.name).." over "..tostring(prev and prev.name).." at "..minetest.pos_to_string(pos))
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
if is_liquid(cur) then
|
||||||
|
pos.y = pos.y - 1
|
||||||
|
-- minetest.log("action", "Found surface: "..minetest.pos_to_string(pos).." "..tostring(prev and prev.name).." under "..tostring(cur and cur.name))
|
||||||
|
return pos, prev
|
||||||
|
end
|
||||||
|
prev = cur
|
||||||
|
end
|
||||||
|
else -- find down
|
||||||
|
while true do
|
||||||
|
pos.y = pos.y - 1
|
||||||
|
local prev = cur
|
||||||
|
local cur = vm:get_node_at(pos)
|
||||||
|
if not cur or cur.name == "ignore" then
|
||||||
|
-- minetest.log("action", "No ground, "..tostring(cur and cur.name).." below "..tostring(prev and prev.name).." at "..minetest.pos_to_string(pos))
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
if is_solid_not_tree(cur) then
|
||||||
|
if is_liquid(prev) then
|
||||||
|
-- minetest.log("action", "Found surface: "..minetest.pos_to_string(pos).." "..(cur and cur.name).." over "..(prev and prev.name))
|
||||||
|
return pos, cur
|
||||||
|
else
|
||||||
|
-- minetest.log("action", "No ground, "..tostring(cur and cur.name).." below "..tostring(prev and prev.name).." at "..minetest.pos_to_string(pos))
|
||||||
|
return nil, nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local find_under_water_surface_vm = vl_terraforming.find_under_water_surface_vm
|
||||||
|
|
||||||
--- find suitable height for a structure of this size
|
--- find suitable height for a structure of this size
|
||||||
-- @param vm VoxelManip: to read data
|
-- @param vm VoxelManip: to read data
|
||||||
-- @param cpos vector: center
|
-- @param cpos vector: center
|
||||||
@ -164,10 +217,14 @@ local find_liquid_surface_vm = vl_terraforming.find_liquid_surface_vm
|
|||||||
function vl_terraforming.find_level_vm(vm, cpos, size, tolerance, mode)
|
function vl_terraforming.find_level_vm(vm, cpos, size, tolerance, mode)
|
||||||
local find_ground = find_ground_vm
|
local find_ground = find_ground_vm
|
||||||
if mode == "liquid_surface" or mode == "liquid" then find_ground = find_liquid_surface_vm end
|
if mode == "liquid_surface" or mode == "liquid" then find_ground = find_liquid_surface_vm end
|
||||||
|
if mode == "under_water" or mode == "water" then find_ground = find_under_water_surface_vm end
|
||||||
if mode == "under_air" then find_ground = find_under_air_vm end
|
if mode == "under_air" then find_ground = find_under_air_vm end
|
||||||
-- begin at center, then top-left and clockwise
|
-- begin at center, then top-left and clockwise
|
||||||
local pos, surface_material = find_ground(vm, cpos)
|
local pos, surface_material = find_ground(vm, cpos)
|
||||||
if not pos then return nil, nil end
|
if not pos then
|
||||||
|
-- minetest.log("action", "[vl_terraforming] no ground at starting position "..minetest.pos_to_string(cpos).." mode "..tostring(mode or "default"))
|
||||||
|
return nil, nil
|
||||||
|
end
|
||||||
local ys = { pos.y }
|
local ys = { pos.y }
|
||||||
pos.y = pos.y + 1 -- position above surface
|
pos.y = pos.y + 1 -- position above surface
|
||||||
if size.x == 1 and size.z == 1 then return pos end
|
if size.x == 1 and size.z == 1 then return pos end
|
||||||
@ -200,8 +257,8 @@ function vl_terraforming.find_level_vm(vm, cpos, size, tolerance, mode)
|
|||||||
end
|
end
|
||||||
-- well supported base, not too uneven?
|
-- well supported base, not too uneven?
|
||||||
if #ys < 4 or min(ys[#ys-1]-ys[1], ys[#ys]-ys[2]) > tolerance then
|
if #ys < 4 or min(ys[#ys-1]-ys[1], ys[#ys]-ys[2]) > tolerance then
|
||||||
--minetest.log("action", "[vl_terraforming] ground too uneven: "..#ys.." positions: "..({dump(ys):gsub("[\n\t ]+", " ")})[1]
|
-- minetest.log("action", "[vl_terraforming] ground too uneven: "..#ys.." positions: "..({dump(ys):gsub("[\n\t ]+", " ")})[1]
|
||||||
-- .." tolerance "..tostring(#ys > 2 and min(ys[#ys-1]-ys[1], ys[#ys]-ys[2])).." > "..tolerance)
|
-- .." tolerance "..tostring(#ys > 2 and min(ys[#ys-1]-ys[1], ys[#ys]-ys[2])).." > "..tolerance)
|
||||||
return nil, nil
|
return nil, nil
|
||||||
end
|
end
|
||||||
cpos.y = floor(0.5 * (ys[floor(1 + (#ys - 1) * 0.5)] + ys[ceil(1 + (#ys - 1) * 0.5)]) + 1) -- median except for largest, rounded, over surface
|
cpos.y = floor(0.5 * (ys[floor(1 + (#ys - 1) * 0.5)] + ys[ceil(1 + (#ys - 1) * 0.5)]) + 1) -- median except for largest, rounded, over surface
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
name = vl_terraforming
|
name = vl_terraforming
|
||||||
author = kno10
|
author = kno10
|
||||||
description = Terraforming API for VoxeLibre and Mineclonia
|
description = Terraforming API
|
||||||
|
@ -40,7 +40,7 @@ mcl_doTileDrops (Blocks have drops) bool true
|
|||||||
mcl_explosions_griefing (Explosions destroy blocks) bool true
|
mcl_explosions_griefing (Explosions destroy blocks) bool true
|
||||||
|
|
||||||
# Comma separated list of disabled structure names
|
# Comma separated list of disabled structure names
|
||||||
mcl_disabled_structures (Disabled structures) string
|
vl_structures_disabled (Disabled structures) string
|
||||||
|
|
||||||
# Comma separated list of disabled event names
|
# Comma separated list of disabled event names
|
||||||
mcl_disabled_events (Disabled events) string
|
mcl_disabled_events (Disabled events) string
|
||||||
@ -350,7 +350,7 @@ mcl_logging_mobs_spawn (Log Mob Spawning) bool false
|
|||||||
mcl_logging_mapgen (Chunk generation logging) bool false
|
mcl_logging_mapgen (Chunk generation logging) bool false
|
||||||
|
|
||||||
# If enabled generated structures will be logged
|
# If enabled generated structures will be logged
|
||||||
mcl_logging_structures (Structure generation logging) bool false
|
vl_structures_logging (Structure generation logging) bool false
|
||||||
|
|
||||||
#Complete debug logging for mcl_signs events. Use this if you have issues with signs.
|
#Complete debug logging for mcl_signs events. Use this if you have issues with signs.
|
||||||
mcl_logging_mcl_signs (Complete debug logging for mcl_signs) bool false
|
mcl_logging_mcl_signs (Complete debug logging for mcl_signs) bool false
|
||||||
|
Loading…
Reference in New Issue
Block a user