mirror of
https://github.com/sys4-fr/server-nalc.git
synced 2024-12-28 19:50:33 +01:00
977 lines
25 KiB
Lua
977 lines
25 KiB
Lua
-- Nether Mod (based on Nyanland by Jeija, Catapult by XYZ, and Livehouse by neko259)
|
|
-- lkjoel (main developer, code, ideas, textures)
|
|
-- == CONTRIBUTERS ==
|
|
-- jordan4ibanez (code, ideas, textures)
|
|
-- Gilli (code, ideas, textures, mainly for the Glowstone)
|
|
-- Death Dealer (code, ideas, textures)
|
|
-- LolManKuba (ideas, textures)
|
|
-- IPushButton2653 (ideas, textures)
|
|
-- Menche (textures)
|
|
-- sdzen (ideas)
|
|
-- godkiller447 (ideas)
|
|
-- If I didn't list you, please let me know!
|
|
|
|
if not rawget(_G, "nether") then
|
|
nether = {}
|
|
end
|
|
|
|
--== EDITABLE OPTIONS ==--
|
|
|
|
--says some information.
|
|
nether.info = true
|
|
|
|
-- tell everyone about the generation
|
|
nether.inform_all = minetest.is_singleplayer()
|
|
|
|
--1:<a bit of information> 2:<acceptable amount of information> 3:<lots of text>
|
|
nether.max_spam = 2
|
|
|
|
-- Depth of the nether
|
|
local nether_middle = -20000
|
|
|
|
-- forest bottom perlin multiplication
|
|
local f_bottom_scale = 4
|
|
|
|
-- forest bottom height
|
|
local f_h_min = nether_middle+10
|
|
|
|
-- forest top height
|
|
local f_h_max = f_h_min+250
|
|
|
|
-- Frequency of trees in the nether forest (higher is less frequent)
|
|
local tree_rarity = 200
|
|
|
|
-- Frequency of glowflowers in the nether forest (higher is less frequent)
|
|
local glowflower_rarity = 120
|
|
|
|
-- Frequency of nether grass in the nether forest (higher is less frequent)
|
|
local grass_rarity = 2
|
|
|
|
-- Frequency of nether mushrooms in the nether forest (higher is less frequent)
|
|
local mushroom_rarity = 80
|
|
|
|
-- Frequency of trees in the nether forest (higher is less frequent)
|
|
local tree_rarity = 200
|
|
|
|
-- height of the nether generation's end
|
|
nether.start = f_h_max+100
|
|
|
|
-- Height of the nether (bottom of the nether is nether_middle - NETHER_HEIGHT)
|
|
local NETHER_HEIGHT = 30
|
|
|
|
-- Maximum amount of randomness in the map generation
|
|
NETHER_RANDOM = 2
|
|
|
|
-- Frequency of Glowstone on the "roof" of the Nether (higher is less frequent)
|
|
local GLOWSTONE_FREQ_ROOF = 500
|
|
|
|
-- Frequency of lava (higher is less frequent)
|
|
local LAVA_FREQ = 100
|
|
|
|
local nether_structure_freq = 350
|
|
local NETHER_SHROOM_FREQ = 100
|
|
|
|
-- Maximum height of lava
|
|
--LAVA_HEIGHT = 2
|
|
-- Frequency of Glowstone on lava (higher is less frequent)
|
|
--GLOWSTONE_FREQ_LAVA = 2
|
|
-- Height of nether structures
|
|
--NETHER_TREESIZE = 2
|
|
-- Frequency of apples in a nether structure (higher is less frequent)
|
|
--NETHER_APPLE_FREQ = 5
|
|
-- Frequency of healing apples in a nether structure (higher is less frequent)
|
|
--NETHER_HEAL_APPLE_FREQ = 10
|
|
-- Start position for the Throne of Hades (y is relative to the bottom of the nether)
|
|
--HADES_THRONE_STARTPOS = {x=0, y=1, z=0}
|
|
-- Spawn pos for when the nether hasn't been loaded yet (i.e. no portal in the nether) (y is relative to the bottom of the nether)
|
|
--NETHER_SPAWNPOS = {x=0, y=5, z=0}
|
|
-- Structure of the nether portal (all is relative to the nether portal creator block)
|
|
|
|
--== END OF EDITABLE OPTIONS ==--
|
|
|
|
if nether.info then
|
|
function nether:inform(msg, spam, t)
|
|
if spam <= self.max_spam then
|
|
local info
|
|
if t then
|
|
info = string.format("[nether] "..msg.." after ca. %.2fs", os.clock() - t)
|
|
else
|
|
info = "[nether] "..msg
|
|
end
|
|
print(info)
|
|
if self.inform_all then
|
|
minetest.chat_send_all(info)
|
|
end
|
|
end
|
|
end
|
|
else
|
|
function nether:inform()
|
|
end
|
|
end
|
|
|
|
|
|
local path = minetest.get_modpath("nether")
|
|
dofile(path.."/weird_mapgen_noise.lua")
|
|
dofile(path.."/items.lua")
|
|
dofile(path.."/furnace.lua")
|
|
dofile(path.."/pearl.lua")
|
|
|
|
local function table_contains(t, v)
|
|
for _,i in pairs(t) do
|
|
if i == v then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
local function dif(z1, z2)
|
|
if z1 < 0
|
|
and z2 < 0 then
|
|
z1,z2 = -z1,-z2
|
|
end
|
|
return math.abs(z1-z2)
|
|
end
|
|
|
|
local function pymg(x1, x2, z1, z2)
|
|
return math.max(dif(x1, x2), dif(z1, z2))
|
|
end
|
|
|
|
local function r_area(manip, width, height, pos)
|
|
local emerged_pos1, emerged_pos2 = manip:read_from_map(
|
|
{x=pos.x-width, y=pos.y, z=pos.z-width},
|
|
{x=pos.x+width, y=pos.y+height, z=pos.z+width}
|
|
)
|
|
return VoxelArea:new({MinEdge=emerged_pos1, MaxEdge=emerged_pos2})
|
|
end
|
|
|
|
local function set_vm_data(manip, nodes, pos, t1, name, generated)
|
|
manip:set_data(nodes)
|
|
manip:write_to_map()
|
|
local spam = 2
|
|
if generated then
|
|
spam = 3
|
|
end
|
|
nether:inform(name.." grew at ("..pos.x.."|"..pos.y.."|"..pos.z..")", spam, t1)
|
|
if not generated then
|
|
local t1 = os.clock()
|
|
manip:update_map()
|
|
nether:inform("map updated", spam, t1)
|
|
end
|
|
end
|
|
|
|
local function fix_light(minp, maxp)
|
|
local manip = minetest.get_voxel_manip()
|
|
local emerged_pos1, emerged_pos2 = manip:read_from_map(minp, maxp)
|
|
area = VoxelArea:new({MinEdge=emerged_pos1, MaxEdge=emerged_pos2})
|
|
nodes = manip:get_data()
|
|
|
|
manip:set_data(nodes)
|
|
manip:write_to_map()
|
|
manip:update_map()
|
|
end
|
|
|
|
-- Generated variables
|
|
local NETHER_BOTTOM = (nether_middle - NETHER_HEIGHT)
|
|
nether.buildings = NETHER_BOTTOM+12
|
|
local NETHER_ROOF_ABS = (nether_middle - NETHER_RANDOM)
|
|
local f_yscale_top = (f_h_max-f_h_min)/2
|
|
local f_yscale_bottom = f_yscale_top/2
|
|
--HADES_THRONE_STARTPOS_ABS = {x=HADES_THRONE_STARTPOS.x, y=(NETHER_BOTTOM + HADES_THRONE_STARTPOS.y), z=HADES_THRONE_STARTPOS.z}
|
|
--LAVA_Y = (NETHER_BOTTOM + LAVA_HEIGHT)
|
|
--HADES_THRONE_ABS = {}
|
|
--HADES_THRONE_ENDPOS_ABS = {}
|
|
--HADES_THRONE_GENERATED = minetest.get_worldpath() .. "/netherhadesthrone.txt"
|
|
--NETHER_SPAWNPOS_ABS = {x=NETHER_SPAWNPOS.x, y=(NETHER_BOTTOM + NETHER_SPAWNPOS.y), z=NETHER_SPAWNPOS.z}
|
|
--[[for i,v in ipairs(HADES_THRONE) do
|
|
v.pos.x = v.pos.x + HADES_THRONE_STARTPOS_ABS.x
|
|
v.pos.y = v.pos.y + HADES_THRONE_STARTPOS_ABS.y
|
|
v.pos.z = v.pos.z + HADES_THRONE_STARTPOS_ABS.z
|
|
HADES_THRONE_ABS[i] = v
|
|
end
|
|
local htx = 0
|
|
local hty = 0
|
|
local htz = 0
|
|
for i,v in ipairs(HADES_THRONE_ABS) do
|
|
if v.pos.x > htx then
|
|
htx = v.pos.x
|
|
end
|
|
if v.pos.y > hty then
|
|
hty = v.pos.y
|
|
end
|
|
if v.pos.z > htz then
|
|
htz = v.pos.z
|
|
end
|
|
end
|
|
HADES_THRONE_ENDPOS_ABS = {x=htx, y=hty, z=htz}]]
|
|
|
|
local c
|
|
local function define_contents()
|
|
c = {
|
|
air = minetest.get_content_id("air"),
|
|
lava = minetest.get_content_id("default:lava_source"),
|
|
gravel = minetest.get_content_id("default:gravel"),
|
|
coal = minetest.get_content_id("default:stone_with_coal"),
|
|
diamond = minetest.get_content_id("default:stone_with_diamond"),
|
|
mese = minetest.get_content_id("default:mese"),
|
|
|
|
glowstone = minetest.get_content_id("glow:stone"), --https://github.com/Zeg9/minetest-glow
|
|
|
|
nether_shroom = minetest.get_content_id("riesenpilz:nether_shroom"),
|
|
|
|
netherrack = minetest.get_content_id("nether:netherrack"),
|
|
netherrack_tiled = minetest.get_content_id("nether:netherrack_tiled"),
|
|
netherrack_black = minetest.get_content_id("nether:netherrack_black"),
|
|
netherrack_blue = minetest.get_content_id("nether:netherrack_blue"),
|
|
netherrack_brick = minetest.get_content_id("nether:netherrack_brick"),
|
|
white = minetest.get_content_id("nether:white"),
|
|
|
|
nether_vine = minetest.get_content_id("nether:vine"),
|
|
blood = minetest.get_content_id("nether:blood"),
|
|
blood_top = minetest.get_content_id("nether:blood_top"),
|
|
blood_stem = minetest.get_content_id("nether:blood_stem"),
|
|
nether_apple = minetest.get_content_id("nether:apple"),
|
|
|
|
nether_tree = minetest.get_content_id("nether:tree"),
|
|
nether_tree_corner = minetest.get_content_id("nether:tree_corner"),
|
|
nether_leaves = minetest.get_content_id("nether:leaves"),
|
|
nether_grass = {
|
|
minetest.get_content_id("nether:grass_small"),
|
|
minetest.get_content_id("nether:grass_middle"),
|
|
minetest.get_content_id("nether:grass_big")
|
|
},
|
|
glowflower = minetest.get_content_id("nether:glowflower"),
|
|
nether_dirt = minetest.get_content_id("nether:dirt"),
|
|
nether_dirt_top = minetest.get_content_id("nether:dirt_top"),
|
|
nether_dirt_bottom = minetest.get_content_id("nether:dirt_bottom"),
|
|
}
|
|
end
|
|
|
|
local pr, contents_defined
|
|
|
|
local function return_nether_ore(id, glowstone)
|
|
if glowstone
|
|
and pr:next(0,GLOWSTONE_FREQ_ROOF) == 1 then
|
|
return c.glowstone
|
|
end
|
|
if id == c.coal then
|
|
return c.netherrack_tiled
|
|
end
|
|
if id == c.gravel then
|
|
return c.netherrack_black
|
|
end
|
|
if id == c.diamond then
|
|
return c.netherrack_blue
|
|
end
|
|
if id == c.mese then
|
|
return c.white
|
|
end
|
|
return c.netherrack
|
|
end
|
|
|
|
local f_perlins = {}
|
|
|
|
--local perlin1 = minetest.get_perlin(13,3, 0.5, 50) --Get map specific perlin
|
|
-- local perlin2 = minetest.get_perlin(133,3, 0.5, 10)
|
|
-- local perlin3 = minetest.get_perlin(112,3, 0.5, 5)
|
|
local tmp = f_yscale_top*4
|
|
local tmp2 = tmp/f_bottom_scale
|
|
local perlins = {
|
|
{
|
|
seed = 13,
|
|
octaves = 3,
|
|
persist = 0.5,
|
|
spread = {x=50, y=50, z=50},
|
|
scale = 1,
|
|
offset = 0,
|
|
},
|
|
{
|
|
seed = 133,
|
|
octaves = 3,
|
|
persist = 0.5,
|
|
spread = {x=10, y=10, z=10},
|
|
scale = 1,
|
|
offset = 0,
|
|
},
|
|
{
|
|
seed = 112,
|
|
octaves = 3,
|
|
persist = 0.5,
|
|
spread = {x=5, y=5, z=5},
|
|
scale = 1,
|
|
offset = 0,
|
|
},
|
|
--[[forest_bottom = {
|
|
seed = 11,
|
|
octaves = 3,
|
|
persist = 0.8,
|
|
spread = {x=tmp2, y=tmp2, z=tmp2},
|
|
scale = 1,
|
|
offset = 0,
|
|
},]]
|
|
forest_top = {
|
|
seed = 21,
|
|
octaves = 3,
|
|
persist = 0.8,
|
|
spread = {x=tmp, y=tmp, z=tmp},
|
|
scale = 1,
|
|
offset = 0,
|
|
},
|
|
}
|
|
|
|
local info = true
|
|
local structures_enabled = true
|
|
local vine_maxlength = math.floor(NETHER_HEIGHT/4+0.5)
|
|
-- Create the Nether
|
|
minetest.register_on_generated(function(minp, maxp, seed)
|
|
if not (maxp.y >= NETHER_BOTTOM-100 and minp.y <= nether.start) then --avoid big map generation
|
|
return
|
|
end
|
|
local addpos = {}
|
|
|
|
local t1 = os.clock()
|
|
nether:inform("generates at: x=["..minp.x.."; "..maxp.x.."]; y=["..minp.y.."; "..maxp.y.."]; z=["..minp.z.."; "..maxp.z.."]", 2)
|
|
|
|
if not contents_defined then
|
|
define_contents()
|
|
contents_defined = true
|
|
end
|
|
|
|
local buildings = 0
|
|
if maxp.y <= NETHER_BOTTOM then
|
|
buildings = 1
|
|
elseif minp.y <= nether.buildings then
|
|
buildings = 2
|
|
end
|
|
|
|
local vm, emin, emax = minetest.get_mapgen_object("voxelmanip")
|
|
local data = vm:get_data()
|
|
local area = VoxelArea:new{MinEdge=emin, MaxEdge=emax}
|
|
|
|
pr = PseudoRandom(seed+33)
|
|
local tab,num = {},1
|
|
local trees,num_trees = {},1
|
|
|
|
--local perlin1 = minetest.get_perlin(13,3, 0.5, 50) --Get map specific perlin
|
|
--local perlin2 = minetest.get_perlin(133,3, 0.5, 10)
|
|
--local perlin3 = minetest.get_perlin(112,3, 0.5, 5)
|
|
|
|
local side_length = maxp.x - minp.x + 1
|
|
local map_lengths_xyz = {x=side_length, y=side_length, z=side_length}
|
|
|
|
local pmap1 = minetest.get_perlin_map(perlins[1], map_lengths_xyz):get2dMap_flat(minp)
|
|
local pmap2 = minetest.get_perlin_map(perlins[2], map_lengths_xyz):get2dMap_flat(minp)
|
|
local pmap3 = minetest.get_perlin_map(perlins[3], map_lengths_xyz):get2dMap_flat(minp)
|
|
|
|
local forest_possible = maxp.y > f_h_min and minp.y < f_h_max
|
|
|
|
--local pmap_f_bottom = minetest.get_perlin_map(perlins.forest_bottom, map_lengths_xyz):get2dMap_flat(minp)
|
|
local perlin_f_bottom, pmap_f_top
|
|
if forest_possible then
|
|
perlin_f_bottom = minetest.get_perlin(11, 3, 0.8, tmp2)
|
|
pmap_f_top = minetest.get_perlin_map(perlins.forest_top, map_lengths_xyz):get2dMap_flat(minp)
|
|
end
|
|
|
|
local num2, tab2
|
|
if buildings >= 1 then
|
|
num2 = 1
|
|
tab2 = nether_weird_noise({x=minp.x, y=nether.buildings-79, z=minp.z}, pymg, 200, 8, 10, 79)
|
|
end
|
|
|
|
local count = 0
|
|
for z=minp.z, maxp.z do
|
|
for x=minp.x, maxp.x do
|
|
|
|
count = count+1
|
|
|
|
local test = pmap1[count]+1
|
|
local test2 = pmap2[count]
|
|
local test3 = math.abs(pmap3[count])
|
|
|
|
local t = math.floor(test*3+0.5)
|
|
|
|
if test2 < 0 then
|
|
h = math.floor(test2*3+0.5)-1
|
|
else
|
|
h = 3+t+pr:next(0,NETHER_RANDOM)
|
|
end
|
|
|
|
local generate_vine = false
|
|
if test3 >= 0.72+pr:next(0,NETHER_RANDOM)/10
|
|
and pr:next(0,NETHER_RANDOM) == 1 then
|
|
generate_vine = true
|
|
end
|
|
|
|
local bottom = NETHER_BOTTOM+h
|
|
local top = nether_middle-pr:next(0,NETHER_RANDOM)+t
|
|
|
|
local py_h = 0
|
|
local difn, noisp, py_h_g
|
|
if buildings >= 1 then
|
|
py_h = tab2[num2].y
|
|
num2 = num2+1
|
|
|
|
difn = nether.buildings-py_h
|
|
if difn == 5 then
|
|
noisp = 1
|
|
elseif difn < 5 then
|
|
noisp = 2
|
|
end
|
|
py_h_g = nether.buildings-7
|
|
end
|
|
|
|
if buildings == 1
|
|
and noisp then
|
|
if noisp == 1 then
|
|
for y=minp.y, maxp.y do
|
|
local p_addpos = area:index(x, y, z)
|
|
data[p_addpos] = c.netherrack_brick
|
|
end
|
|
else
|
|
for y=minp.y, maxp.y do
|
|
local p_addpos = area:index(x, y, z)
|
|
data[p_addpos] = c.lava
|
|
end
|
|
end
|
|
else
|
|
|
|
local r_structure = pr:next(1,nether_structure_freq)
|
|
local r_shroom = pr:next(1,NETHER_SHROOM_FREQ)
|
|
local r_glowstone = pr:next(0,GLOWSTONE_FREQ_ROOF)
|
|
local r_vine_length = pr:next(1,vine_maxlength)
|
|
|
|
local f_bottom, f_top, is_forest, f_h_dirt
|
|
if forest_possible then
|
|
local p = {x=math.floor(x/f_bottom_scale), z=math.floor(z/f_bottom_scale)}
|
|
local pstr = p.x.." "..p.z
|
|
if not f_perlins[pstr] then
|
|
f_perlins[pstr] = math.floor(f_h_min+(math.abs(perlin_f_bottom:get2d({x=p.x, y=p.z})+1))*f_yscale_bottom+0.5)
|
|
end
|
|
f_bottom = f_perlins[pstr]+math.random(0,f_bottom_scale-1)
|
|
f_top = math.floor(f_h_max-(pmap_f_top[count]+1)*f_yscale_top+0.5)
|
|
is_forest = f_bottom < f_top
|
|
f_h_dirt = f_bottom-pr:next(0,1)
|
|
end
|
|
|
|
for y=minp.y, maxp.y do
|
|
local p_addpos = area:index(x, y, z)
|
|
local d_p_addp = data[p_addpos]
|
|
--if py_h >= maxp.y-4 then
|
|
if y <= py_h
|
|
and noisp then
|
|
if noisp == 1 then
|
|
data[p_addpos] = c.netherrack_brick
|
|
elseif noisp == 2 then
|
|
if y == py_h then
|
|
data[p_addpos] = c.netherrack_brick
|
|
elseif y == py_h_g
|
|
and pr:next(1,3) <= 2 then
|
|
data[p_addpos] = c.netherrack
|
|
elseif y <= py_h_g then
|
|
data[p_addpos] = c.lava
|
|
else
|
|
data[p_addpos] = c.air
|
|
end
|
|
end
|
|
elseif d_p_addp ~= c.air then
|
|
|
|
if is_forest
|
|
and y == f_bottom then
|
|
data[p_addpos] = c.nether_dirt_top
|
|
elseif is_forest
|
|
and y < f_bottom
|
|
and y >= f_h_dirt then
|
|
data[p_addpos] = c.nether_dirt
|
|
elseif is_forest
|
|
and y == f_h_dirt-1 then
|
|
data[p_addpos] = c.nether_dirt_bottom
|
|
elseif is_forest
|
|
and y == f_h_dirt+1 then
|
|
if pr:next(1,tree_rarity) == 1 then
|
|
trees[num_trees] = {x=x, y=y, z=z}
|
|
num_trees = num_trees+1
|
|
elseif pr:next(1,mushroom_rarity) == 1 then
|
|
data[p_addpos] = c.nether_shroom
|
|
elseif pr:next(1,glowflower_rarity) == 1 then
|
|
data[p_addpos] = c.glowflower
|
|
elseif pr:next(1,grass_rarity) == 1 then
|
|
data[p_addpos] = c.nether_grass[pr:next(1,3)]
|
|
else
|
|
data[p_addpos] = c.air
|
|
end
|
|
elseif is_forest
|
|
and y > f_bottom
|
|
and y < f_top then
|
|
if not table_contains(
|
|
{c.nether_tree, c.nether_tree_corner, c.nether_leaves, c.nether_fruit},
|
|
d_p_addp
|
|
) then
|
|
data[p_addpos] = c.air
|
|
end
|
|
|
|
elseif y <= NETHER_BOTTOM then
|
|
if y <= bottom then
|
|
data[p_addpos] = return_nether_ore(d_p_addp, true)
|
|
else
|
|
data[p_addpos] = c.lava
|
|
end
|
|
elseif r_structure == 1
|
|
and y == bottom then
|
|
tab[num] = {x=x, y=y-1, z=z}
|
|
num = num+1
|
|
elseif y <= bottom then
|
|
if pr:next(1,LAVA_FREQ) == 1 then
|
|
data[p_addpos] = c.lava
|
|
else
|
|
data[p_addpos] = return_nether_ore(d_p_addp, false)
|
|
end
|
|
elseif r_shroom == 1
|
|
and r_structure ~= 1
|
|
and y == bottom+1 then
|
|
data[p_addpos] = c.nether_shroom
|
|
elseif (y == top and r_glowstone == 1) then
|
|
data[p_addpos] = c.glowstone
|
|
elseif y >= top then
|
|
data[p_addpos] = return_nether_ore(d_p_addp, true)
|
|
elseif y <= top-1
|
|
and generate_vine
|
|
and y >= top-r_vine_length then
|
|
data[p_addpos] = c.nether_vine
|
|
else
|
|
data[p_addpos] = c.air
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
vm:set_data(data)
|
|
-- vm:set_lighting(12)
|
|
-- vm:calc_lighting()
|
|
-- vm:update_liquids()
|
|
vm:write_to_map()
|
|
|
|
nether:inform("nodes set", 2, t1)
|
|
|
|
local t2 = os.clock()
|
|
|
|
if structures_enabled then --Trees:
|
|
for _,v in ipairs(tab) do
|
|
nether.grow_netherstructure(v, true)
|
|
end
|
|
end
|
|
|
|
if forest_possible then --Trees:
|
|
for _,v in ipairs(trees) do
|
|
nether.grow_tree(v, true)
|
|
end
|
|
end
|
|
|
|
nether:inform("trees set", 2, t2)
|
|
|
|
t2 = os.clock()
|
|
fix_light(minp, maxp)
|
|
|
|
nether:inform("map updated", 2, t2)
|
|
|
|
nether:inform("done", 1, t1)
|
|
end)
|
|
|
|
|
|
function nether.grow_netherstructure(pos, generated)
|
|
local t1 = os.clock()
|
|
|
|
if not contents_defined then
|
|
define_contents()
|
|
contents_defined = true
|
|
end
|
|
|
|
if not pos.x then print(dump(pos))
|
|
nether:inform("Error: "..dump(pos), 1)
|
|
return
|
|
end
|
|
|
|
local height = 6
|
|
local manip = minetest.get_voxel_manip()
|
|
local area = r_area(manip, 2, height, pos)
|
|
local nodes = manip:get_data()
|
|
|
|
for i = 0, height-1 do
|
|
nodes[area:index(pos.x, pos.y+i, pos.z)] = c.blood_stem
|
|
end
|
|
|
|
for i = -1,1 do
|
|
for j = -1,1 do
|
|
nodes[area:index(pos.x+i, pos.y+height, pos.z+j)] = c.blood_top
|
|
end
|
|
end
|
|
|
|
for k = -1, 1, 2 do
|
|
for l = -2+1, 2 do
|
|
local p1 = {pos.x+2*k, pos.y+height, pos.z-l*k}
|
|
local p2 = {pos.x+l*k, pos.y+height, pos.z+2*k}
|
|
local udat = c.blood_top
|
|
if math.random(2) == 1 then
|
|
nodes[area:index(p1[1], p1[2], p1[3])] = c.blood_top
|
|
nodes[area:index(p2[1], p2[2], p2[3])] = c.blood_top
|
|
udat = c.blood
|
|
end
|
|
nodes[area:index(p1[1], p1[2]-1, p1[3])] = udat
|
|
nodes[area:index(p2[1], p2[2]-1, p2[3])] = udat
|
|
end
|
|
for l = 0, 1 do
|
|
for _,p in ipairs({
|
|
{pos.x+k, pos.y+height-1, pos.z-l*k},
|
|
{pos.x+l*k, pos.y+height-1, pos.z+k},
|
|
}) do
|
|
if math.random(2) == 1 then
|
|
nodes[area:index(p[1], p[2], p[3])] = c.nether_apple
|
|
--elseif math.random(10) == 1 then
|
|
-- nodes[area:index(p[1], p[2], p[3])] = c.apple
|
|
end
|
|
end
|
|
end
|
|
end
|
|
set_vm_data(manip, nodes, pos, t1, "blood", generated)
|
|
end
|
|
|
|
|
|
local function update_minmax(min, max, p)
|
|
min.x = math.min(min.x, p.x)
|
|
max.x = math.max(max.x, p.x)
|
|
min.z = math.min(min.z, p.z)
|
|
max.z = math.max(max.z, p.z)
|
|
return min, max
|
|
end
|
|
|
|
local fruit_chances = {}
|
|
for y = -2,1 do --like a hyperbola
|
|
fruit_chances[y] = math.floor(-4/(y-2)+0.5)
|
|
end
|
|
|
|
local dirs = {
|
|
{-1, 0, 12, 19},
|
|
{1, 0, 12, 13},
|
|
{0, 1, 4},
|
|
{0, -1, 4, 10},
|
|
}
|
|
|
|
local h_max = 26
|
|
local h_stem_min = 3
|
|
local h_stem_max = 7
|
|
local h_arm_min = 2
|
|
local h_arm_max = 6
|
|
local r_arm_min = 1
|
|
local r_arm_max = 5
|
|
local fruit_rarity = 25 --a bigger number results in less fruits
|
|
local leaf_thickness = 3 --a bigger number results in more blank trees
|
|
|
|
local h_trunk_max = h_max-h_arm_max
|
|
|
|
function nether.grow_tree(pos, generated)
|
|
local t1 = os.clock()
|
|
|
|
if not contents_defined then
|
|
define_contents()
|
|
contents_defined = true
|
|
end
|
|
|
|
local min = vector.new(pos)
|
|
local max = vector.new(pos)
|
|
min.y = min.y-1
|
|
max.y = max.y+h_max
|
|
|
|
local trunks = {}
|
|
local trunk_corners = {}
|
|
local h_stem = math.random(h_stem_min, h_stem_max)
|
|
local stems = {{x=pos.x, y=pos.y+h_stem, z=pos.z}}
|
|
local fi
|
|
while not fi do
|
|
for n,p in pairs(stems) do
|
|
local used_dirs = {}
|
|
for _,dir in pairs(dirs) do
|
|
if math.random(1,2) == 1 then
|
|
table.insert(used_dirs, dir)
|
|
end
|
|
end
|
|
if not used_dirs[1] then
|
|
local dir1 = math.random(4)
|
|
local dir2 = math.random(3)
|
|
if dir1 <= dir2 then
|
|
dir2 = dir2+1
|
|
end
|
|
used_dirs[1] = dirs[dir1]
|
|
used_dirs[2] = dirs[dir2]
|
|
end
|
|
for _,dir in pairs(used_dirs) do
|
|
local p = vector.new(p)
|
|
local r = math.random(r_arm_min, r_arm_max)
|
|
for j = 1,r do
|
|
local x = p.x+j*dir[1]
|
|
local z = p.z+j*dir[2]
|
|
trunks[x.." "..p.y.." "..z] = dir[3]
|
|
end
|
|
r = r+1
|
|
p.x = p.x+r*dir[1]
|
|
p.z = p.z+r*dir[2]
|
|
trunk_corners[p.x.." "..p.y.." "..p.z] = dir[4] or dir[3]
|
|
local h = math.random(h_arm_min, h_arm_max)
|
|
for i = 1,h do
|
|
trunks[p.x.." "..p.y+i.." "..p.z] = true
|
|
end
|
|
p.y = p.y+h
|
|
table.insert(stems, p)
|
|
end
|
|
if p.y > pos.y+h_trunk_max then
|
|
fi = true
|
|
break
|
|
end
|
|
stems[n] = nil
|
|
end
|
|
end
|
|
local leaves = {}
|
|
local fruits = {}
|
|
local trunk_ps = {}
|
|
local count = 0
|
|
for n,par in pairs(trunks) do
|
|
local p = {}
|
|
p.x, p.y, p.z = unpack(string.split(n, " "))
|
|
if par ~= true then
|
|
p.par = par
|
|
end
|
|
table.insert(trunk_ps, p)
|
|
count = count+1
|
|
if count > leaf_thickness then
|
|
count = 0
|
|
for y = -2,2 do
|
|
local fruit_chance = fruit_chances[y]
|
|
for z = -2,2 do
|
|
for x = -2,2 do
|
|
local dist = math.sqrt(x*x+y*y+z*z)
|
|
if math.floor(dist) ~= 0
|
|
and math.random(1, dist) == 1 then
|
|
local pstr = p.x+x.." "..p.y+y.." "..p.z+z
|
|
if not trunks[pstr] then
|
|
if math.random(1, fruit_rarity) == 1
|
|
and fruit_chance
|
|
and math.random(1, fruit_chance) == 1 then
|
|
fruits[pstr] = true
|
|
else
|
|
leaves[pstr] = true
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local leaf_ps = {}
|
|
for n,_ in pairs(leaves) do
|
|
local p = {}
|
|
p.x, p.y, p.z = unpack(string.split(n, " "))
|
|
table.insert(leaf_ps, p)
|
|
min, max = update_minmax(min, max, p)
|
|
end
|
|
|
|
for n,_ in pairs(trunks) do
|
|
local p = {}
|
|
p.x, _, p.z = unpack(string.split(n, " "))
|
|
min, max = update_minmax(min, max, p)
|
|
end
|
|
|
|
for i = -1,h_stem+1 do
|
|
table.insert(trunk_ps, {x=pos.x, y=pos.y+i, z=pos.z, par=0})
|
|
end
|
|
|
|
local trunk_corner_ps = {}
|
|
for n,par in pairs(trunk_corners) do
|
|
local p = {}
|
|
p.x, p.y, p.z = unpack(string.split(n, " "))
|
|
p.par = par
|
|
table.insert(trunk_corner_ps, p)
|
|
end
|
|
|
|
local fruit_ps = {}
|
|
for n,_ in pairs(fruits) do
|
|
local p = {}
|
|
p.x, p.y, p.z = unpack(string.split(n, " "))
|
|
table.insert(fruit_ps, p)
|
|
end
|
|
|
|
local manip = minetest.get_voxel_manip()
|
|
local emerged_pos1, emerged_pos2 = manip:read_from_map(min, max)
|
|
local area = VoxelArea:new({MinEdge=emerged_pos1, MaxEdge=emerged_pos2})
|
|
local nodes = manip:get_data()
|
|
local param2s = manip:get_param2_data()
|
|
|
|
for _,p in pairs(leaf_ps) do
|
|
p = area:indexp(p)
|
|
if nodes[p] == c.air then
|
|
nodes[p] = c.nether_leaves
|
|
param2s[p] = math.random(0,44)
|
|
end
|
|
end
|
|
|
|
for _,p in pairs(fruit_ps) do
|
|
p = area:indexp(p)
|
|
if nodes[p] == c.air then
|
|
nodes[p] = c.nether_apple
|
|
--param2s[p] = math.random(0,44)
|
|
end
|
|
end
|
|
|
|
for _,p in pairs(trunk_ps) do
|
|
local par = p.par
|
|
p = area:indexp(p)
|
|
if par then
|
|
param2s[p] = par
|
|
end
|
|
nodes[p] = c.nether_tree
|
|
end
|
|
|
|
for _,p in pairs(trunk_corner_ps) do
|
|
local par = p.par
|
|
p = area:indexp(p)
|
|
nodes[p] = c.nether_tree_corner
|
|
param2s[p] = par
|
|
end
|
|
|
|
--calculating took ca. 0.07 - 0.18 [s]
|
|
manip:set_data(nodes)
|
|
manip:set_param2_data(param2s)
|
|
manip:write_to_map()
|
|
local spam = 2
|
|
if generated then
|
|
spam = 3
|
|
end
|
|
nether:inform("a nether tree grew at ("..pos.x.."|"..pos.y.."|"..pos.z..")", spam, t1)
|
|
if not generated then
|
|
local t1 = os.clock()
|
|
manip:update_map()
|
|
nether:inform("map updated", spam, t1)
|
|
end
|
|
end
|
|
|
|
|
|
--abms
|
|
|
|
minetest.register_abm({
|
|
nodenames = {"nether:sapling"},
|
|
neighbors = {"nether:blood_top"},
|
|
interval = 20,
|
|
chance = 8,
|
|
action = function(pos)
|
|
local under = {x=pos.x, y=pos.y-1, z=pos.z}
|
|
if minetest.get_node_light(pos) < 4
|
|
and minetest.get_node({x=pos.x, y=pos.y+1, z=pos.z}).name == "air"
|
|
and minetest.get_node(under).name == "nether:blood_top" then
|
|
nether.grow_netherstructure(under)
|
|
end
|
|
end
|
|
})
|
|
|
|
minetest.register_abm({
|
|
nodenames = {"nether:tree_sapling"},
|
|
neighbors = {"group:nether_dirt"},
|
|
interval = 864,
|
|
chance = 100,
|
|
action = function(pos)
|
|
if minetest.get_node({x=pos.x, y=pos.y+2, z=pos.z}).name == "air"
|
|
and minetest.get_node({x=pos.x, y=pos.y+1, z=pos.z}).name == "air" then
|
|
local udata = minetest.registered_nodes[minetest.get_node({x=pos.x, y=pos.y-1, z=pos.z}).name]
|
|
if udata
|
|
and udata.groups
|
|
and udata.groups.nether_dirt then
|
|
nether.grow_tree(pos)
|
|
end
|
|
end
|
|
end
|
|
})
|
|
|
|
minetest.register_abm({
|
|
nodenames = {"nether:netherrack_soil"},
|
|
neighbors = {"air"},
|
|
interval = 20,
|
|
chance = 20,
|
|
action = function(pos, node)
|
|
local par2 = node.param2
|
|
if par2 == 0 then
|
|
return
|
|
end
|
|
pos.y = pos.y+1
|
|
if (minetest.get_node_light(pos) or 16) > 7 then --mushrooms grow at dark places
|
|
return
|
|
end
|
|
if minetest.get_node(pos).name == "air" then
|
|
minetest.set_node(pos, {name="riesenpilz:nether_shroom"})
|
|
pos.y = pos.y-1
|
|
minetest.set_node(pos, {name="nether:netherrack_soil", param2=par2-1})
|
|
end
|
|
end
|
|
})
|
|
|
|
local function grass_allowed(pos)
|
|
local nd = minetest.get_node(pos).name
|
|
if nd == "air" then
|
|
return true
|
|
end
|
|
if nd == "ignore" then
|
|
return 0
|
|
end
|
|
local data = minetest.registered_nodes[nd]
|
|
local drawtype = data.drawtype
|
|
if drawtype
|
|
and drawtype ~= "normal"
|
|
and drawtype ~= "liquid"
|
|
and drawtype ~= "flowingliquid" then
|
|
return true
|
|
end
|
|
local light = data.light_source
|
|
if light
|
|
and light > 0 then
|
|
return true
|
|
end
|
|
return false
|
|
end
|
|
|
|
minetest.register_abm({
|
|
nodenames = {"nether:dirt"},
|
|
interval = 20,
|
|
chance = 9,
|
|
action = function(pos)
|
|
local allowed = grass_allowed({x=pos.x, y=pos.y+1, z=pos.z})
|
|
if allowed == 0 then
|
|
return
|
|
end
|
|
if allowed then
|
|
minetest.set_node(pos, {name="nether:dirt_top"})
|
|
end
|
|
end
|
|
})
|
|
|
|
minetest.register_abm({
|
|
nodenames = {"nether:dirt_top"},
|
|
interval = 30,
|
|
chance = 9,
|
|
action = function(pos)
|
|
local allowed = grass_allowed({x=pos.x, y=pos.y+1, z=pos.z})
|
|
if allowed == 0 then
|
|
return
|
|
end
|
|
if not allowed then
|
|
minetest.set_node(pos, {name="nether:dirt"})
|
|
end
|
|
end
|
|
})
|
|
|
|
|
|
minetest.register_privilege("nether", "Allows sending players to nether and extracting them")
|
|
|
|
dofile(path.."/crafting.lua")
|
|
dofile(path.."/portal.lua")
|
|
dofile(path.."/guide.lua")
|
|
|
|
nether:inform("loaded!", 1)
|