mg-cd2025/snow.lua
2017-12-10 14:02:32 +01:00

177 lines
5.7 KiB
Lua

local function getmp(x)
return (80 * math.floor((x + 32) / 80)) - 32
end
local function get_minp(pos)
return {x=getmp(pos.x), y=getmp(pos.y), z=getmp(pos.z)}
end
--[[local snow_timer = 0
local snow_range = 1
minetest.register_globalstep(function(dtime)
snow_timer = snow_timer + dtime
if snow_timer < 1 then return end
snow_timer = 0
local noise_temperature_raw = minetest.get_perlin(763, 7, 0.5, 512)
local noise_humidity_raw = minetest.get_perlin(834, 7, 0.5, 512)
for _, player in ipairs(minetest.get_connected_players()) do
local pos = player:getpos()
if pos.y > -50 and pos.y < 150 then
--local to_update = {}
--local to_update_index = 1
local minp = get_minp(pos)
local biome_table = get_biome_table(minp, noise_humidity_raw, noise_temperature_raw, snow_range+1)
local vm = minetest.get_voxel_manip()
local eminp, emaxp = vm:read_from_map({x=minp.x-80*snow_range, y=-50, z=minp.z-80*snow_range}, {x=minp.x+79+80*snow_range, y=150, z=minp.z+79+80*snow_range})
--print(dump(eminp), dump(emaxp))
local a = VoxelArea:new{MinEdge=eminp, MaxEdge=emaxp}
local data = vm:get_data()
for i = 1, 80*(snow_range+1)*(snow_range+1) do
local x = math.random(eminp.x, emaxp.x)
local z = math.random(eminp.z, emaxp.z)
local biome = get_nearest_biome(biome_table, x, z)
if biome.t < -0.4 then
local y = emaxp.y
while y >= eminp.y and data[a:index(x, y, z)] == c_air do
y = y - 1
end
if y >= eminp.y and y < emaxp.y then
if data[a:index(x, y, z)] == c_snow then
if minetest.add_node_level({x=x, y=y, z=z}, 7) ~= 0 then
--data[a:index(x, y+1, z)] = c_snow
minetest.set_node({x=x, y=y+1, z=z}, {name = "default:snow"})
end
else
--data[a:index(x, y+1, z)] = c_snow
minetest.set_node({x=x, y=y+1, z=z}, {name = "default:snow"})
end
--to_update[to_update_index] = {x=x,y=y+1,z=z}
--to_update_index = to_update_index+1
end
end
end
--vm:set_data(data)
--vm:write_to_map()
--for _, pos in ipairs(to_update) do
-- nodeupdate(pos)
--end
end
end
end)]]
local function add_snow_level(pos)
local level = minetest.get_node_level(pos)
if level <= 28 then
minetest.add_node_level(pos, 7)
return 0
else
return 7
end
end
local is_snowing = false
local to_snow = {}
local to_snow_index = 0
local SNOW_RANGE = 16
local SNOW_BLOCK_SIZE = 4
local SNOW_STEPS = 10
local SNOW_TIMES_PER_BLOCK = 4
minetest.register_globalstep(function(dtime)
if math.random(1, 5000) == 1 then
is_snowing = not is_snowing
end
if to_snow_index == 0 then
if is_snowing then
for _, player in ipairs(minetest.get_connected_players()) do
local pos = player:getpos()
if pos.y > -50 and pos.y < 150 then
local x = SNOW_BLOCK_SIZE*math.floor((pos.x/SNOW_BLOCK_SIZE)+0.5)
local z = SNOW_BLOCK_SIZE*math.floor((pos.z/SNOW_BLOCK_SIZE)+0.5)
for xi = -SNOW_RANGE, SNOW_RANGE do
for zi = -SNOW_RANGE, SNOW_RANGE do
to_snow_index = to_snow_index + 1
to_snow[to_snow_index] = {x=x+SNOW_BLOCK_SIZE*xi, y=0, z=z+SNOW_BLOCK_SIZE*zi}
end
end
end
end
end
else
local noise_temperature_raw = minetest.get_perlin(763, 7, 0.5, 512)
local noise_humidity_raw = minetest.get_perlin(834, 7, 0.5, 512)
for s = 1, math.min(SNOW_STEPS, to_snow_index) do
local pos = to_snow[to_snow_index]
--print(dump(pos))
to_snow[to_snow_index] = nil
to_snow_index = to_snow_index - 1
local minp = get_minp(pos)
local biome_table = get_biome_table(minp, noise_humidity_raw, noise_temperature_raw)
local vm = minetest.get_voxel_manip()
local eminp, emaxp = vm:read_from_map({x=pos.x, y=-50, z=pos.z}, {x=pos.x + SNOW_BLOCK_SIZE - 1, y=150, z=pos.z + SNOW_BLOCK_SIZE - 1})
--print(eminp.y, emaxp.y)
--print(dump(eminp),dump(emaxp))
local a = VoxelArea:new{MinEdge=eminp, MaxEdge=emaxp}
local data = vm:get_data()
for i = 1, SNOW_TIMES_PER_BLOCK do
local x = math.random(pos.x, pos.x + SNOW_BLOCK_SIZE - 1)
local z = math.random(pos.z, pos.z + SNOW_BLOCK_SIZE - 1)
local biome = get_nearest_biome(biome_table, x, z)
if biome.t < -0.4 then
local y = emaxp.y
while y >= eminp.y and (data[a:index(x, y, z)] == c_air or data[a:index(x, y, z)] == c_ignore) do
y = y - 1
end
if y >= eminp.y and y < emaxp.y then
if data[a:index(x, y, z)] == c_snow then
--if minetest.add_node_level({x=x, y=y, z=z}, 7) ~= 0 then
if add_snow_level({x=x, y=y, z=z}) ~= 0 then
-- do nothing, max stack
--minetest.set_node({x=x, y=y, z=z}, {name = "default:snowblock"})
end
elseif data[a:index(x, y, z)] == c_snowblock or data[a:index(x, y, z)] == c_water then
-- do nothing
else
minetest.set_node({x=x, y=y+1, z=z}, {name = "default:snow"})
end
end
end
end
end
end
end)
minetest.register_abm({
nodenames = {"mg:pineleaves"},
interval = 2,
chance = 4,
action = function(pos, node)
local above = {x=pos.x, y=pos.y+1, z=pos.z}
if minetest.get_node(above).name == "default:snow" then
local level = minetest.get_node_level(above)
if level >= 14 then
for i = 1, 15 do
local p = {x=pos.x, y=pos.y-i, z=pos.z}
local n = minetest.get_node(p).name
if n ~= "air" and n ~= "mg:pineleaves" then
if n == "default:snow" then
--if minetest.add_node_level(p, 7) == 0 then
if add_snow_level(p) == 0 then
minetest.add_node_level(above, -7)
end
else
local above_p = {x=p.x, y=p.y+1, z=p.z}
if minetest.get_node(above_p).name == "air" then
minetest.set_node(above_p, {name = "default:snow"})
minetest.add_node_level(above, -7)
end
end
return
end
end
end
end
end
})