Compare commits
23 Commits
b9fcd16b11
...
164177c5f4
Author | SHA1 | Date |
---|---|---|
kno10 | 164177c5f4 | |
kno10 | 590606b667 | |
kno10 | 30d5eb537c | |
kno10 | 03fddcf91b | |
kno10 | ac3b1a1798 | |
kno10 | ab57507414 | |
kno10 | b06d44aeec | |
kno10 | 28d2ca816d | |
kno10 | 2b45aa28c5 | |
kno10 | 29355d30db | |
kno10 | 77186b4a3a | |
kno10 | 13820496db | |
Mikita Wiśniewski | 41b188caea | |
kno10 | ae7995d195 | |
kno10 | e293cbe631 | |
the-real-herowl | fd6cac5f0c | |
teknomunk | e864cc19ed | |
teknomunk | 66c3c014a1 | |
teknomunk | 7807093b50 | |
teknomunk | f6c3f4bd16 | |
teknomunk | 96a03b1923 | |
teknomunk | 2145470f63 | |
teknomunk | 2ca0ccd8fe |
|
@ -928,8 +928,6 @@ end
|
||||||
-- falling and fall damage
|
-- falling and fall damage
|
||||||
-- returns true if mob died
|
-- returns true if mob died
|
||||||
function mob_class:falling(pos, moveresult)
|
function mob_class:falling(pos, moveresult)
|
||||||
if moveresult and moveresult.touching_ground then return false end
|
|
||||||
|
|
||||||
if self.fly and self.state ~= "die" then
|
if self.fly and self.state ~= "die" then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
@ -952,7 +950,13 @@ function mob_class:falling(pos, moveresult)
|
||||||
new_acceleration = vector.new(0, DEFAULT_FALL_SPEED, 0)
|
new_acceleration = vector.new(0, DEFAULT_FALL_SPEED, 0)
|
||||||
elseif v.y <= 0 and v.y > self.fall_speed then
|
elseif v.y <= 0 and v.y > self.fall_speed then
|
||||||
-- fall downwards at set speed
|
-- fall downwards at set speed
|
||||||
|
if moveresult and moveresult.touching_ground then
|
||||||
|
-- when touching ground, retain a minimal gravity to keep the touching_ground flag
|
||||||
|
-- but also to not get upwards acceleration with large dtime when on bouncy ground
|
||||||
|
new_acceleration = vector.new(0, self.fall_speed * 0.01, 0)
|
||||||
|
else
|
||||||
new_acceleration = vector.new(0, self.fall_speed, 0)
|
new_acceleration = vector.new(0, self.fall_speed, 0)
|
||||||
|
end
|
||||||
else
|
else
|
||||||
-- stop accelerating once max fall speed hit
|
-- stop accelerating once max fall speed hit
|
||||||
new_acceleration =vector.zero()
|
new_acceleration =vector.zero()
|
||||||
|
|
|
@ -72,17 +72,23 @@ local axolotl = {
|
||||||
fly = true,
|
fly = true,
|
||||||
fly_in = { "mcl_core:water_source", "mclx_core:river_water_source" },
|
fly_in = { "mcl_core:water_source", "mclx_core:river_water_source" },
|
||||||
breathes_in_water = true,
|
breathes_in_water = true,
|
||||||
jump = true,
|
jump = false, -- would get them out of the water too often
|
||||||
damage = 2,
|
damage = 2,
|
||||||
reach = 2,
|
reach = 2,
|
||||||
attack_type = "dogfight",
|
attack_type = "dogfight",
|
||||||
attack_animals = true,
|
attack_animals = true,
|
||||||
specific_attack = {
|
specific_attack = {
|
||||||
"extra_mobs_cod",
|
"mobs_mc:cod",
|
||||||
"extra_mobs_glow_squid",
|
"mobs_mc:glow_squid",
|
||||||
"extra_mobs_salmon",
|
"mobs_mc:salmon",
|
||||||
"extra_mobs_tropical_fish",
|
"mobs_mc:tropical_fish",
|
||||||
"mobs_mc_squid"
|
"mobs_mc:squid",
|
||||||
|
"mobs_mc:zombie", -- todo: only drowned?
|
||||||
|
"mobs_mc:baby_zombie",
|
||||||
|
"mobs_mc:husk",
|
||||||
|
"mobs_mc:baby_husk",
|
||||||
|
"mobs_mc:guardian_elder",
|
||||||
|
"mobs_mc:guardian",
|
||||||
},
|
},
|
||||||
runaway = true,
|
runaway = true,
|
||||||
}
|
}
|
||||||
|
|
|
@ -157,6 +157,7 @@ function mcl_weather.rain.clear()
|
||||||
mcl_weather.rain.remove_sound(player)
|
mcl_weather.rain.remove_sound(player)
|
||||||
mcl_weather.rain.remove_player(player)
|
mcl_weather.rain.remove_player(player)
|
||||||
mcl_weather.remove_spawners_player(player)
|
mcl_weather.remove_spawners_player(player)
|
||||||
|
player:set_clouds({color="#FFF0EF"})
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
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 NIGHT_VISION_RATIO = 0.45
|
local NIGHT_VISION_RATIO = 0.45
|
||||||
|
local DEBUG = false
|
||||||
|
|
||||||
-- Settings
|
-- Settings
|
||||||
local minimum_update_interval = { 250e3 }
|
local minimum_update_interval = { 250e3 }
|
||||||
|
@ -190,8 +191,8 @@ end
|
||||||
|
|
||||||
function skycolor_utils.convert_to_rgb(minval, maxval, current_val, colors)
|
function skycolor_utils.convert_to_rgb(minval, maxval, current_val, colors)
|
||||||
-- Clamp current_val to valid range
|
-- Clamp current_val to valid range
|
||||||
current_val = math.min(minval, current_val)
|
current_val = math.max(minval, current_val)
|
||||||
current_val = math.max(maxval, current_val)
|
current_val = math.min(maxval, current_val)
|
||||||
|
|
||||||
-- Rescale current_val from a number between minval and maxval to a number between 1 and #colors
|
-- Rescale current_val from a number between minval and maxval to a number between 1 and #colors
|
||||||
local scaled_value = (current_val - minval) / (maxval - minval) * (#colors - 1) + 1.0
|
local scaled_value = (current_val - minval) / (maxval - minval) * (#colors - 1) + 1.0
|
||||||
|
@ -199,7 +200,7 @@ function skycolor_utils.convert_to_rgb(minval, maxval, current_val, colors)
|
||||||
-- Get the first color's values
|
-- Get the first color's values
|
||||||
local index1 = math.floor(scaled_value)
|
local index1 = math.floor(scaled_value)
|
||||||
local color1 = colors[index1]
|
local color1 = colors[index1]
|
||||||
local frac1 = scaled_value - index1
|
local frac1 = 1.0 - (scaled_value - index1)
|
||||||
|
|
||||||
-- Get the second color's values
|
-- Get the second color's values
|
||||||
local index2 = math.min(index1 + 1, #colors) -- clamp to maximum color index (will occur if index1 == #colors)
|
local index2 = math.min(index1 + 1, #colors) -- clamp to maximum color index (will occur if index1 == #colors)
|
||||||
|
@ -207,11 +208,32 @@ function skycolor_utils.convert_to_rgb(minval, maxval, current_val, colors)
|
||||||
local color2 = colors[index2]
|
local color2 = colors[index2]
|
||||||
|
|
||||||
-- Interpolate between color1 and color2
|
-- Interpolate between color1 and color2
|
||||||
return {
|
local res = {
|
||||||
r = math.floor(frac1 * color1.r + frac2 * color2.r),
|
r = math.floor(frac1 * color1.r + frac2 * color2.r),
|
||||||
g = math.floor(frac1 * color1.g + frac2 * color2.g),
|
g = math.floor(frac1 * color1.g + frac2 * color2.g),
|
||||||
b = math.floor(frac1 * color1.b + frac2 * color2.b),
|
b = math.floor(frac1 * color1.b + frac2 * color2.b),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if DEBUG then
|
||||||
|
minetest.log(dump({
|
||||||
|
minval = minval,
|
||||||
|
maxval = maxval,
|
||||||
|
current_val = current_val,
|
||||||
|
colors = colors,
|
||||||
|
res = res,
|
||||||
|
scaled_value = scaled_value,
|
||||||
|
|
||||||
|
frac1 = frac1,
|
||||||
|
index1 = index1,
|
||||||
|
color1 = color1,
|
||||||
|
|
||||||
|
frac2 = frac2,
|
||||||
|
index2 = index2,
|
||||||
|
color2 = color2,
|
||||||
|
}))
|
||||||
|
end
|
||||||
|
|
||||||
|
return res
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Simple getter. Either returns user given players list or get all connected players if none provided
|
-- Simple getter. Either returns user given players list or get all connected players if none provided
|
||||||
|
|
|
@ -40,18 +40,21 @@ function dimension_handlers.overworld(player, sky_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Use overworld defaults
|
-- Use overworld defaults
|
||||||
local day_color = mcl_weather.skycolor.get_sky_layer_color(0.15)
|
local day_color = mcl_weather.skycolor.get_sky_layer_color(0.5)
|
||||||
local dawn_color = mcl_weather.skycolor.get_sky_layer_color(0.27)
|
local dawn_color = mcl_weather.skycolor.get_sky_layer_color(0.27)
|
||||||
local night_color = mcl_weather.skycolor.get_sky_layer_color(0.1)
|
local night_color = mcl_weather.skycolor.get_sky_layer_color(0.1)
|
||||||
sky_data.sky = {
|
sky_data.sky = {
|
||||||
type = "regular",
|
type = "regular",
|
||||||
sky_color = {
|
sky_color = {
|
||||||
day_sky = day_color,
|
day_sky = day_color or "#7BA4FF",
|
||||||
day_horizon = day_color,
|
day_horizon = day_color or "#C0D8FF",
|
||||||
dawn_sky = dawn_color,
|
dawn_sky = dawn_color or "7BA4FF",
|
||||||
dawn_horizon = dawn_color,
|
dawn_horizon = dawn_color or "#C0D8FF",
|
||||||
night_sky = night_color,
|
night_sky = night_color or "000000",
|
||||||
night_horizon = night_color,
|
night_horizon = night_color or "4A6790",
|
||||||
|
fog_sun_tint = "#ff5f33",
|
||||||
|
fog_moon_tint = nil,
|
||||||
|
fog_tint_type = "custom",
|
||||||
},
|
},
|
||||||
clouds = true,
|
clouds = true,
|
||||||
}
|
}
|
||||||
|
@ -75,18 +78,15 @@ function dimension_handlers.overworld(player, sky_data)
|
||||||
local day_color = mcl_weather.skycolor.get_sky_layer_color(0.5)
|
local day_color = mcl_weather.skycolor.get_sky_layer_color(0.5)
|
||||||
local dawn_color = mcl_weather.skycolor.get_sky_layer_color(0.75)
|
local dawn_color = mcl_weather.skycolor.get_sky_layer_color(0.75)
|
||||||
local night_color = mcl_weather.skycolor.get_sky_layer_color(0)
|
local night_color = mcl_weather.skycolor.get_sky_layer_color(0)
|
||||||
sky_data.sky = {
|
table.update(sky_data.sky.sky_color,{
|
||||||
type = "regular",
|
day_sky = day_color or "#7BA4FF",
|
||||||
sky_color = {
|
day_horizon = day_color or "#C0D8FF",
|
||||||
day_sky = day_color,
|
dawn_sky = dawn_color or "7BA4FF",
|
||||||
day_horizon = day_color,
|
dawn_horizon = dawn_color or "#C0D8FF",
|
||||||
dawn_sky = dawn_color,
|
night_sky = night_color or "000000",
|
||||||
dawn_horizon = dawn_color,
|
night_horizon = night_color or "4A6790",
|
||||||
night_sky = night_color,
|
fog_tint_type = "default",
|
||||||
night_horizon = night_color,
|
})
|
||||||
},
|
|
||||||
clouds = true,
|
|
||||||
}
|
|
||||||
sky_data.sun = {visible = false, sunrise_visible = false}
|
sky_data.sun = {visible = false, sunrise_visible = false}
|
||||||
sky_data.moon = {visible = false}
|
sky_data.moon = {visible = false}
|
||||||
sky_data.stars = {visible = false}
|
sky_data.stars = {visible = false}
|
||||||
|
@ -164,7 +164,8 @@ function dimension_handlers.nether(player, sky_data)
|
||||||
end
|
end
|
||||||
|
|
||||||
function dimension_handlers.void(player, sky_data)
|
function dimension_handlers.void(player, sky_data)
|
||||||
sky_data.sky = { type = "plain",
|
sky_data.sky = {
|
||||||
|
type = "plain",
|
||||||
base_color = "#000000",
|
base_color = "#000000",
|
||||||
clouds = false,
|
clouds = false,
|
||||||
}
|
}
|
||||||
|
|
|
@ -75,6 +75,7 @@ function mcl_weather.has_snow(pos)
|
||||||
end
|
end
|
||||||
|
|
||||||
function mcl_weather.snow.set_sky_box()
|
function mcl_weather.snow.set_sky_box()
|
||||||
|
if mcl_weather.skycolor.current_layer_name() ~= "weather-pack-snow-sky" then
|
||||||
mcl_weather.skycolor.add_layer(
|
mcl_weather.skycolor.add_layer(
|
||||||
"weather-pack-snow-sky",
|
"weather-pack-snow-sky",
|
||||||
{{r=0, g=0, b=0},
|
{{r=0, g=0, b=0},
|
||||||
|
@ -82,6 +83,7 @@ function mcl_weather.snow.set_sky_box()
|
||||||
{r=135, g=135, b=135},
|
{r=135, g=135, b=135},
|
||||||
{r=85, g=86, b=86},
|
{r=85, g=86, b=86},
|
||||||
{r=0, g=0, b=0}})
|
{r=0, g=0, b=0}})
|
||||||
|
end
|
||||||
mcl_weather.skycolor.active = true
|
mcl_weather.skycolor.active = true
|
||||||
for _, player in pairs(get_connected_players()) do
|
for _, player in pairs(get_connected_players()) do
|
||||||
player:set_clouds({color="#ADADADE8"})
|
player:set_clouds({color="#ADADADE8"})
|
||||||
|
|
|
@ -23,6 +23,7 @@ minetest.register_globalstep(function(dtime)
|
||||||
mcl_weather.rain.make_weather()
|
mcl_weather.rain.make_weather()
|
||||||
|
|
||||||
if mcl_weather.thunder.init_done == false then
|
if mcl_weather.thunder.init_done == false then
|
||||||
|
if mcl_weather.skycolor.current_layer_name() ~= "weather-pack-thunder-sky" then
|
||||||
mcl_weather.skycolor.add_layer("weather-pack-thunder-sky", {
|
mcl_weather.skycolor.add_layer("weather-pack-thunder-sky", {
|
||||||
{r=0, g=0, b=0},
|
{r=0, g=0, b=0},
|
||||||
{r=40, g=40, b=40},
|
{r=40, g=40, b=40},
|
||||||
|
@ -30,6 +31,7 @@ minetest.register_globalstep(function(dtime)
|
||||||
{r=40, g=40, b=40},
|
{r=40, g=40, b=40},
|
||||||
{r=0, g=0, b=0},
|
{r=0, g=0, b=0},
|
||||||
})
|
})
|
||||||
|
end
|
||||||
mcl_weather.skycolor.active = true
|
mcl_weather.skycolor.active = true
|
||||||
for _, player in pairs(get_connected_players()) do
|
for _, player in pairs(get_connected_players()) do
|
||||||
player:set_clouds({color="#3D3D3FE8"})
|
player:set_clouds({color="#3D3D3FE8"})
|
||||||
|
|
|
@ -5,9 +5,6 @@
|
||||||
--- Copyright (C) 2022 - 2023, Michieal. See License.txt
|
--- Copyright (C) 2022 - 2023, Michieal. See License.txt
|
||||||
|
|
||||||
-- CONSTS
|
-- CONSTS
|
||||||
local DOUBLE_DROP_CHANCE = 8
|
|
||||||
-- Used everywhere. Often this is just the name, but it makes sense to me as BAMBOO, because that's how I think of it...
|
|
||||||
-- "BAMBOO" goes here.
|
|
||||||
local BAMBOO = "mcl_bamboo:bamboo"
|
local BAMBOO = "mcl_bamboo:bamboo"
|
||||||
local BAMBOO_ENDCAP_NAME = "mcl_bamboo:bamboo_endcap"
|
local BAMBOO_ENDCAP_NAME = "mcl_bamboo:bamboo_endcap"
|
||||||
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
||||||
|
@ -16,7 +13,7 @@ local BAMBOO_PLANK = BAMBOO .. "_plank"
|
||||||
local modname = minetest.get_current_modname()
|
local modname = minetest.get_current_modname()
|
||||||
local S = minetest.get_translator(modname)
|
local S = minetest.get_translator(modname)
|
||||||
local node_sound = mcl_sounds.node_sound_wood_defaults()
|
local node_sound = mcl_sounds.node_sound_wood_defaults()
|
||||||
local pr = PseudoRandom((os.time() + 15766) * 12) -- switched from math.random() to PseudoRandom because the random wasn't very random.
|
local pr = PseudoRandom((os.time() + 15766) * 12)
|
||||||
|
|
||||||
local on_rotate
|
local on_rotate
|
||||||
if minetest.get_modpath("screwdriver") then
|
if minetest.get_modpath("screwdriver") then
|
||||||
|
@ -31,33 +28,7 @@ local bamboo_def = {
|
||||||
paramtype = "light",
|
paramtype = "light",
|
||||||
groups = {handy = 1, axey = 1, choppy = 1, dig_by_piston = 1, plant = 1, non_mycelium_plant = 1, flammable = 3},
|
groups = {handy = 1, axey = 1, choppy = 1, dig_by_piston = 1, plant = 1, non_mycelium_plant = 1, flammable = 3},
|
||||||
sounds = node_sound,
|
sounds = node_sound,
|
||||||
|
drop = BAMBOO,
|
||||||
drop = {
|
|
||||||
max_items = 1,
|
|
||||||
-- From the API:
|
|
||||||
-- max_items: Maximum number of item lists to drop.
|
|
||||||
-- The entries in 'items' are processed in order. For each:
|
|
||||||
-- Item filtering is applied, chance of drop is applied, if both are
|
|
||||||
-- successful the entire item list is dropped.
|
|
||||||
-- Entry processing continues until the number of dropped item lists
|
|
||||||
-- equals 'max_items'.
|
|
||||||
-- Therefore, entries should progress from low to high drop chance.
|
|
||||||
items = {
|
|
||||||
-- Examples:
|
|
||||||
{
|
|
||||||
-- 1 in DOUBLE_DROP_CHANCE chance of dropping.
|
|
||||||
-- Default rarity is '1'.
|
|
||||||
rarity = DOUBLE_DROP_CHANCE,
|
|
||||||
items = {BAMBOO .. " 2"},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
-- 1 in 1 chance of dropping. (Note: this means that it will drop 100% of the time.)
|
|
||||||
-- Default rarity is '1'.
|
|
||||||
rarity = 1,
|
|
||||||
items = {BAMBOO},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
inventory_image = "mcl_bamboo_bamboo_shoot.png",
|
inventory_image = "mcl_bamboo_bamboo_shoot.png",
|
||||||
wield_image = "mcl_bamboo_bamboo_shoot.png",
|
wield_image = "mcl_bamboo_bamboo_shoot.png",
|
||||||
|
@ -86,7 +57,6 @@ local bamboo_def = {
|
||||||
on_rotate = on_rotate,
|
on_rotate = on_rotate,
|
||||||
|
|
||||||
on_place = function(itemstack, placer, pointed_thing)
|
on_place = function(itemstack, placer, pointed_thing)
|
||||||
|
|
||||||
if not pointed_thing then
|
if not pointed_thing then
|
||||||
return itemstack
|
return itemstack
|
||||||
end
|
end
|
||||||
|
@ -241,9 +211,6 @@ local bamboo_def = {
|
||||||
if node_above and ((bamboo_node and bamboo_node > 0) or node_above.name == BAMBOO_ENDCAP_NAME) then
|
if node_above and ((bamboo_node and bamboo_node > 0) or node_above.name == BAMBOO_ENDCAP_NAME) then
|
||||||
minetest.remove_node(new_pos)
|
minetest.remove_node(new_pos)
|
||||||
minetest.sound_play(node_sound.dug, sound_params, true)
|
minetest.sound_play(node_sound.dug, sound_params, true)
|
||||||
if pr:next(1, DOUBLE_DROP_CHANCE) == 1 then
|
|
||||||
minetest.add_item(new_pos, istack)
|
|
||||||
end
|
|
||||||
minetest.add_item(new_pos, istack)
|
minetest.add_item(new_pos, istack)
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
|
|
|
@ -9,8 +9,6 @@ local SIDE_SCAFFOLDING = false
|
||||||
local SIDE_SCAFFOLD_NAME = "mcl_bamboo:scaffolding_horizontal"
|
local SIDE_SCAFFOLD_NAME = "mcl_bamboo:scaffolding_horizontal"
|
||||||
-- ---------------------------------------------------------------------------
|
-- ---------------------------------------------------------------------------
|
||||||
local SCAFFOLDING_NAME = "mcl_bamboo:scaffolding"
|
local SCAFFOLDING_NAME = "mcl_bamboo:scaffolding"
|
||||||
-- Used everywhere. Often this is just the name, but it makes sense to me as BAMBOO, because that's how I think of it...
|
|
||||||
-- "BAMBOO" goes here.
|
|
||||||
local BAMBOO = "mcl_bamboo:bamboo"
|
local BAMBOO = "mcl_bamboo:bamboo"
|
||||||
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
||||||
|
|
||||||
|
|
|
@ -7,8 +7,6 @@
|
||||||
|
|
||||||
-- LOCALS
|
-- LOCALS
|
||||||
local modname = minetest.get_current_modname()
|
local modname = minetest.get_current_modname()
|
||||||
-- Used everywhere. Often this is just the name, but it makes sense to me as BAMBOO, because that's how I think of it...
|
|
||||||
-- "BAMBOO" goes here.
|
|
||||||
local BAMBOO = "mcl_bamboo:bamboo"
|
local BAMBOO = "mcl_bamboo:bamboo"
|
||||||
|
|
||||||
mcl_bamboo = {}
|
mcl_bamboo = {}
|
||||||
|
|
|
@ -5,8 +5,6 @@
|
||||||
--- These are all of the fuel recipes and all of the crafting recipes, consolidated into one place.
|
--- These are all of the fuel recipes and all of the crafting recipes, consolidated into one place.
|
||||||
--- Copyright (C) 2022 - 2023, Michieal. See License.txt
|
--- Copyright (C) 2022 - 2023, Michieal. See License.txt
|
||||||
|
|
||||||
-- Used everywhere. Often this is just the name, but it makes sense to me as BAMBOO, because that's how I think of it...
|
|
||||||
-- "BAMBOO" goes here.
|
|
||||||
local BAMBOO = "mcl_bamboo:bamboo"
|
local BAMBOO = "mcl_bamboo:bamboo"
|
||||||
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
local BAMBOO_PLANK = BAMBOO .. "_plank"
|
||||||
-- Craftings
|
-- Craftings
|
||||||
|
|
|
@ -171,7 +171,8 @@ minetest.register_craft({
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
mcl_farming:add_plant("plant_beetroot", "mcl_farming:beetroot", {"mcl_farming:beetroot_0", "mcl_farming:beetroot_1", "mcl_farming:beetroot_2"}, 68, 3)
|
-- beetroots grow at 2/3rd of the default speed
|
||||||
|
mcl_farming:add_plant("plant_beetroot", "mcl_farming:beetroot", {"mcl_farming:beetroot_0", "mcl_farming:beetroot_1", "mcl_farming:beetroot_2"}, 8.7012, 35)
|
||||||
|
|
||||||
if minetest.get_modpath("doc") then
|
if minetest.get_modpath("doc") then
|
||||||
for i = 1, 2 do
|
for i = 1, 2 do
|
||||||
|
|
|
@ -118,7 +118,7 @@ minetest.register_craft({
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
mcl_farming:add_plant("plant_carrot", "mcl_farming:carrot", {"mcl_farming:carrot_1", "mcl_farming:carrot_2", "mcl_farming:carrot_3", "mcl_farming:carrot_4", "mcl_farming:carrot_5", "mcl_farming:carrot_6", "mcl_farming:carrot_7"}, 25, 20)
|
mcl_farming:add_plant("plant_carrot", "mcl_farming:carrot", {"mcl_farming:carrot_1", "mcl_farming:carrot_2", "mcl_farming:carrot_3", "mcl_farming:carrot_4", "mcl_farming:carrot_5", "mcl_farming:carrot_6", "mcl_farming:carrot_7"}, 5.8013, 35)
|
||||||
|
|
||||||
if minetest.get_modpath("doc") then
|
if minetest.get_modpath("doc") then
|
||||||
for i=2,7 do
|
for i=2,7 do
|
||||||
|
|
|
@ -123,10 +123,10 @@ local stem_def = {
|
||||||
}
|
}
|
||||||
|
|
||||||
-- Register stem growth
|
-- Register stem growth
|
||||||
mcl_farming:add_plant("plant_melon_stem", "mcl_farming:melontige_unconnect", {"mcl_farming:melontige_1", "mcl_farming:melontige_2", "mcl_farming:melontige_3", "mcl_farming:melontige_4", "mcl_farming:melontige_5", "mcl_farming:melontige_6", "mcl_farming:melontige_7"}, 30, 5)
|
mcl_farming:add_plant("plant_melon_stem", "mcl_farming:melontige_unconnect", {"mcl_farming:melontige_1", "mcl_farming:melontige_2", "mcl_farming:melontige_3", "mcl_farming:melontige_4", "mcl_farming:melontige_5", "mcl_farming:melontige_6", "mcl_farming:melontige_7"}, 5.8014, 35)
|
||||||
|
|
||||||
-- Register actual melon, connected stems and stem-to-melon growth
|
-- Register actual melon, connected stems and stem-to-melon growth
|
||||||
mcl_farming:add_gourd("mcl_farming:melontige_unconnect", "mcl_farming:melontige_linked", "mcl_farming:melontige_unconnect", stem_def, stem_drop, "mcl_farming:melon", melon_base_def, 25, 15, "mcl_farming_melon_stem_connected.png^[colorize:#FFA800:127")
|
mcl_farming:add_gourd("mcl_farming:melontige_unconnect", "mcl_farming:melontige_linked", "mcl_farming:melontige_unconnect", stem_def, stem_drop, "mcl_farming:melon", melon_base_def, 5.8015, 35, "mcl_farming_melon_stem_connected.png^[colorize:#FFA800:127")
|
||||||
|
|
||||||
-- Items and crafting
|
-- Items and crafting
|
||||||
minetest.register_craftitem("mcl_farming:melon_item", {
|
minetest.register_craftitem("mcl_farming:melon_item", {
|
||||||
|
|
|
@ -135,7 +135,7 @@ minetest.register_craft({
|
||||||
cooktime = 10,
|
cooktime = 10,
|
||||||
})
|
})
|
||||||
|
|
||||||
mcl_farming:add_plant("plant_potato", "mcl_farming:potato", {"mcl_farming:potato_1", "mcl_farming:potato_2", "mcl_farming:potato_3", "mcl_farming:potato_4", "mcl_farming:potato_5", "mcl_farming:potato_6", "mcl_farming:potato_7"}, 19.75, 20)
|
mcl_farming:add_plant("plant_potato", "mcl_farming:potato", {"mcl_farming:potato_1", "mcl_farming:potato_2", "mcl_farming:potato_3", "mcl_farming:potato_4", "mcl_farming:potato_5", "mcl_farming:potato_6", "mcl_farming:potato_7"}, 5.8016, 35)
|
||||||
|
|
||||||
minetest.register_on_item_eat(function (hp_change, replace_with_item, itemstack, user, pointed_thing)
|
minetest.register_on_item_eat(function (hp_change, replace_with_item, itemstack, user, pointed_thing)
|
||||||
|
|
||||||
|
|
|
@ -180,10 +180,10 @@ if minetest.get_modpath("mcl_armor") then
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Register stem growth
|
-- Register stem growth
|
||||||
mcl_farming:add_plant("plant_pumpkin_stem", "mcl_farming:pumpkintige_unconnect", {"mcl_farming:pumpkin_1", "mcl_farming:pumpkin_2", "mcl_farming:pumpkin_3", "mcl_farming:pumpkin_4", "mcl_farming:pumpkin_5", "mcl_farming:pumpkin_6", "mcl_farming:pumpkin_7"}, 30, 5)
|
mcl_farming:add_plant("plant_pumpkin_stem", "mcl_farming:pumpkintige_unconnect", {"mcl_farming:pumpkin_1", "mcl_farming:pumpkin_2", "mcl_farming:pumpkin_3", "mcl_farming:pumpkin_4", "mcl_farming:pumpkin_5", "mcl_farming:pumpkin_6", "mcl_farming:pumpkin_7"}, 5.8017, 35)
|
||||||
|
|
||||||
-- Register actual pumpkin, connected stems and stem-to-pumpkin growth
|
-- Register actual pumpkin, connected stems and stem-to-pumpkin growth
|
||||||
mcl_farming:add_gourd("mcl_farming:pumpkintige_unconnect", "mcl_farming:pumpkintige_linked", "mcl_farming:pumpkintige_unconnect", stem_def, stem_drop, "mcl_farming:pumpkin", pumpkin_base_def, 30, 15, "mcl_farming_pumpkin_stem_connected.png^[colorize:#FFA800:127")
|
mcl_farming:add_gourd("mcl_farming:pumpkintige_unconnect", "mcl_farming:pumpkintige_linked", "mcl_farming:pumpkintige_unconnect", stem_def, stem_drop, "mcl_farming:pumpkin", pumpkin_base_def, 5.8018, 35, "mcl_farming_pumpkin_stem_connected.png^[colorize:#FFA800:127")
|
||||||
|
|
||||||
-- Steal function to properly disconnect a carved pumpkin
|
-- Steal function to properly disconnect a carved pumpkin
|
||||||
pumpkin_face_base_def.after_destruct = minetest.registered_nodes["mcl_farming:pumpkin"].after_destruct
|
pumpkin_face_base_def.after_destruct = minetest.registered_nodes["mcl_farming:pumpkin"].after_destruct
|
||||||
|
|
|
@ -5,73 +5,99 @@
|
||||||
--
|
--
|
||||||
local math = math
|
local math = math
|
||||||
local vector = vector
|
local vector = vector
|
||||||
|
local random = math.random
|
||||||
|
local floor = math.floor
|
||||||
|
|
||||||
local plant_lists = {}
|
local plant_lists = {}
|
||||||
mcl_farming.plant_lists = plant_lists -- export
|
mcl_farming.plant_lists = plant_lists -- export
|
||||||
local plant_nodename_to_id_list = {}
|
local plant_nodename_to_id = {} -- map nodes to plants
|
||||||
|
local plant_step_from_name = {} -- map nodes to growth steps
|
||||||
|
|
||||||
local time_speed = tonumber(minetest.settings:get("time_speed")) or 72
|
local growth_factor = tonumber(minetest.settings:get("vl_plant_growth")) or 1.0
|
||||||
local time_multiplier = time_speed > 0 and (86400 / time_speed) or 0
|
|
||||||
|
|
||||||
local function get_intervals_counter(pos, interval, chance)
|
-- wetness of the surroundings
|
||||||
if time_multiplier == 0 then return 0 end
|
-- dry farmland = 1 point
|
||||||
-- "wall clock time", so plants continue to grow while sleeping
|
-- wet farmland = 3 points
|
||||||
local current_game_time = (minetest.get_day_count() + minetest.get_timeofday()) * time_multiplier
|
-- center point gives + 1 point, so 2 resp. 4
|
||||||
local approx_interval = math.max(interval, 1) * math.max(chance, 1)
|
-- neighbors only 25%
|
||||||
|
local function get_moisture_level(pos)
|
||||||
local meta = minetest.get_meta(pos)
|
local n = vector.offset(pos, 0, -1, 0)
|
||||||
local last_game_time = meta:get_float("last_gametime")
|
local totalm = 1
|
||||||
if last_game_time < 1 then
|
for z = -1,1 do
|
||||||
last_game_time = current_game_time - approx_interval * 0.5
|
n.z = pos.z + z
|
||||||
elseif last_game_time == current_game_time then
|
for x = -1,1 do
|
||||||
current_game_time = current_game_time + approx_interval
|
n.x = pos.x + x
|
||||||
|
local ndef = minetest.registered_nodes[minetest.get_node(n).name]
|
||||||
|
local soil = ndef and ndef.groups.soil
|
||||||
|
if soil and soil >= 2 then
|
||||||
|
local m = soil > 2 and 3 or 1
|
||||||
|
-- corners have less weight
|
||||||
|
if x ~= 0 or z ~= 0 then m = m * 0.25 end
|
||||||
|
totalm = totalm + m
|
||||||
end
|
end
|
||||||
meta:set_float("last_gametime", current_game_time)
|
end
|
||||||
return (current_game_time - last_game_time) / approx_interval
|
end
|
||||||
|
return totalm
|
||||||
end
|
end
|
||||||
|
|
||||||
local function get_avg_light_level(pos)
|
-- moisture penalty function:
|
||||||
local meta = minetest.get_meta(pos)
|
-- 0.5 if both on the x axis and the z axis at least one of the same plants grows
|
||||||
-- EWMA would use a single variable:
|
-- 0.5 if at least one diagonal neighbor is the same
|
||||||
-- local avg = meta:get_float("avg_light")
|
-- 1.0 otherwise
|
||||||
-- avg = avg + (node_light - avg) * 0.985
|
-- we cannot use the names directly, because growth is encoded in the names
|
||||||
-- meta.set_float("avg_light", avg)
|
local function get_same_crop_penalty(pos)
|
||||||
local summary = meta:get_int("avg_light_summary")
|
local name = minetest.get_node(pos).name
|
||||||
local counter = meta:get_int("avg_light_count")
|
local plant = plant_nodename_to_id[name]
|
||||||
if counter > 99 then
|
if not plant then return 1 end
|
||||||
summary, counter = math.ceil(summary * 0.5), 50
|
local n = vector.copy(pos)
|
||||||
|
-- check adjacent positions, avoid vector allocations and reduce node accesses
|
||||||
|
n.x = pos.x - 1
|
||||||
|
local dx = plant_nodename_to_id[minetest.get_node(n).name] == plant
|
||||||
|
n.x = pos.x + 1
|
||||||
|
dx = dx or plant_nodename_to_id[minetest.get_node(n).name] == plant
|
||||||
|
if dx then -- no need to check z otherwise
|
||||||
|
n.x = pos.x
|
||||||
|
n.z = pos.z - 1
|
||||||
|
local dz = plant_nodename_to_id[minetest.get_node(n).name] == plant
|
||||||
|
n.z = pos.z + 1
|
||||||
|
dz = dz or plant_nodename_to_id[minetest.get_node(n).name] == plant
|
||||||
|
if dz then return 0.5 end
|
||||||
end
|
end
|
||||||
local node_light = minetest.get_node_light(pos)
|
-- check diagonals, clockwise
|
||||||
if node_light ~= nil then
|
n.x, n.z = pos.x - 1, pos.z - 1
|
||||||
summary, counter = summary + node_light, counter + 1
|
if plant_nodename_to_id[minetest.get_node(n).name] == plant then return 0.5 end
|
||||||
meta:set_int("avg_light_summary", summary)
|
n.x = pos.x + 1
|
||||||
meta:set_int("avg_light_count", counter)
|
if plant_nodename_to_id[minetest.get_node(n).name] == plant then return 0.5 end
|
||||||
end
|
n.z = pos.z + 1
|
||||||
return math.ceil(summary / counter)
|
if plant_nodename_to_id[minetest.get_node(n).name] == plant then return 0.5 end
|
||||||
|
n.x = pos.x - 1
|
||||||
|
if plant_nodename_to_id[minetest.get_node(n).name] == plant then return 0.5 end
|
||||||
|
return 1
|
||||||
end
|
end
|
||||||
|
|
||||||
function mcl_farming:add_plant(identifier, full_grown, names, interval, chance)
|
function mcl_farming:add_plant(identifier, full_grown, names, interval, chance)
|
||||||
|
interval = growth_factor > 0 and (interval / growth_factor) or 0
|
||||||
local plant_info = {}
|
local plant_info = {}
|
||||||
plant_info.full_grown = full_grown
|
plant_info.full_grown = full_grown
|
||||||
plant_info.names = names
|
plant_info.names = names
|
||||||
plant_info.interval = interval
|
plant_info.interval = interval
|
||||||
plant_info.chance = chance
|
plant_info.chance = chance
|
||||||
|
plant_nodename_to_id[full_grown] = identifier
|
||||||
for _, nodename in pairs(names) do
|
for _, nodename in pairs(names) do
|
||||||
plant_nodename_to_id_list[nodename] = identifier
|
plant_nodename_to_id[nodename] = identifier
|
||||||
end
|
end
|
||||||
plant_info.step_from_name = {}
|
|
||||||
for i, name in ipairs(names) do
|
for i, name in ipairs(names) do
|
||||||
plant_info.step_from_name[name] = i
|
plant_step_from_name[name] = i
|
||||||
end
|
end
|
||||||
plant_lists[identifier] = plant_info
|
plant_lists[identifier] = plant_info
|
||||||
|
if interval == 0 then return end -- growth disabled
|
||||||
minetest.register_abm({
|
minetest.register_abm({
|
||||||
label = string.format("Farming plant growth (%s)", identifier),
|
label = string.format("Farming plant growth (%s)", identifier),
|
||||||
nodenames = names,
|
nodenames = names,
|
||||||
interval = interval,
|
interval = interval,
|
||||||
chance = chance,
|
chance = chance,
|
||||||
action = function(pos, node)
|
action = function(pos, node)
|
||||||
local low_speed = minetest.get_node(vector.offset(pos, 0, -1, 0)).name ~= "mcl_farming:soil_wet"
|
mcl_farming:grow_plant(identifier, pos, node, 1, false)
|
||||||
mcl_farming:grow_plant(identifier, pos, node, 1, false, low_speed)
|
|
||||||
end,
|
end,
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
@ -81,40 +107,30 @@ end
|
||||||
-- pos: Position
|
-- pos: Position
|
||||||
-- node: Node table
|
-- node: Node table
|
||||||
-- stages: Number of stages to advance (optional, defaults to 1)
|
-- stages: Number of stages to advance (optional, defaults to 1)
|
||||||
-- ignore_light: if true, ignore light requirements for growing
|
-- ignore_light_water: if true, ignore light and water requirements for growing
|
||||||
-- low_speed: grow more slowly (not wet), default false
|
|
||||||
-- Returns true if plant has been grown by 1 or more stages.
|
-- Returns true if plant has been grown by 1 or more stages.
|
||||||
-- Returns false if nothing changed.
|
-- Returns false if nothing changed.
|
||||||
function mcl_farming:grow_plant(identifier, pos, node, stages, ignore_light, low_speed)
|
function mcl_farming:grow_plant(identifier, pos, node, stages, ignore_light_water)
|
||||||
stages = stages or 1
|
-- number of missed interval ticks, for catch-up in block loading
|
||||||
local plant_info = plant_lists[identifier]
|
local plant_info = plant_lists[identifier]
|
||||||
local intervals_counter = get_intervals_counter(pos, plant_info.interval, plant_info.chance)
|
if not plant_info then return end
|
||||||
if stages > 0 then intervals_counters = intervals_counter - 1 end
|
if not ignore_light_water then
|
||||||
if low_speed then -- 10% speed approximately
|
if (minetest.get_node_light(pos, 0.5) or 0) < 0 then return false end -- day light
|
||||||
if intervals_counter < 1.01 and math.random(0, 9) > 0 then return false end
|
local odds = floor(25 / (get_moisture_level(pos) * get_same_crop_penalty(pos))) + 1
|
||||||
intervals_counter = intervals_counter / 10
|
for i = 1,stages do
|
||||||
end
|
-- compared to info from the MC wiki, our ABM runs a third as often, hence we use triple the chance
|
||||||
if not ignore_light and intervals_counter < 1.5 then
|
if random() * odds >= 3 then stages = stages - 1 end
|
||||||
local light = minetest.get_node_light(pos)
|
|
||||||
if not light or light < 10 then return false end
|
|
||||||
end
|
|
||||||
|
|
||||||
if intervals_counter >= 1.5 then
|
|
||||||
local average_light_level = get_avg_light_level(pos)
|
|
||||||
if average_light_level < 0.1 then return false end
|
|
||||||
if average_light_level < 10 then
|
|
||||||
intervals_counter = intervals_counter * average_light_level / 10
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local step = plant_info.step_from_name[node.name]
|
|
||||||
if step == nil then return false end
|
|
||||||
stages = stages + math.floor(intervals_counter)
|
|
||||||
if stages == 0 then return false end
|
if stages == 0 then return false end
|
||||||
local new_node = { name = plant_info.names[step + stages] or plant_info.full_grown }
|
local step = plant_step_from_name[node.name]
|
||||||
new_node.param = node.param
|
if step == nil then return false end
|
||||||
new_node.param2 = node.param2
|
minetest.set_node(pos, {
|
||||||
minetest.set_node(pos, new_node)
|
name = plant_info.names[step + stages] or plant_info.full_grown,
|
||||||
|
param = node.param,
|
||||||
|
param2 = node.param2,
|
||||||
|
})
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -157,40 +173,13 @@ end
|
||||||
|
|
||||||
|
|
||||||
function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, stem_itemstring, stem_def, stem_drop, gourd_itemstring, gourd_def, grow_interval, grow_chance, connected_stem_texture)
|
function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, stem_itemstring, stem_def, stem_drop, gourd_itemstring, gourd_def, grow_interval, grow_chance, connected_stem_texture)
|
||||||
|
grow_interval = growth_factor > 0 and (grow_interval / growth_factor) or 0
|
||||||
local connected_stem_names = {
|
local connected_stem_names = {
|
||||||
connected_stem_basename .. "_r",
|
connected_stem_basename .. "_r",
|
||||||
connected_stem_basename .. "_l",
|
connected_stem_basename .. "_l",
|
||||||
connected_stem_basename .. "_t",
|
connected_stem_basename .. "_t",
|
||||||
connected_stem_basename .. "_b" }
|
connected_stem_basename .. "_b" }
|
||||||
|
|
||||||
-- Connect the stem at stempos to the first neighboring gourd block.
|
|
||||||
-- No-op if not a stem or no gourd block found
|
|
||||||
local function try_connect_stem(stempos)
|
|
||||||
local stem = minetest.get_node(stempos)
|
|
||||||
if stem.name ~= full_unconnected_stem then return false end
|
|
||||||
-- four directions, but avoid table allocation
|
|
||||||
local neighbor = vector.offset(stempos, 1, 0, 0)
|
|
||||||
if minetest.get_node(neighbor).name == gourd_itemstring then
|
|
||||||
minetest.swap_node(stempos, { name = connected_stem_names[1] })
|
|
||||||
return true
|
|
||||||
end
|
|
||||||
local neighbor = vector.offset(stempos, -1, 0, 0)
|
|
||||||
if minetest.get_node(neighbor).name == gourd_itemstring then
|
|
||||||
minetest.swap_node(stempos, { name = connected_stem_names[2] })
|
|
||||||
return true
|
|
||||||
end
|
|
||||||
local neighbor = vector.offset(stempos, 0, 0, 1)
|
|
||||||
if minetest.get_node(neighbor).name == gourd_itemstring then
|
|
||||||
minetest.swap_node(stempos, { name = connected_stem_names[3] })
|
|
||||||
return true
|
|
||||||
end
|
|
||||||
local neighbor = vector.offset(stempos, 0, 0, -1)
|
|
||||||
if minetest.get_node(neighbor).name == gourd_itemstring then
|
|
||||||
minetest.swap_node(stempos, { name = connected_stem_names[4] })
|
|
||||||
return true
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Register gourd
|
-- Register gourd
|
||||||
if not gourd_def.after_destruct then
|
if not gourd_def.after_destruct then
|
||||||
gourd_def.after_destruct = function(blockpos, oldnode)
|
gourd_def.after_destruct = function(blockpos, oldnode)
|
||||||
|
@ -200,34 +189,21 @@ function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, s
|
||||||
local stempos = vector.offset(blockpos, -1, 0, 0)
|
local stempos = vector.offset(blockpos, -1, 0, 0)
|
||||||
if minetest.get_node(stempos).name == connected_stem_names[1] then
|
if minetest.get_node(stempos).name == connected_stem_names[1] then
|
||||||
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
||||||
try_connect_stem(stempos)
|
|
||||||
end
|
end
|
||||||
local stempos = vector.offset(blockpos, 1, 0, 0)
|
local stempos = vector.offset(blockpos, 1, 0, 0)
|
||||||
if minetest.get_node(stempos).name == connected_stem_names[2] then
|
if minetest.get_node(stempos).name == connected_stem_names[2] then
|
||||||
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
||||||
try_connect_stem(stempos)
|
|
||||||
end
|
end
|
||||||
local stempos = vector.offset(blockpos, 0, 0, -1)
|
local stempos = vector.offset(blockpos, 0, 0, -1)
|
||||||
if minetest.get_node(stempos).name == connected_stem_names[3] then
|
if minetest.get_node(stempos).name == connected_stem_names[3] then
|
||||||
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
||||||
try_connect_stem(stempos)
|
|
||||||
end
|
end
|
||||||
local stempos = vector.offset(blockpos, 0, 0, 1)
|
local stempos = vector.offset(blockpos, 0, 0, 1)
|
||||||
if minetest.get_node(stempos).name == connected_stem_names[4] then
|
if minetest.get_node(stempos).name == connected_stem_names[4] then
|
||||||
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
minetest.swap_node(stempos, { name = full_unconnected_stem })
|
||||||
try_connect_stem(stempos)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if not gourd_def.on_construct then
|
|
||||||
function gourd_def.on_construct(blockpos)
|
|
||||||
-- Connect all unconnected stems at full size
|
|
||||||
try_connect_stem(vector.offset(blockpos, 1, 0, 0))
|
|
||||||
try_connect_stem(vector.offset(blockpos, -1, 0, 0))
|
|
||||||
try_connect_stem(vector.offset(blockpos, 0, 0, 1))
|
|
||||||
try_connect_stem(vector.offset(blockpos, 0, 0, -1))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
minetest.register_node(gourd_itemstring, gourd_def)
|
minetest.register_node(gourd_itemstring, gourd_def)
|
||||||
|
|
||||||
-- Register unconnected stem
|
-- Register unconnected stem
|
||||||
|
@ -243,8 +219,8 @@ function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, s
|
||||||
stem_def.drop = stem_def.drop or stem_drop
|
stem_def.drop = stem_def.drop or stem_drop
|
||||||
stem_def.groups = stem_def.groups or { dig_immediate = 3, not_in_creative_inventory = 1, plant = 1, attached_node = 1, dig_by_water = 1, destroy_by_lava_flow = 1 }
|
stem_def.groups = stem_def.groups or { dig_immediate = 3, not_in_creative_inventory = 1, plant = 1, attached_node = 1, dig_by_water = 1, destroy_by_lava_flow = 1 }
|
||||||
stem_def.sounds = stem_def.sounds or mcl_sounds.node_sound_leaves_defaults()
|
stem_def.sounds = stem_def.sounds or mcl_sounds.node_sound_leaves_defaults()
|
||||||
stem_def.on_construct = stem_def.on_construct or try_connect_stem
|
|
||||||
minetest.register_node(stem_itemstring, stem_def)
|
minetest.register_node(stem_itemstring, stem_def)
|
||||||
|
plant_nodename_to_id[stem_itemstring] = stem_itemstring
|
||||||
|
|
||||||
-- Register connected stems
|
-- Register connected stems
|
||||||
|
|
||||||
|
@ -307,22 +283,14 @@ function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, s
|
||||||
sounds = mcl_sounds.node_sound_leaves_defaults(),
|
sounds = mcl_sounds.node_sound_leaves_defaults(),
|
||||||
_mcl_blast_resistance = 0,
|
_mcl_blast_resistance = 0,
|
||||||
})
|
})
|
||||||
|
plant_nodename_to_id[connected_stem_names[i]] = stem_itemstring
|
||||||
|
|
||||||
if minetest.get_modpath("doc") then
|
if minetest.get_modpath("doc") then
|
||||||
doc.add_entry_alias("nodes", full_unconnected_stem, "nodes", connected_stem_names[i])
|
doc.add_entry_alias("nodes", full_unconnected_stem, "nodes", connected_stem_names[i])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Check for a suitable spot to grow
|
if grow_interval == 0 then return end
|
||||||
local function check_neighbor_soil(blockpos)
|
|
||||||
if minetest.get_node(blockpos).name ~= "air" then return false end
|
|
||||||
local floorpos = vector.offset(blockpos, 0, -1, 0)
|
|
||||||
local floorname = minetest.get_node(floorpos).name
|
|
||||||
if floorname == "mcl_core:dirt" then return true end
|
|
||||||
local floordef = minetest.registered_nodes[floorname]
|
|
||||||
return floordef.groups.grass_block or floordef.groups.dirt or (floordef.groups.soil or 0) >= 2
|
|
||||||
end
|
|
||||||
|
|
||||||
minetest.register_abm({
|
minetest.register_abm({
|
||||||
label = "Grow gourd stem to gourd (" .. full_unconnected_stem .. " → " .. gourd_itemstring .. ")",
|
label = "Grow gourd stem to gourd (" .. full_unconnected_stem .. " → " .. gourd_itemstring .. ")",
|
||||||
nodenames = { full_unconnected_stem },
|
nodenames = { full_unconnected_stem },
|
||||||
|
@ -330,39 +298,27 @@ function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, s
|
||||||
interval = grow_interval,
|
interval = grow_interval,
|
||||||
chance = grow_chance,
|
chance = grow_chance,
|
||||||
action = function(stempos)
|
action = function(stempos)
|
||||||
local light = minetest.get_node_light(stempos)
|
local light = minetest.get_node_light(stempos, 0.5)
|
||||||
if not light or light <= 10 then return end
|
if not light or light < 9 then return end
|
||||||
-- Check the four neighbors and filter out neighbors where gourds can't grow
|
-- Pick one neighbor and check if it can be used to grow
|
||||||
local neighbor, dir, nchance = nil, -1, 1 -- reservoir sampling
|
local dir = random(1, 4) -- pick direction at random
|
||||||
if nchance == 1 or math.random(1, nchance) == 1 then
|
local neighbor = (dir == 1 and vector.offset(stempos, 1, 0, 0))
|
||||||
local blockpos = vector.offset(stempos, 1, 0, 0)
|
or (dir == 2 and vector.offset(stempos, -1, 0, 0))
|
||||||
if check_neighbor_soil(blockpos) then
|
or (dir == 3 and vector.offset(stempos, 0, 0, 1))
|
||||||
neighbor, dir, nchance = blockpos, 1, nchance + 1
|
or vector.offset(stempos, 0, 0, -1)
|
||||||
end
|
if minetest.get_node(neighbor).name ~= "air" then return end -- occupied
|
||||||
end
|
-- check for suitable floor -- in contrast to MC, we think everything solid is fine
|
||||||
if nchance == 1 or math.random(1, nchance) == 1 then
|
local floorpos = vector.offset(neighbor, 0, -1, 0)
|
||||||
local blockpos = vector.offset(stempos, -1, 0, 0)
|
local floorname = minetest.get_node(floorpos).name
|
||||||
if check_neighbor_soil(blockpos) then
|
local floordef = minetest.registered_nodes[floorname]
|
||||||
neighbor, dir, nchance = blockpos, 2, nchance + 1
|
if not floordef or not floordef.walkable then return end
|
||||||
end
|
|
||||||
end
|
-- check moisture level
|
||||||
if nchance == 1 or math.random(1, nchance) == 1 then
|
local odds = floor(25 / (get_moisture_level(stempos) * get_same_crop_penalty(stempos))) + 1
|
||||||
local blockpos = vector.offset(stempos, 0, 0, 1)
|
-- we triple the odds, and rather call the ABM less often
|
||||||
if check_neighbor_soil(blockpos) then
|
if random() * odds >= 3 then return end
|
||||||
neighbor, dir, nchance = blockpos, 3, nchance + 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
if nchance == 1 or math.random(1, nchance) == 1 then
|
|
||||||
local blockpos = vector.offset(stempos, 0, 0, -1)
|
|
||||||
if check_neighbor_soil(blockpos) then
|
|
||||||
neighbor, dir, nchance = blockpos, 4, nchance + 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Gourd needs at least 1 free neighbor to grow
|
|
||||||
if not neighbor then return end
|
|
||||||
minetest.swap_node(stempos, { name = connected_stem_names[dir] })
|
minetest.swap_node(stempos, { name = connected_stem_names[dir] })
|
||||||
-- Place the gourd
|
|
||||||
if gourd_def.paramtype2 == "facedir" then
|
if gourd_def.paramtype2 == "facedir" then
|
||||||
local p2 = (dir == 1 and 3) or (dir == 2 and 1) or (dir == 3 and 2) or 0
|
local p2 = (dir == 1 and 3) or (dir == 2 and 1) or (dir == 3 and 2) or 0
|
||||||
minetest.add_node(neighbor, { name = gourd_itemstring, param2 = p2 })
|
minetest.add_node(neighbor, { name = gourd_itemstring, param2 = p2 })
|
||||||
|
@ -371,8 +327,7 @@ function mcl_farming:add_gourd(full_unconnected_stem, connected_stem_basename, s
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Reset farmland, etc. to dirt when the gourd grows on top
|
-- Reset farmland, etc. to dirt when the gourd grows on top
|
||||||
local floorpos = vector.offset(neighbor, 0, -1, 0)
|
if (floordef.groups.dirtifies_below_solid or 0) > 0 then
|
||||||
if minetest.get_item_group(minetest.get_node(floorpos).name, "dirtifies_below_solid") == 1 then
|
|
||||||
minetest.set_node(floorpos, { name = "mcl_core:dirt" })
|
minetest.set_node(floorpos, { name = "mcl_core:dirt" })
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
|
@ -409,10 +364,21 @@ minetest.register_lbm({
|
||||||
nodenames = { "group:plant" },
|
nodenames = { "group:plant" },
|
||||||
run_at_every_load = true,
|
run_at_every_load = true,
|
||||||
action = function(pos, node, dtime_s)
|
action = function(pos, node, dtime_s)
|
||||||
local identifier = plant_nodename_to_id_list[node.name]
|
local identifier = plant_nodename_to_id[node.name]
|
||||||
if not identifier then return end
|
if not identifier then return end
|
||||||
local low_speed = minetest.get_node(vector.offset(pos, 0, -1, 0)).name ~= "mcl_farming:soil_wet"
|
|
||||||
mcl_farming:grow_plant(identifier, pos, node, 0, false, low_speed)
|
local plant_info = plant_lists[identifier]
|
||||||
|
if not plant_info then return end
|
||||||
|
local rolls = floor(dtime_s / plant_info.interval)
|
||||||
|
if rolls <= 0 then return end
|
||||||
|
-- simulate how often the block will be ticked
|
||||||
|
local stages = 0
|
||||||
|
for i = 1,rolls do
|
||||||
|
if random(1, plant_info.chance) == 1 then stages = stages + 1 end
|
||||||
|
end
|
||||||
|
if stages > 0 then
|
||||||
|
mcl_farming:grow_plant(identifier, pos, node, stages, false)
|
||||||
|
end
|
||||||
end,
|
end,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -15,10 +15,6 @@ minetest.register_node("mcl_farming:soil", {
|
||||||
{-0.5, -0.5, -0.5, 0.5, 0.4375, 0.5},
|
{-0.5, -0.5, -0.5, 0.5, 0.4375, 0.5},
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
on_construct = function(pos)
|
|
||||||
local meta = minetest.get_meta(pos)
|
|
||||||
meta:set_int("wet", 0)
|
|
||||||
end,
|
|
||||||
groups = {handy=1,shovely=1, dirtifies_below_solid=1, dirtifier=1, soil=2, soil_sapling=1, deco_block=1 },
|
groups = {handy=1,shovely=1, dirtifies_below_solid=1, dirtifier=1, soil=2, soil_sapling=1, deco_block=1 },
|
||||||
sounds = mcl_sounds.node_sound_dirt_defaults(),
|
sounds = mcl_sounds.node_sound_dirt_defaults(),
|
||||||
_mcl_blast_resistance = 0.6,
|
_mcl_blast_resistance = 0.6,
|
||||||
|
@ -38,10 +34,6 @@ minetest.register_node("mcl_farming:soil_wet", {
|
||||||
{-0.5, -0.5, -0.5, 0.5, 0.4375, 0.5},
|
{-0.5, -0.5, -0.5, 0.5, 0.4375, 0.5},
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
on_construct = function(pos)
|
|
||||||
local meta = minetest.get_meta(pos)
|
|
||||||
meta:set_int("wet", 7)
|
|
||||||
end,
|
|
||||||
groups = {handy=1,shovely=1, not_in_creative_inventory=1, dirtifies_below_solid=1, dirtifier=1, soil=3, soil_sapling=1 },
|
groups = {handy=1,shovely=1, not_in_creative_inventory=1, dirtifies_below_solid=1, dirtifier=1, soil=3, soil_sapling=1 },
|
||||||
sounds = mcl_sounds.node_sound_dirt_defaults(),
|
sounds = mcl_sounds.node_sound_dirt_defaults(),
|
||||||
_mcl_blast_resistance = 0.6,
|
_mcl_blast_resistance = 0.6,
|
||||||
|
@ -51,75 +43,54 @@ minetest.register_node("mcl_farming:soil_wet", {
|
||||||
minetest.register_abm({
|
minetest.register_abm({
|
||||||
label = "Farmland hydration",
|
label = "Farmland hydration",
|
||||||
nodenames = {"mcl_farming:soil", "mcl_farming:soil_wet"},
|
nodenames = {"mcl_farming:soil", "mcl_farming:soil_wet"},
|
||||||
interval = 15,
|
interval = 2.73,
|
||||||
chance = 4,
|
chance = 25,
|
||||||
action = function(pos, node)
|
action = function(pos, node)
|
||||||
-- Get wetness value
|
|
||||||
local meta = minetest.get_meta(pos)
|
|
||||||
local wet = meta:get_int("wet")
|
|
||||||
if not wet then
|
|
||||||
if node.name == "mcl_farming:soil" then
|
|
||||||
wet = 0
|
|
||||||
else
|
|
||||||
wet = 7
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Turn back into dirt when covered by solid node
|
-- Turn back into dirt when covered by solid node
|
||||||
local above_node = minetest.get_node_or_nil({x=pos.x,y=pos.y+1,z=pos.z})
|
local above_node = minetest.get_node_or_nil(vector.offset(pos, 0, 1, 0))
|
||||||
if above_node then
|
if above_node and minetest.get_item_group(above_node.name, "solid") ~= 0 then
|
||||||
if minetest.get_item_group(above_node.name, "solid") ~= 0 then
|
|
||||||
node.name = "mcl_core:dirt"
|
node.name = "mcl_core:dirt"
|
||||||
minetest.set_node(pos, node)
|
minetest.set_node(pos, node)
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
-- Check an area of 9×2×9 around the node for nodename (9×9 on same level and 9×9 below)
|
-- in rain, become wet, do not decay
|
||||||
local function check_surroundings(pos, nodename)
|
if mcl_weather and mcl_weather.rain.raining and mcl_weather.is_outdoor(pos) then
|
||||||
local nodes = minetest.find_nodes_in_area({x=pos.x-4,y=pos.y,z=pos.z-4}, {x=pos.x+4,y=pos.y+1,z=pos.z+4}, {nodename})
|
if node.name == "mcl_farming:soil" then
|
||||||
return #nodes > 0
|
|
||||||
end
|
|
||||||
|
|
||||||
if check_surroundings(pos, "group:water") then
|
|
||||||
if node.name ~= "mcl_farming:soil_wet" then
|
|
||||||
-- Make it wet
|
|
||||||
node.name = "mcl_farming:soil_wet"
|
node.name = "mcl_farming:soil_wet"
|
||||||
minetest.set_node(pos, node)
|
minetest.set_node(pos, node)
|
||||||
end
|
end
|
||||||
else -- No water nearby.
|
|
||||||
-- The decay branch (make farmland dry or turn back to dirt)
|
|
||||||
|
|
||||||
-- Don't decay while it's raining
|
|
||||||
if mcl_weather.rain.raining then
|
|
||||||
if mcl_weather.is_outdoor(pos) then
|
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
|
-- Check an area of 9x2x9 around the node for nodename (9x9 on same level and 9x9 above)
|
||||||
|
-- include "ignore" to detect unloaded blocks
|
||||||
|
local nodes, counts = minetest.find_nodes_in_area(vector.offset(pos, -4, 0, -4), vector.offset(pos, 4, 1, 4), {"group:water", "ignore"})
|
||||||
|
local ignore = counts.ignore or 0
|
||||||
|
local has_water, fully_loaded = #nodes > ignore, ignore == 0
|
||||||
|
|
||||||
|
-- Hydrate by rain or water, do not decay
|
||||||
|
if has_water then
|
||||||
|
if node.name == "mcl_farming:soil" then
|
||||||
|
node.name = "mcl_farming:soil_wet"
|
||||||
|
minetest.set_node(pos, node)
|
||||||
end
|
end
|
||||||
-- No decay near unloaded areas since these might include water.
|
|
||||||
if not check_surroundings(pos, "ignore") then
|
|
||||||
if wet <= 0 then
|
|
||||||
--local n_def = minetest.registered_nodes[node.name] or nil
|
|
||||||
local nn = minetest.get_node_or_nil({x=pos.x,y=pos.y+1,z=pos.z})
|
|
||||||
if not nn or not nn.name then
|
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
local nn_def = minetest.registered_nodes[nn.name] or nil
|
-- No decay near unloaded areas (ignore) since these might include water.
|
||||||
|
if not fully_loaded then return end
|
||||||
|
|
||||||
if nn_def and minetest.get_item_group(nn.name, "plant") == 0 then
|
-- Decay: make wet farmland dry up
|
||||||
node.name = "mcl_core:dirt"
|
if node.name == "mcl_farming:soil_wet" then
|
||||||
|
node.name = "mcl_farming:soil"
|
||||||
minetest.set_node(pos, node)
|
minetest.set_node(pos, node)
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
else
|
-- Revert to dirt if wetness is 0, and no plant above
|
||||||
if wet == 7 then
|
local above = minetest.get_node_or_nil(vector.offset(pos, 0, 1, 0))
|
||||||
node.name = "mcl_farming:soil"
|
if minetest.get_item_group(above.name, "plant") == 0 then
|
||||||
minetest.swap_node(pos, node)
|
node.name = "mcl_core:dirt"
|
||||||
end
|
minetest.set_node(pos, node)
|
||||||
-- Slowly count down wetness
|
|
||||||
meta:set_int("wet", wet-1)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
})
|
})
|
||||||
|
|
|
@ -99,8 +99,8 @@ minetest.register_craftitem("mcl_farming:sweet_berry", {
|
||||||
})
|
})
|
||||||
minetest.register_alias("mcl_sweet_berry:sweet_berry", "mcl_farming:sweet_berry")
|
minetest.register_alias("mcl_sweet_berry:sweet_berry", "mcl_farming:sweet_berry")
|
||||||
|
|
||||||
-- TODO: Find proper interval and chance values for sweet berry bushes. Current interval and chance values are copied from mcl_farming:beetroot which has similar growth stages.
|
-- TODO: Find proper interval and chance values for sweet berry bushes. Current interval and chance values are copied from mcl_farming:beetroot which has similar growth stages, 2/3rd of the default.
|
||||||
mcl_farming:add_plant("plant_sweet_berry_bush", "mcl_farming:sweet_berry_bush_3", {"mcl_farming:sweet_berry_bush_0", "mcl_farming:sweet_berry_bush_1", "mcl_farming:sweet_berry_bush_2"}, 68, 3)
|
mcl_farming:add_plant("plant_sweet_berry_bush", "mcl_farming:sweet_berry_bush_3", {"mcl_farming:sweet_berry_bush_0", "mcl_farming:sweet_berry_bush_1", "mcl_farming:sweet_berry_bush_2"}, 8.7019, 35)
|
||||||
|
|
||||||
local function berry_damage_check(obj)
|
local function berry_damage_check(obj)
|
||||||
local p = obj:get_pos()
|
local p = obj:get_pos()
|
||||||
|
|
|
@ -99,7 +99,7 @@ minetest.register_node("mcl_farming:wheat", {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
mcl_farming:add_plant("plant_wheat", "mcl_farming:wheat", {"mcl_farming:wheat_1", "mcl_farming:wheat_2", "mcl_farming:wheat_3", "mcl_farming:wheat_4", "mcl_farming:wheat_5", "mcl_farming:wheat_6", "mcl_farming:wheat_7"}, 25, 20)
|
mcl_farming:add_plant("plant_wheat", "mcl_farming:wheat", {"mcl_farming:wheat_1", "mcl_farming:wheat_2", "mcl_farming:wheat_3", "mcl_farming:wheat_4", "mcl_farming:wheat_5", "mcl_farming:wheat_6", "mcl_farming:wheat_7"}, 5.8020, 35)
|
||||||
|
|
||||||
minetest.register_craftitem("mcl_farming:wheat_item", {
|
minetest.register_craftitem("mcl_farming:wheat_item", {
|
||||||
description = S("Wheat"),
|
description = S("Wheat"),
|
||||||
|
|
|
@ -45,6 +45,9 @@ mcl_disabled_structures (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
|
||||||
|
|
||||||
|
# Control the relative plant growth speed (default: 1)
|
||||||
|
vl_plant_growth (Plant growth factor) float 1.0 0 100
|
||||||
|
|
||||||
[Players]
|
[Players]
|
||||||
# If enabled, players respawn at the bed they last lay on instead of normal
|
# If enabled, players respawn at the bed they last lay on instead of normal
|
||||||
# spawn.
|
# spawn.
|
||||||
|
|
Loading…
Reference in New Issue