2024-09-20 15:00:49 +03:00
-- Possible future improvements:
-- * rewrite to use node timers instead of ABMs, but needs benchmarking
-- * redesign the catch-up logic
-- * switch to exponentially-weighted moving average for light instead using a single variable to conserve IO
--
2021-05-29 17:12:33 +03:00
local math = math
2024-09-20 15:00:49 +03:00
local vector = vector
2024-10-10 18:32:36 +03:00
local random = math.random
local floor = math.floor
2021-05-29 17:12:33 +03:00
2021-03-12 02:10:50 +03:00
local plant_lists = { }
2024-09-20 15:00:49 +03:00
mcl_farming.plant_lists = plant_lists -- export
2024-10-13 22:42:42 +03:00
local plant_nodename_to_id = { } -- map nodes to plants
2024-10-10 17:23:54 +03:00
local plant_step_from_name = { } -- map nodes to growth steps
local growth_factor = tonumber ( minetest.settings : get ( " vl_plant_growth " ) ) or 1.0
2020-09-05 20:49:12 +03:00
2024-10-13 22:42:42 +03:00
-- wetness of the surroundings
-- dry farmland = 1 point
-- wet farmland = 3 points
2024-10-16 01:24:07 +03:00
-- center point gives + 1 point, so 2 resp. 4
-- neighbors only 25%
2024-10-09 23:15:30 +03:00
local function get_moisture_level ( pos )
local n = vector.offset ( pos , 0 , - 1 , 0 )
local totalm = 1
for z = - 1 , 1 do
n.z = pos.z + z
for x = - 1 , 1 do
n.x = pos.x + x
local ndef = minetest.registered_nodes [ minetest.get_node ( n ) . name ]
2024-10-13 22:42:42 +03:00
local soil = ndef and ndef.groups . soil
if soil and soil >= 2 then
2024-11-03 01:41:38 +03:00
local m = soil > 2 and 3 or 1
2024-10-13 22:42:42 +03:00
-- corners have less weight
2024-10-16 01:24:07 +03:00
if x ~= 0 or z ~= 0 then m = m * 0.25 end
2024-10-13 22:42:42 +03:00
totalm = totalm + m
end
2024-10-09 23:15:30 +03:00
end
end
return totalm
end
-- moisture penalty function:
2024-10-13 22:42:42 +03:00
-- 0.5 if both on the x axis and the z axis at least one of the same plants grows
-- 0.5 if at least one diagonal neighbor is the same
2024-10-09 23:15:30 +03:00
-- 1.0 otherwise
2024-10-13 22:42:42 +03:00
-- we cannot use the names directly, because growth is encoded in the names
local function get_same_crop_penalty ( pos )
2024-10-09 23:15:30 +03:00
local name = minetest.get_node ( pos ) . name
2024-10-13 22:42:42 +03:00
local plant = plant_nodename_to_id [ name ]
2024-10-16 01:24:07 +03:00
if not plant then return 1 end
2024-10-13 22:42:42 +03:00
local n = vector.copy ( pos )
-- check adjacent positions, avoid vector allocations and reduce node accesses
2024-10-09 23:15:30 +03:00
n.x = pos.x - 1
2024-10-13 22:42:42 +03:00
local dx = plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant
2024-10-09 23:15:30 +03:00
n.x = pos.x + 1
2024-10-13 22:42:42 +03:00
dx = dx or plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant
if dx then -- no need to check z otherwise
2024-10-09 23:15:30 +03:00
n.x = pos.x
n.z = pos.z - 1
2024-10-13 22:42:42 +03:00
local dz = plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant
2024-10-09 23:15:30 +03:00
n.z = pos.z + 1
2024-10-13 22:42:42 +03:00
dz = dz or plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant
2024-10-09 23:15:30 +03:00
if dz then return 0.5 end
end
-- check diagonals, clockwise
n.x , n.z = pos.x - 1 , pos.z - 1
2024-10-13 22:42:42 +03:00
if plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant then return 0.5 end
2024-10-09 23:15:30 +03:00
n.x = pos.x + 1
2024-10-13 22:42:42 +03:00
if plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant then return 0.5 end
2024-10-09 23:15:30 +03:00
n.z = pos.z + 1
2024-10-13 22:42:42 +03:00
if plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant then return 0.5 end
2024-10-09 23:15:30 +03:00
n.x = pos.x - 1
2024-10-13 22:42:42 +03:00
if plant_nodename_to_id [ minetest.get_node ( n ) . name ] == plant then return 0.5 end
2024-10-09 23:15:30 +03:00
return 1
end
2017-04-01 04:54:58 +03:00
function mcl_farming : add_plant ( identifier , full_grown , names , interval , chance )
2024-10-10 17:23:54 +03:00
interval = growth_factor > 0 and ( interval / growth_factor ) or 0
2024-09-20 15:00:49 +03:00
local plant_info = { }
plant_info.full_grown = full_grown
plant_info.names = names
plant_info.interval = interval
plant_info.chance = chance
2024-10-16 01:24:07 +03:00
plant_nodename_to_id [ full_grown ] = identifier
2024-09-20 15:00:49 +03:00
for _ , nodename in pairs ( names ) do
2024-10-13 22:42:42 +03:00
plant_nodename_to_id [ nodename ] = identifier
2024-09-20 15:00:49 +03:00
end
for i , name in ipairs ( names ) do
2024-10-10 17:23:54 +03:00
plant_step_from_name [ name ] = i
2024-09-20 15:00:49 +03:00
end
plant_lists [ identifier ] = plant_info
2024-10-10 17:23:54 +03:00
if interval == 0 then return end -- growth disabled
2017-03-14 01:09:27 +03:00
minetest.register_abm ( {
2017-05-15 01:45:54 +03:00
label = string.format ( " Farming plant growth (%s) " , identifier ) ,
2017-03-14 01:09:27 +03:00
nodenames = names ,
interval = interval ,
chance = chance ,
action = function ( pos , node )
2024-10-09 23:15:30 +03:00
mcl_farming : grow_plant ( identifier , pos , node , 1 , false )
2017-04-01 04:54:58 +03:00
end ,
2017-03-14 01:09:27 +03:00
} )
end
2017-04-01 04:54:58 +03:00
-- Attempts to advance a plant at pos by one or more growth stages (if possible)
-- identifier: Identifier of plant as defined by mcl_farming:add_plant
-- pos: Position
-- node: Node table
-- stages: Number of stages to advance (optional, defaults to 1)
2024-10-10 17:23:54 +03:00
-- ignore_light_water: if true, ignore light and water requirements for growing
2017-05-25 04:24:11 +03:00
-- Returns true if plant has been grown by 1 or more stages.
-- Returns false if nothing changed.
2024-10-10 17:23:54 +03:00
function mcl_farming : grow_plant ( identifier , pos , node , stages , ignore_light_water )
-- number of missed interval ticks, for catch-up in block loading
2020-09-05 20:49:12 +03:00
local plant_info = plant_lists [ identifier ]
2024-10-13 22:42:42 +03:00
if not plant_info then return end
2024-10-10 17:23:54 +03:00
if not ignore_light_water then
2024-10-16 22:19:38 +03:00
if ( minetest.get_node_light ( pos , 0.5 ) or 0 ) < 0 then return false end -- day light
2024-10-13 22:42:42 +03:00
local odds = floor ( 25 / ( get_moisture_level ( pos ) * get_same_crop_penalty ( pos ) ) ) + 1
2024-10-10 17:23:54 +03:00
for i = 1 , stages do
2024-10-16 01:24:07 +03:00
-- compared to info from the MC wiki, our ABM runs a third as often, hence we use triple the chance
if random ( ) * odds >= 3 then stages = stages - 1 end
2020-09-05 20:49:12 +03:00
end
end
2024-09-20 15:00:49 +03:00
if stages == 0 then return false end
2024-10-10 17:23:54 +03:00
local step = plant_step_from_name [ node.name ]
if step == nil then return false end
minetest.set_node ( pos , {
name = plant_info.names [ step + stages ] or plant_info.full_grown ,
2024-10-15 11:18:22 +03:00
param = node.param ,
param2 = node.param2 ,
2024-10-10 17:23:54 +03:00
} )
2017-05-25 04:24:11 +03:00
return true
2017-04-01 04:54:58 +03:00
end
2017-03-14 01:09:27 +03:00
function mcl_farming : place_seed ( itemstack , placer , pointed_thing , plantname )
local pt = pointed_thing
2024-09-20 15:00:49 +03:00
if not pt or pt.type ~= " node " then return end
2017-03-14 01:09:27 +03:00
-- Use pointed node's on_rightclick function first, if present
local node = minetest.get_node ( pt.under )
if placer and not placer : get_player_control ( ) . sneak then
if minetest.registered_nodes [ node.name ] and minetest.registered_nodes [ node.name ] . on_rightclick then
return minetest.registered_nodes [ node.name ] . on_rightclick ( pt.under , node , placer , itemstack ) or itemstack
end
end
2017-03-14 05:56:33 +03:00
2024-09-20 15:00:49 +03:00
if minetest.get_node ( pt.above ) . name ~= " air " then return end
local farmland = minetest.registered_nodes [ minetest.get_node ( vector.offset ( pt.above , 0 , - 1 , 0 ) ) . name ]
if not farmland or ( farmland.groups . soil or 0 ) < 2 then return end
minetest.sound_play ( minetest.registered_nodes [ plantname ] . sounds.place , { pos = pt.above } , true )
minetest.add_node ( pt.above , { name = plantname , param2 = minetest.registered_nodes [ plantname ] . place_param2 } )
2017-03-14 01:09:27 +03:00
2024-09-20 15:00:49 +03:00
if not minetest.is_creative_enabled ( placer : get_player_name ( ) ) then itemstack : take_item ( ) end
2017-03-14 01:09:27 +03:00
return itemstack
end
2017-03-14 00:30:37 +03:00
--[[ Helper function to create a gourd (e.g. melon, pumpkin), the connected stem nodes as
2022-11-16 15:17:51 +03:00
- full_unconnected_stem : itemstring of the full - grown but unconnected stem node . This node must already be done
2017-03-14 00:30:37 +03:00
- connected_stem_basename : prefix of the itemstrings used for the 4 connected stem nodes to create
2017-03-14 06:17:35 +03:00
- stem_itemstring : Desired itemstring of the fully - grown unconnected stem node
- stem_def : Partial node definition of the fully - grown unconnected stem node . Many fields are already defined . You need to add ` tiles ` and ` description ` at minimum . Don ' t define on_construct without good reason
- stem_drop : Drop probability table for all stem
2017-03-14 00:30:37 +03:00
- gourd_itemstring : Desired itemstring of the full gourd node
2022-02-23 20:14:44 +03:00
- gourd_def : ( almost ) full definition of the gourd node . This function will add on_construct and after_destruct to the definition for unconnecting any connected stems
2017-03-14 00:43:47 +03:00
- grow_interval : Will attempt to grow a gourd periodically at this interval in seconds
- grow_chance : Chance of 1 / grow_chance to grow a gourd next to the full unconnected stem after grow_interval has passed . Must be a natural number
2017-07-17 15:26:25 +03:00
- connected_stem_texture : Texture of the connected stem
2017-03-14 00:30:37 +03:00
] ]
2024-09-20 15:00:49 +03:00
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 )
2024-10-10 17:23:54 +03:00
grow_interval = growth_factor > 0 and ( grow_interval / growth_factor ) or 0
2021-05-23 00:50:28 +03:00
local connected_stem_names = {
2017-03-14 00:30:37 +03:00
connected_stem_basename .. " _r " ,
connected_stem_basename .. " _l " ,
connected_stem_basename .. " _t " ,
2024-09-20 15:00:49 +03:00
connected_stem_basename .. " _b " }
2017-03-14 05:56:33 +03:00
2017-03-14 00:30:37 +03:00
-- Register gourd
2022-02-23 20:14:44 +03:00
if not gourd_def.after_destruct then
gourd_def.after_destruct = function ( blockpos , oldnode )
2017-03-14 00:30:37 +03:00
-- Disconnect any connected stems, turning them back to normal stems
2024-09-20 15:00:49 +03:00
-- four directions, but avoid using a table
-- opposite directions to above, as we go from groud to stem now!
local stempos = vector.offset ( blockpos , - 1 , 0 , 0 )
if minetest.get_node ( stempos ) . name == connected_stem_names [ 1 ] then
minetest.swap_node ( stempos , { name = full_unconnected_stem } )
end
local stempos = vector.offset ( blockpos , 1 , 0 , 0 )
if minetest.get_node ( stempos ) . name == connected_stem_names [ 2 ] then
minetest.swap_node ( stempos , { name = full_unconnected_stem } )
end
local stempos = vector.offset ( blockpos , 0 , 0 , - 1 )
if minetest.get_node ( stempos ) . name == connected_stem_names [ 3 ] then
minetest.swap_node ( stempos , { name = full_unconnected_stem } )
end
local stempos = vector.offset ( blockpos , 0 , 0 , 1 )
if minetest.get_node ( stempos ) . name == connected_stem_names [ 4 ] then
minetest.swap_node ( stempos , { name = full_unconnected_stem } )
2017-03-14 00:30:37 +03:00
end
end
end
minetest.register_node ( gourd_itemstring , gourd_def )
2017-03-14 06:17:35 +03:00
-- Register unconnected stem
-- Default values for the stem definition
if not stem_def.selection_box then
2024-09-20 15:00:49 +03:00
stem_def.selection_box = { type = " fixed " , fixed = { { - 0.15 , - 0.5 , - 0.15 , 0.15 , 0.5 , 0.15 } } }
end
stem_def.paramtype = stem_def.paramtype or " light "
stem_def.drawtype = stem_def.drawtype or " plantlike "
stem_def.walkable = stem_def.walkable or false
stem_def.sunlight_propagates = stem_def.sunlight_propagates == nil or stem_def.sunlight_propagates
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.sounds = stem_def.sounds or mcl_sounds.node_sound_leaves_defaults ( )
2017-03-14 06:17:35 +03:00
minetest.register_node ( stem_itemstring , stem_def )
2024-10-16 01:24:07 +03:00
plant_nodename_to_id [ stem_itemstring ] = stem_itemstring
2017-03-14 06:17:35 +03:00
2017-03-14 00:30:37 +03:00
-- Register connected stems
local connected_stem_tiles = {
2024-09-20 15:00:49 +03:00
{ " blank.png " , -- top
2022-11-16 15:17:51 +03:00
" blank.png " , -- bottom
" blank.png " , -- right
" blank.png " , -- left
connected_stem_texture , -- back
2024-09-20 15:00:49 +03:00
connected_stem_texture .. " ^[transformFX " -- front
2017-03-14 00:30:37 +03:00
} ,
2024-09-20 15:00:49 +03:00
{ " blank.png " , -- top
2022-11-16 15:17:51 +03:00
" blank.png " , -- bottom
" blank.png " , -- right
" blank.png " , -- left
2024-09-20 15:00:49 +03:00
connected_stem_texture .. " ^[transformFX " , -- back
2022-11-16 15:17:51 +03:00
connected_stem_texture , -- front
2017-03-14 00:30:37 +03:00
} ,
2024-09-20 15:00:49 +03:00
{ " blank.png " , -- top
2022-11-16 15:17:51 +03:00
" blank.png " , -- bottom
2023-05-27 18:27:01 +03:00
connected_stem_texture .. " ^[transformFX " , -- right
2022-11-16 15:17:51 +03:00
connected_stem_texture , -- left
2024-09-20 15:00:49 +03:00
" blank.png " , -- back
2022-11-16 15:17:51 +03:00
" blank.png " , -- front
2017-03-14 00:30:37 +03:00
} ,
2024-09-20 15:00:49 +03:00
{ " blank.png " , -- top
2022-11-16 15:17:51 +03:00
" blank.png " , -- bottom
connected_stem_texture , -- right
2023-05-27 18:27:01 +03:00
connected_stem_texture .. " ^[transformFX " , -- left
2024-09-20 15:00:49 +03:00
" blank.png " , -- back
2022-11-16 15:17:51 +03:00
" blank.png " , -- front
2017-03-14 00:30:37 +03:00
}
}
local connected_stem_nodebox = {
2022-11-16 15:17:51 +03:00
{ - 0.5 , - 0.5 , 0 , 0.5 , 0.5 , 0 } ,
{ - 0.5 , - 0.5 , 0 , 0.5 , 0.5 , 0 } ,
{ 0 , - 0.5 , - 0.5 , 0 , 0.5 , 0.5 } ,
{ 0 , - 0.5 , - 0.5 , 0 , 0.5 , 0.5 } ,
2017-03-14 00:30:37 +03:00
}
2017-03-14 04:43:48 +03:00
local connected_stem_selectionbox = {
2022-11-16 15:17:51 +03:00
{ - 0.1 , - 0.5 , - 0.1 , 0.5 , 0.2 , 0.1 } ,
{ - 0.5 , - 0.5 , - 0.1 , 0.1 , 0.2 , 0.1 } ,
{ - 0.1 , - 0.5 , - 0.1 , 0.1 , 0.2 , 0.5 } ,
{ - 0.1 , - 0.5 , - 0.5 , 0.1 , 0.2 , 0.1 } ,
2017-03-14 04:43:48 +03:00
}
2017-03-14 00:30:37 +03:00
2022-11-16 15:17:51 +03:00
for i = 1 , 4 do
2017-03-14 00:30:37 +03:00
minetest.register_node ( connected_stem_names [ i ] , {
_doc_items_create_entry = false ,
paramtype = " light " ,
sunlight_propagates = true ,
walkable = false ,
2017-03-14 06:17:35 +03:00
drop = stem_drop ,
2017-03-14 00:30:37 +03:00
drawtype = " nodebox " ,
2024-09-20 15:00:49 +03:00
node_box = { type = " fixed " , fixed = connected_stem_nodebox [ i ] } ,
selection_box = { type = " fixed " , fixed = connected_stem_selectionbox [ i ] } ,
2017-03-14 00:30:37 +03:00
tiles = connected_stem_tiles [ i ] ,
2021-02-18 16:00:17 +03:00
use_texture_alpha = minetest.features . use_texture_alpha_string_modes and " clip " or true ,
2024-09-20 15:00:49 +03:00
groups = { dig_immediate = 3 , not_in_creative_inventory = 1 , plant = 1 , attached_node = 1 , dig_by_water = 1 , destroy_by_lava_flow = 1 } ,
2017-03-14 00:30:37 +03:00
sounds = mcl_sounds.node_sound_leaves_defaults ( ) ,
_mcl_blast_resistance = 0 ,
} )
2024-10-16 01:24:07 +03:00
plant_nodename_to_id [ connected_stem_names [ i ] ] = stem_itemstring
2017-03-21 06:56:16 +03:00
if minetest.get_modpath ( " doc " ) then
doc.add_entry_alias ( " nodes " , full_unconnected_stem , " nodes " , connected_stem_names [ i ] )
end
2017-03-14 00:30:37 +03:00
end
2024-10-10 17:23:54 +03:00
if grow_interval == 0 then return end
2017-03-14 00:30:37 +03:00
minetest.register_abm ( {
2022-11-16 15:17:51 +03:00
label = " Grow gourd stem to gourd ( " .. full_unconnected_stem .. " → " .. gourd_itemstring .. " ) " ,
nodenames = { full_unconnected_stem } ,
neighbors = { " air " } ,
2017-03-14 00:43:47 +03:00
interval = grow_interval ,
chance = grow_chance ,
2017-03-14 00:30:37 +03:00
action = function ( stempos )
2024-10-13 22:42:42 +03:00
local light = minetest.get_node_light ( stempos , 0.5 )
2024-10-09 23:15:30 +03:00
if not light or light < 9 then return end
2024-10-09 19:01:20 +03:00
-- Pick one neighbor and check if it can be used to grow
2024-10-10 18:32:36 +03:00
local dir = random ( 1 , 4 ) -- pick direction at random
2024-10-09 19:01:20 +03:00
local neighbor = ( dir == 1 and vector.offset ( stempos , 1 , 0 , 0 ) )
or ( dir == 2 and vector.offset ( stempos , - 1 , 0 , 0 ) )
or ( dir == 3 and vector.offset ( stempos , 0 , 0 , 1 ) )
or vector.offset ( stempos , 0 , 0 , - 1 )
if minetest.get_node ( neighbor ) . name ~= " air " then return end -- occupied
2024-11-02 23:05:57 +03:00
-- check for suitable floor -- in contrast to MC, we think everything solid is fine
2024-10-09 19:01:20 +03:00
local floorpos = vector.offset ( neighbor , 0 , - 1 , 0 )
local floorname = minetest.get_node ( floorpos ) . name
local floordef = minetest.registered_nodes [ floorname ]
2024-11-02 23:05:57 +03:00
if not floordef or not floordef.walkable then return end
2024-09-20 15:00:49 +03:00
2024-10-09 23:15:30 +03:00
-- check moisture level
2024-10-13 22:42:42 +03:00
local odds = floor ( 25 / ( get_moisture_level ( stempos ) * get_same_crop_penalty ( stempos ) ) ) + 1
2024-10-16 01:24:07 +03:00
-- we triple the odds, and rather call the ABM less often
if random ( ) * odds >= 3 then return end
2024-10-09 23:15:30 +03:00
2024-09-20 15:00:49 +03:00
minetest.swap_node ( stempos , { name = connected_stem_names [ dir ] } )
if gourd_def.paramtype2 == " facedir " then
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 } )
else
minetest.add_node ( neighbor , { name = gourd_itemstring } )
end
-- Reset farmland, etc. to dirt when the gourd grows on top
2024-10-09 19:01:20 +03:00
if ( floordef.groups . dirtifies_below_solid or 0 ) > 0 then
2024-09-20 15:00:49 +03:00
minetest.set_node ( floorpos , { name = " mcl_core:dirt " } )
end
2017-03-14 00:30:37 +03:00
end ,
} )
end
2017-07-21 20:47:20 +03:00
-- Used for growing gourd stems. Returns the intermediate color between startcolor and endcolor at a step
-- * startcolor: ColorSpec in table form for the stem in its lowest growing stage
-- * endcolor: ColorSpec in table form for the stem in its final growing stage
-- * step: The nth growth step. Counting starts at 1
-- * step_count: The number of total growth steps
function mcl_farming : stem_color ( startcolor , endcolor , step , step_count )
2024-09-20 15:00:49 +03:00
local mix = ( step - 1 ) / ( step_count - 1 )
return string.format ( " #%02X%02X%02X " ,
math.max ( 0 , math.min ( 255 , math.round ( ( 1 - mix ) * startcolor.r + mix * endcolor.r ) ) ) ,
math.max ( 0 , math.min ( 255 , math.round ( ( 1 - mix ) * startcolor.g + mix * endcolor.g ) ) ) ,
math.max ( 0 , math.min ( 255 , math.round ( ( 1 - mix ) * startcolor.b + mix * endcolor.b ) ) ) )
2017-07-21 20:47:20 +03:00
end
2020-09-05 20:49:12 +03:00
2022-12-23 17:05:23 +03:00
--[[Get a callback that either eats the item or plants it.
Used for on_place callbacks for craft items which are seeds that can also be consumed .
2022-12-24 21:38:32 +03:00
] ]
2022-12-23 17:05:23 +03:00
function mcl_farming : get_seed_or_eat_callback ( plantname , hp_change )
return function ( itemstack , placer , pointed_thing )
2024-09-20 15:00:49 +03:00
return mcl_farming : place_seed ( itemstack , placer , pointed_thing , plantname )
or minetest.do_item_eat ( hp_change , nil , itemstack , placer , pointed_thing )
2022-12-23 17:05:23 +03:00
end
end
2020-09-05 20:49:12 +03:00
minetest.register_lbm ( {
label = " Add growth for unloaded farming plants " ,
name = " mcl_farming:growth " ,
2022-11-16 15:17:51 +03:00
nodenames = { " group:plant " } ,
2020-09-05 20:49:12 +03:00
run_at_every_load = true ,
2024-09-20 15:00:49 +03:00
action = function ( pos , node , dtime_s )
2024-10-13 22:42:42 +03:00
local identifier = plant_nodename_to_id [ node.name ]
2024-09-20 15:00:49 +03:00
if not identifier then return end
2024-10-16 22:19:38 +03:00
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
2024-10-10 17:23:54 +03:00
end ,
} )