forked from mtcontrib/farming
Code tweak and tidy, fixed rhubarb spawning
This commit is contained in:
parent
dc894725c7
commit
171a67ebf2
12
hoes.lua
12
hoes.lua
@ -2,17 +2,21 @@
|
|||||||
-- Hoe registration function
|
-- Hoe registration function
|
||||||
|
|
||||||
farming.register_hoe = function(name, def)
|
farming.register_hoe = function(name, def)
|
||||||
|
|
||||||
-- Check for : prefix (register new hoes in your mod's namespace)
|
-- Check for : prefix (register new hoes in your mod's namespace)
|
||||||
if name:sub(1,1) ~= ":" then
|
if name:sub(1,1) ~= ":" then
|
||||||
name = ":" .. name
|
name = ":" .. name
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Check def table
|
-- Check def table
|
||||||
if def.description == nil then
|
if def.description == nil then
|
||||||
def.description = "Hoe"
|
def.description = "Hoe"
|
||||||
end
|
end
|
||||||
|
|
||||||
if def.inventory_image == nil then
|
if def.inventory_image == nil then
|
||||||
def.inventory_image = "unknown_item.png"
|
def.inventory_image = "unknown_item.png"
|
||||||
end
|
end
|
||||||
|
|
||||||
if def.recipe == nil then
|
if def.recipe == nil then
|
||||||
def.recipe = {
|
def.recipe = {
|
||||||
{"air","air",""},
|
{"air","air",""},
|
||||||
@ -20,9 +24,11 @@ farming.register_hoe = function(name, def)
|
|||||||
{"","group:stick",""}
|
{"","group:stick",""}
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
if def.max_uses == nil then
|
if def.max_uses == nil then
|
||||||
def.max_uses = 30
|
def.max_uses = 30
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Register the tool
|
-- Register the tool
|
||||||
minetest.register_tool(name, {
|
minetest.register_tool(name, {
|
||||||
description = def.description,
|
description = def.description,
|
||||||
@ -31,6 +37,7 @@ farming.register_hoe = function(name, def)
|
|||||||
return farming.hoe_on_use(itemstack, user, pointed_thing, def.max_uses)
|
return farming.hoe_on_use(itemstack, user, pointed_thing, def.max_uses)
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
|
|
||||||
-- Register its recipe
|
-- Register its recipe
|
||||||
if def.material == nil then
|
if def.material == nil then
|
||||||
minetest.register_craft({
|
minetest.register_craft({
|
||||||
@ -52,7 +59,9 @@ end
|
|||||||
-- Turns dirt with group soil=1 into soil
|
-- Turns dirt with group soil=1 into soil
|
||||||
|
|
||||||
function farming.hoe_on_use(itemstack, user, pointed_thing, uses)
|
function farming.hoe_on_use(itemstack, user, pointed_thing, uses)
|
||||||
|
|
||||||
local pt = pointed_thing
|
local pt = pointed_thing
|
||||||
|
|
||||||
-- check if pointing at a node
|
-- check if pointing at a node
|
||||||
if not pt or pt.type ~= "node" then
|
if not pt or pt.type ~= "node" then
|
||||||
return
|
return
|
||||||
@ -87,10 +96,13 @@ function farming.hoe_on_use(itemstack, user, pointed_thing, uses)
|
|||||||
|
|
||||||
-- turn the node into soil, wear out item and play sound
|
-- turn the node into soil, wear out item and play sound
|
||||||
minetest.set_node(pt.under, {name = "farming:soil"})
|
minetest.set_node(pt.under, {name = "farming:soil"})
|
||||||
|
|
||||||
minetest.sound_play("default_dig_crumbly", {pos = pt.under, gain = 0.5})
|
minetest.sound_play("default_dig_crumbly", {pos = pt.under, gain = 0.5})
|
||||||
|
|
||||||
if not minetest.setting_getbool("creative_mode") then
|
if not minetest.setting_getbool("creative_mode") then
|
||||||
itemstack:add_wear(65535/(uses-1))
|
itemstack:add_wear(65535/(uses-1))
|
||||||
end
|
end
|
||||||
|
|
||||||
return itemstack
|
return itemstack
|
||||||
end
|
end
|
||||||
|
|
||||||
|
125
init.lua
125
init.lua
@ -1,5 +1,5 @@
|
|||||||
--[[
|
--[[
|
||||||
Minetest Farming Redo Mod 1.22 (26th October 2015)
|
Minetest Farming Redo Mod 1.22 (8th December 2015)
|
||||||
by TenPlus1
|
by TenPlus1
|
||||||
NEW growing routine by prestidigitator
|
NEW growing routine by prestidigitator
|
||||||
auto-refill by crabman77
|
auto-refill by crabman77
|
||||||
@ -21,7 +21,9 @@ local DEBUG_abm_runs = 0
|
|||||||
local DEBUG_abm_time = 0
|
local DEBUG_abm_time = 0
|
||||||
local DEBUG_timer_runs = 0
|
local DEBUG_timer_runs = 0
|
||||||
local DEBUG_timer_time = 0
|
local DEBUG_timer_time = 0
|
||||||
|
|
||||||
if farming.DEBUG then
|
if farming.DEBUG then
|
||||||
|
|
||||||
function farming.DEBUG.reset_times()
|
function farming.DEBUG.reset_times()
|
||||||
DEBUG_abm_runs = 0
|
DEBUG_abm_runs = 0
|
||||||
DEBUG_abm_time = 0
|
DEBUG_abm_time = 0
|
||||||
@ -30,6 +32,7 @@ if farming.DEBUG then
|
|||||||
end
|
end
|
||||||
|
|
||||||
function farming.DEBUG.report_times()
|
function farming.DEBUG.report_times()
|
||||||
|
|
||||||
local abm_n = DEBUG_abm_runs
|
local abm_n = DEBUG_abm_runs
|
||||||
local abm_dt = DEBUG_abm_time
|
local abm_dt = DEBUG_abm_time
|
||||||
local abm_avg = (abm_n > 0 and abm_dt / abm_n) or 0
|
local abm_avg = (abm_n > 0 and abm_dt / abm_n) or 0
|
||||||
@ -37,15 +40,15 @@ if farming.DEBUG then
|
|||||||
local timer_dt = DEBUG_timer_time
|
local timer_dt = DEBUG_timer_time
|
||||||
local timer_avg = (timer_n > 0 and timer_dt / timer_n) or 0
|
local timer_avg = (timer_n > 0 and timer_dt / timer_n) or 0
|
||||||
local dt = abm_dt + timer_dt
|
local dt = abm_dt + timer_dt
|
||||||
print("ABM ran for "..abm_dt.."µs over "..abm_n.." runs: "..
|
|
||||||
abm_avg.."µs/run")
|
print("ABM ran for "..abm_dt.."µs over "..abm_n.." runs: "..abm_avg.."µs/run")
|
||||||
print("Timer ran for "..timer_dt.."µs over "..timer_n.." runs: "..
|
print("Timer ran for "..timer_dt.."µs over "..timer_n.." runs: "..timer_avg.."µs/run")
|
||||||
timer_avg.."µs/run")
|
|
||||||
print("Total farming time: "..dt.."µs")
|
print("Total farming time: "..dt.."µs")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local statistics = dofile(farming.path.."/statistics.lua")
|
local statistics = dofile(farming.path.."/statistics.lua")
|
||||||
|
|
||||||
dofile(farming.path.."/soil.lua")
|
dofile(farming.path.."/soil.lua")
|
||||||
dofile(farming.path.."/hoes.lua")
|
dofile(farming.path.."/hoes.lua")
|
||||||
dofile(farming.path.."/grass.lua")
|
dofile(farming.path.."/grass.lua")
|
||||||
@ -95,12 +98,14 @@ end
|
|||||||
-- If true, count elapsed day time. Otherwise, count elapsed night time.
|
-- If true, count elapsed day time. Otherwise, count elapsed night time.
|
||||||
-- @return
|
-- @return
|
||||||
-- The amount of day or night time that has elapsed.
|
-- The amount of day or night time that has elapsed.
|
||||||
--
|
|
||||||
local function day_or_night_time(t_game, t_day, dt, count_day)
|
|
||||||
local t1_day = t_day - dt / SECS_PER_CYCLE
|
|
||||||
|
|
||||||
|
local function day_or_night_time(t_game, t_day, dt, count_day)
|
||||||
|
|
||||||
|
local t1_day = t_day - dt / SECS_PER_CYCLE
|
||||||
local t1_c, t2_c -- t1_c < t2_c and t2_c always in [0, 1)
|
local t1_c, t2_c -- t1_c < t2_c and t2_c always in [0, 1)
|
||||||
|
|
||||||
if count_day then
|
if count_day then
|
||||||
|
|
||||||
if t_day < 0.25 then
|
if t_day < 0.25 then
|
||||||
t1_c = t1_day + 0.75 -- Relative to sunup, yesterday
|
t1_c = t1_day + 0.75 -- Relative to sunup, yesterday
|
||||||
t2_c = t_day + 0.75
|
t2_c = t_day + 0.75
|
||||||
@ -119,6 +124,7 @@ local function day_or_night_time(t_game, t_day, dt, count_day)
|
|||||||
end
|
end
|
||||||
|
|
||||||
local dt_c = clamp(t2_c, 0, 0.5) - clamp(t1_c, 0, 0.5) -- this cycle
|
local dt_c = clamp(t2_c, 0, 0.5) - clamp(t1_c, 0, 0.5) -- this cycle
|
||||||
|
|
||||||
if t1_c < -0.5 then
|
if t1_c < -0.5 then
|
||||||
local nc = math.floor(-t1_c)
|
local nc = math.floor(-t1_c)
|
||||||
t1_c = t1_c + nc
|
t1_c = t1_c + nc
|
||||||
@ -166,10 +172,13 @@ local MAX_LIGHT = 1000
|
|||||||
-- Node or position table, or node name.
|
-- Node or position table, or node name.
|
||||||
-- @return
|
-- @return
|
||||||
-- List (plant_name, stage), or nothing (nil) if node isn't loaded
|
-- List (plant_name, stage), or nothing (nil) if node isn't loaded
|
||||||
--
|
|
||||||
local function plant_name_stage(node)
|
local function plant_name_stage(node)
|
||||||
|
|
||||||
local name
|
local name
|
||||||
|
|
||||||
if type(node) == 'table' then
|
if type(node) == 'table' then
|
||||||
|
|
||||||
if node.name then
|
if node.name then
|
||||||
name = node.name
|
name = node.name
|
||||||
elseif node.x and node.y and node.z then
|
elseif node.x and node.y and node.z then
|
||||||
@ -179,11 +188,15 @@ local function plant_name_stage(node)
|
|||||||
else
|
else
|
||||||
name = tostring(node)
|
name = tostring(node)
|
||||||
end
|
end
|
||||||
|
|
||||||
if not name or name == "ignore" then return nil end
|
if not name or name == "ignore" then return nil end
|
||||||
|
|
||||||
local sep_pos = name:find("_[^_]+$")
|
local sep_pos = name:find("_[^_]+$")
|
||||||
|
|
||||||
if sep_pos and sep_pos > 1 then
|
if sep_pos and sep_pos > 1 then
|
||||||
|
|
||||||
local stage = tonumber(name:sub(sep_pos + 1))
|
local stage = tonumber(name:sub(sep_pos + 1))
|
||||||
|
|
||||||
if stage and stage >= 0 then
|
if stage and stage >= 0 then
|
||||||
return name:sub(1, sep_pos - 1), stage
|
return name:sub(1, sep_pos - 1), stage
|
||||||
end
|
end
|
||||||
@ -195,6 +208,7 @@ end
|
|||||||
--- Map from node name to
|
--- Map from node name to
|
||||||
-- { plant_name = ..., name = ..., stage = n, stages_left = { node_name, ... } }
|
-- { plant_name = ..., name = ..., stage = n, stages_left = { node_name, ... } }
|
||||||
local plant_stages = {}
|
local plant_stages = {}
|
||||||
|
|
||||||
farming.plant_stages = plant_stages
|
farming.plant_stages = plant_stages
|
||||||
|
|
||||||
--- Registers the stages of growth of a (possible plant) node.
|
--- Registers the stages of growth of a (possible plant) node.
|
||||||
@ -204,21 +218,26 @@ farming.plant_stages = plant_stages
|
|||||||
-- @return
|
-- @return
|
||||||
-- The (possibly zero) number of stages of growth the plant will go through
|
-- The (possibly zero) number of stages of growth the plant will go through
|
||||||
-- before being fully grown, or nil if not a plant.
|
-- before being fully grown, or nil if not a plant.
|
||||||
--
|
|
||||||
local register_plant_node
|
local register_plant_node
|
||||||
|
|
||||||
-- Recursive helper
|
-- Recursive helper
|
||||||
local function reg_plant_stages(plant_name, stage, force_last)
|
local function reg_plant_stages(plant_name, stage, force_last)
|
||||||
|
|
||||||
local node_name = plant_name and plant_name .. "_" .. stage
|
local node_name = plant_name and plant_name .. "_" .. stage
|
||||||
local node_def = node_name and minetest.registered_nodes[node_name]
|
local node_def = node_name and minetest.registered_nodes[node_name]
|
||||||
|
|
||||||
if not node_def then return nil end
|
if not node_def then return nil end
|
||||||
|
|
||||||
local stages = plant_stages[node_name]
|
local stages = plant_stages[node_name]
|
||||||
|
|
||||||
if stages then return stages end
|
if stages then return stages end
|
||||||
|
|
||||||
if minetest.get_item_group(node_name, "growing") > 0 then
|
if minetest.get_item_group(node_name, "growing") > 0 then
|
||||||
local ns = reg_plant_stages(plant_name, stage + 1, true)
|
|
||||||
|
|
||||||
|
local ns = reg_plant_stages(plant_name, stage + 1, true)
|
||||||
local stages_left = (ns and { ns.name, unpack(ns.stages_left) }) or {}
|
local stages_left = (ns and { ns.name, unpack(ns.stages_left) }) or {}
|
||||||
|
|
||||||
stages = {
|
stages = {
|
||||||
plant_name = plant_name,
|
plant_name = plant_name,
|
||||||
name = node_name,
|
name = node_name,
|
||||||
@ -227,8 +246,10 @@ local function reg_plant_stages(plant_name, stage, force_last)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if #stages_left > 0 then
|
if #stages_left > 0 then
|
||||||
|
|
||||||
local old_constr = node_def.on_construct
|
local old_constr = node_def.on_construct
|
||||||
local old_destr = node_def.on_destruct
|
local old_destr = node_def.on_destruct
|
||||||
|
|
||||||
minetest.override_item(node_name,
|
minetest.override_item(node_name,
|
||||||
{
|
{
|
||||||
on_construct = function(pos)
|
on_construct = function(pos)
|
||||||
@ -246,7 +267,9 @@ local function reg_plant_stages(plant_name, stage, force_last)
|
|||||||
end,
|
end,
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
elseif force_last then
|
elseif force_last then
|
||||||
|
|
||||||
stages = {
|
stages = {
|
||||||
plant_name = plant_name,
|
plant_name = plant_name,
|
||||||
name = node_name,
|
name = node_name,
|
||||||
@ -258,11 +281,14 @@ local function reg_plant_stages(plant_name, stage, force_last)
|
|||||||
end
|
end
|
||||||
|
|
||||||
plant_stages[node_name] = stages
|
plant_stages[node_name] = stages
|
||||||
|
|
||||||
return stages
|
return stages
|
||||||
end
|
end
|
||||||
|
|
||||||
register_plant_node = function(node)
|
register_plant_node = function(node)
|
||||||
|
|
||||||
local plant_name, stage = plant_name_stage(node)
|
local plant_name, stage = plant_name_stage(node)
|
||||||
|
|
||||||
if plant_name then
|
if plant_name then
|
||||||
local stages = reg_plant_stages(plant_name, stage, false)
|
local stages = reg_plant_stages(plant_name, stage, false)
|
||||||
return stages and #stages.stages_left
|
return stages and #stages.stages_left
|
||||||
@ -272,15 +298,22 @@ register_plant_node = function(node)
|
|||||||
end
|
end
|
||||||
|
|
||||||
local function set_growing(pos, stages_left)
|
local function set_growing(pos, stages_left)
|
||||||
|
|
||||||
if not stages_left then return end
|
if not stages_left then return end
|
||||||
|
|
||||||
local timer = minetest.get_node_timer(pos)
|
local timer = minetest.get_node_timer(pos)
|
||||||
|
|
||||||
if stages_left > 0 then
|
if stages_left > 0 then
|
||||||
|
|
||||||
if not timer:is_started() then
|
if not timer:is_started() then
|
||||||
|
|
||||||
local stage_length = statistics.normal(STAGE_LENGTH_AVG, STAGE_LENGTH_DEV)
|
local stage_length = statistics.normal(STAGE_LENGTH_AVG, STAGE_LENGTH_DEV)
|
||||||
|
|
||||||
stage_length = clamp(stage_length, 0.5 * STAGE_LENGTH_AVG, 3.0 * STAGE_LENGTH_AVG)
|
stage_length = clamp(stage_length, 0.5 * STAGE_LENGTH_AVG, 3.0 * STAGE_LENGTH_AVG)
|
||||||
|
|
||||||
timer:set(stage_length, -0.5 * math.random() * STAGE_LENGTH_AVG)
|
timer:set(stage_length, -0.5 * math.random() * STAGE_LENGTH_AVG)
|
||||||
end
|
end
|
||||||
|
|
||||||
elseif timer:is_started() then
|
elseif timer:is_started() then
|
||||||
timer:stop()
|
timer:stop()
|
||||||
end
|
end
|
||||||
@ -292,10 +325,13 @@ end
|
|||||||
-- The node's position.
|
-- The node's position.
|
||||||
-- @param node
|
-- @param node
|
||||||
-- The cached node table if available, or nil.
|
-- The cached node table if available, or nil.
|
||||||
--
|
|
||||||
function farming.handle_growth(pos, node)
|
function farming.handle_growth(pos, node)
|
||||||
|
|
||||||
if not pos then return end
|
if not pos then return end
|
||||||
|
|
||||||
local stages_left = register_plant_node(node or pos)
|
local stages_left = register_plant_node(node or pos)
|
||||||
|
|
||||||
if stages_left then set_growing(pos, stages_left) end
|
if stages_left then set_growing(pos, stages_left) end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -307,14 +343,19 @@ minetest.after(0,
|
|||||||
end)
|
end)
|
||||||
|
|
||||||
local abm_func = farming.handle_growth
|
local abm_func = farming.handle_growth
|
||||||
|
|
||||||
if farming.DEBUG then
|
if farming.DEBUG then
|
||||||
|
|
||||||
local normal_abm_func = abm_func
|
local normal_abm_func = abm_func
|
||||||
|
|
||||||
abm_func = function(...)
|
abm_func = function(...)
|
||||||
local t0 = minetest.get_us_time()
|
local t0 = minetest.get_us_time()
|
||||||
local r = { normal_abm_func(...) }
|
local r = { normal_abm_func(...) }
|
||||||
local t1 = minetest.get_us_time()
|
local t1 = minetest.get_us_time()
|
||||||
|
|
||||||
DEBUG_abm_runs = DEBUG_abm_runs + 1
|
DEBUG_abm_runs = DEBUG_abm_runs + 1
|
||||||
DEBUG_abm_time = DEBUG_abm_time + (t1 - t0)
|
DEBUG_abm_time = DEBUG_abm_time + (t1 - t0)
|
||||||
|
|
||||||
return unpack(r)
|
return unpack(r)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -322,40 +363,51 @@ end
|
|||||||
-- Just in case a growing type or added node is missed (also catches existing
|
-- Just in case a growing type or added node is missed (also catches existing
|
||||||
-- nodes added to map before timers were incorporated).
|
-- nodes added to map before timers were incorporated).
|
||||||
minetest.register_abm({
|
minetest.register_abm({
|
||||||
|
|
||||||
nodenames = { "group:growing" },
|
nodenames = { "group:growing" },
|
||||||
interval = 300,
|
interval = 300,
|
||||||
chance = 1,
|
chance = 1,
|
||||||
action = abm_func})
|
action = abm_func
|
||||||
|
})
|
||||||
|
|
||||||
--- Plant timer function.
|
--- Plant timer function.
|
||||||
--
|
--
|
||||||
-- Grows plants under the right conditions.
|
-- Grows plants under the right conditions.
|
||||||
--
|
|
||||||
function farming.plant_growth_timer(pos, elapsed, node_name)
|
function farming.plant_growth_timer(pos, elapsed, node_name)
|
||||||
|
|
||||||
local stages = plant_stages[node_name]
|
local stages = plant_stages[node_name]
|
||||||
|
|
||||||
if not stages then return false end
|
if not stages then return false end
|
||||||
|
|
||||||
local max_growth = #stages.stages_left
|
local max_growth = #stages.stages_left
|
||||||
|
|
||||||
if max_growth <= 0 then return false end
|
if max_growth <= 0 then return false end
|
||||||
|
|
||||||
if stages.plant_name == "farming:cocoa" then
|
if stages.plant_name == "farming:cocoa" then
|
||||||
|
|
||||||
if not minetest.find_node_near(pos, 1, { "default:jungletree", "moretrees:jungletree_leaves_green" }) then
|
if not minetest.find_node_near(pos, 1, { "default:jungletree", "moretrees:jungletree_leaves_green" }) then
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
local under = minetest.get_node_or_nil({ x = pos.x, y = pos.y - 1, z = pos.z })
|
local under = minetest.get_node_or_nil({ x = pos.x, y = pos.y - 1, z = pos.z })
|
||||||
|
|
||||||
if not under or under.name ~= "farming:soil_wet" then return true end
|
if not under or under.name ~= "farming:soil_wet" then return true end
|
||||||
end
|
end
|
||||||
|
|
||||||
local growth
|
local growth
|
||||||
|
|
||||||
local light_pos = { x = pos.x, y = pos.y, z = pos.z }
|
local light_pos = { x = pos.x, y = pos.y, z = pos.z }
|
||||||
local lambda = elapsed / STAGE_LENGTH_AVG
|
local lambda = elapsed / STAGE_LENGTH_AVG
|
||||||
|
|
||||||
if lambda < 0.1 then return true end
|
if lambda < 0.1 then return true end
|
||||||
|
|
||||||
if max_growth == 1 or lambda < 2.0 then
|
if max_growth == 1 or lambda < 2.0 then
|
||||||
|
|
||||||
local light = (minetest.get_node_light(light_pos) or 0) -- CHANGED
|
local light = (minetest.get_node_light(light_pos) or 0) -- CHANGED
|
||||||
--print ("light level:", light)
|
--print ("light level:", light)
|
||||||
|
|
||||||
if not in_range(light, MIN_LIGHT, MAX_LIGHT) then return true end
|
if not in_range(light, MIN_LIGHT, MAX_LIGHT) then return true end
|
||||||
|
|
||||||
growth = 1
|
growth = 1
|
||||||
else
|
else
|
||||||
local night_light = (minetest.get_node_light(light_pos, 0) or 0) -- CHANGED
|
local night_light = (minetest.get_node_light(light_pos, 0) or 0) -- CHANGED
|
||||||
@ -371,24 +423,32 @@ function farming.plant_growth_timer(pos, elapsed, node_name)
|
|||||||
end
|
end
|
||||||
|
|
||||||
growth = statistics.poisson(lambda, max_growth)
|
growth = statistics.poisson(lambda, max_growth)
|
||||||
|
|
||||||
if growth < 1 then return true end
|
if growth < 1 then return true end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if minetest.registered_nodes[stages.stages_left[growth]] then
|
||||||
minetest.swap_node(pos, { name = stages.stages_left[growth] })
|
minetest.swap_node(pos, { name = stages.stages_left[growth] })
|
||||||
|
else
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
return growth ~= max_growth
|
return growth ~= max_growth
|
||||||
end
|
end
|
||||||
|
|
||||||
if farming.DEBUG then
|
if farming.DEBUG then
|
||||||
|
|
||||||
local timer_func = farming.plant_growth_timer;
|
local timer_func = farming.plant_growth_timer;
|
||||||
|
|
||||||
farming.plant_growth_timer = function(pos, elapsed, node_name)
|
farming.plant_growth_timer = function(pos, elapsed, node_name)
|
||||||
|
|
||||||
local t0 = minetest.get_us_time()
|
local t0 = minetest.get_us_time()
|
||||||
|
|
||||||
local r = { timer_func(pos, elapsed, node_name) }
|
local r = { timer_func(pos, elapsed, node_name) }
|
||||||
|
|
||||||
local t1 = minetest.get_us_time()
|
local t1 = minetest.get_us_time()
|
||||||
|
|
||||||
DEBUG_timer_runs = DEBUG_timer_runs + 1
|
DEBUG_timer_runs = DEBUG_timer_runs + 1
|
||||||
DEBUG_timer_time = DEBUG_timer_time + (t1 - t0)
|
DEBUG_timer_time = DEBUG_timer_time + (t1 - t0)
|
||||||
|
|
||||||
return unpack(r)
|
return unpack(r)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -407,15 +467,24 @@ local can_refill_plant = {
|
|||||||
["farming:raspberry_1"] = "farming:raspberries",
|
["farming:raspberry_1"] = "farming:raspberries",
|
||||||
["farming:rhubarb_1"] = "farming:rhubarb",
|
["farming:rhubarb_1"] = "farming:rhubarb",
|
||||||
["farming:tomato_1"] = "farming:tomato",
|
["farming:tomato_1"] = "farming:tomato",
|
||||||
["farming:wheat_1"] = "farming:seed_wheat"
|
["farming:wheat_1"] = "farming:seed_wheat",
|
||||||
|
["farming:grapes_1"] = "farming:grapes",
|
||||||
|
["farming:beans_1"] = "farming:beans",
|
||||||
|
["farming:rhubarb_1"] = "farming:rhubarb",
|
||||||
|
["farming:cocoa_1"] = "farming:cocoa_beans",
|
||||||
}
|
}
|
||||||
|
|
||||||
function farming.refill_plant(player, plantname, index)
|
function farming.refill_plant(player, plantname, index)
|
||||||
|
|
||||||
local inv = player:get_inventory()
|
local inv = player:get_inventory()
|
||||||
local old_stack = inv:get_stack("main", index)
|
local old_stack = inv:get_stack("main", index)
|
||||||
|
|
||||||
if old_stack:get_name() ~= "" then return end
|
if old_stack:get_name() ~= "" then return end
|
||||||
for i,stack in ipairs(inv:get_list("main")) do
|
|
||||||
|
for i, stack in ipairs(inv:get_list("main")) do
|
||||||
|
|
||||||
if stack:get_name() == plantname and i ~= index then
|
if stack:get_name() == plantname and i ~= index then
|
||||||
|
|
||||||
inv:set_stack("main", index, stack)
|
inv:set_stack("main", index, stack)
|
||||||
stack:clear()
|
stack:clear()
|
||||||
inv:set_stack("main", i, stack)
|
inv:set_stack("main", i, stack)
|
||||||
@ -428,6 +497,7 @@ end -- END refill
|
|||||||
-- Place Seeds on Soil
|
-- Place Seeds on Soil
|
||||||
|
|
||||||
function farming.place_seed(itemstack, placer, pointed_thing, plantname)
|
function farming.place_seed(itemstack, placer, pointed_thing, plantname)
|
||||||
|
|
||||||
local pt = pointed_thing
|
local pt = pointed_thing
|
||||||
|
|
||||||
-- check if pointing at a node
|
-- check if pointing at a node
|
||||||
@ -459,20 +529,26 @@ function farming.place_seed(itemstack, placer, pointed_thing, plantname)
|
|||||||
|
|
||||||
-- if not protected then add node and remove 1 item from the itemstack
|
-- if not protected then add node and remove 1 item from the itemstack
|
||||||
if not minetest.is_protected(pt.above, placer:get_player_name()) then
|
if not minetest.is_protected(pt.above, placer:get_player_name()) then
|
||||||
|
|
||||||
minetest.set_node(pt.above, {name = plantname, param2 = 1})
|
minetest.set_node(pt.above, {name = plantname, param2 = 1})
|
||||||
|
|
||||||
if not minetest.setting_getbool("creative_mode") then
|
if not minetest.setting_getbool("creative_mode") then
|
||||||
|
|
||||||
itemstack:take_item()
|
itemstack:take_item()
|
||||||
|
|
||||||
-- check for refill
|
-- check for refill
|
||||||
if itemstack:get_count() == 0
|
if itemstack:get_count() == 0
|
||||||
and can_refill_plant[plantname] then
|
and can_refill_plant[plantname] then
|
||||||
|
|
||||||
minetest.after(0.10,
|
minetest.after(0.10,
|
||||||
farming.refill_plant,
|
farming.refill_plant,
|
||||||
placer,
|
placer,
|
||||||
can_refill_plant[plantname],
|
can_refill_plant[plantname],
|
||||||
placer:get_wield_index()
|
placer:get_wield_index()
|
||||||
)
|
)
|
||||||
end -- END refill
|
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
return itemstack
|
return itemstack
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -480,6 +556,7 @@ end
|
|||||||
-- Function to register plants (for compatibility)
|
-- Function to register plants (for compatibility)
|
||||||
|
|
||||||
farming.register_plant = function(name, def)
|
farming.register_plant = function(name, def)
|
||||||
|
|
||||||
local mname = name:split(":")[1]
|
local mname = name:split(":")[1]
|
||||||
local pname = name:split(":")[2]
|
local pname = name:split(":")[2]
|
||||||
|
|
||||||
@ -487,15 +564,18 @@ farming.register_plant = function(name, def)
|
|||||||
if not def.description then
|
if not def.description then
|
||||||
def.description = "Seed"
|
def.description = "Seed"
|
||||||
end
|
end
|
||||||
|
|
||||||
if not def.inventory_image then
|
if not def.inventory_image then
|
||||||
def.inventory_image = "unknown_item.png"
|
def.inventory_image = "unknown_item.png"
|
||||||
end
|
end
|
||||||
|
|
||||||
if not def.steps then
|
if not def.steps then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Register seed
|
-- Register seed
|
||||||
minetest.register_node(":" .. mname .. ":seed_" .. pname, {
|
minetest.register_node(":" .. mname .. ":seed_" .. pname, {
|
||||||
|
|
||||||
description = def.description,
|
description = def.description,
|
||||||
tiles = {def.inventory_image},
|
tiles = {def.inventory_image},
|
||||||
inventory_image = def.inventory_image,
|
inventory_image = def.inventory_image,
|
||||||
@ -507,6 +587,7 @@ farming.register_plant = function(name, def)
|
|||||||
walkable = false,
|
walkable = false,
|
||||||
sunlight_propagates = true,
|
sunlight_propagates = true,
|
||||||
selection_box = farming.select,
|
selection_box = farming.select,
|
||||||
|
|
||||||
on_place = function(itemstack, placer, pointed_thing)
|
on_place = function(itemstack, placer, pointed_thing)
|
||||||
return farming.place_seed(itemstack, placer, pointed_thing, mname .. ":"..pname.."_1")
|
return farming.place_seed(itemstack, placer, pointed_thing, mname .. ":"..pname.."_1")
|
||||||
end
|
end
|
||||||
@ -520,6 +601,7 @@ farming.register_plant = function(name, def)
|
|||||||
|
|
||||||
-- Register growing steps
|
-- Register growing steps
|
||||||
for i = 1, def.steps do
|
for i = 1, def.steps do
|
||||||
|
|
||||||
local drop = {
|
local drop = {
|
||||||
items = {
|
items = {
|
||||||
{items = {mname .. ":" .. pname}, rarity = 9 - i},
|
{items = {mname .. ":" .. pname}, rarity = 9 - i},
|
||||||
@ -530,12 +612,14 @@ farming.register_plant = function(name, def)
|
|||||||
}
|
}
|
||||||
|
|
||||||
local g = {snappy = 3, flammable = 2, plant = 1, not_in_creative_inventory = 1, attached_node = 1, growing = 1}
|
local g = {snappy = 3, flammable = 2, plant = 1, not_in_creative_inventory = 1, attached_node = 1, growing = 1}
|
||||||
|
|
||||||
-- Last step doesn't need growing=1 so Abm never has to check these
|
-- Last step doesn't need growing=1 so Abm never has to check these
|
||||||
if i == def.steps then
|
if i == def.steps then
|
||||||
g = {snappy = 3, flammable = 2, plant = 1, not_in_creative_inventory = 1, attached_node = 1}
|
g = {snappy = 3, flammable = 2, plant = 1, not_in_creative_inventory = 1, attached_node = 1}
|
||||||
end
|
end
|
||||||
|
|
||||||
local node_name = mname .. ":" .. pname .. "_" .. i
|
local node_name = mname .. ":" .. pname .. "_" .. i
|
||||||
|
|
||||||
minetest.register_node(node_name, {
|
minetest.register_node(node_name, {
|
||||||
drawtype = "plantlike",
|
drawtype = "plantlike",
|
||||||
waving = 1,
|
waving = 1,
|
||||||
@ -548,6 +632,7 @@ farming.register_plant = function(name, def)
|
|||||||
groups = g,
|
groups = g,
|
||||||
sounds = default.node_sound_leaves_defaults(),
|
sounds = default.node_sound_leaves_defaults(),
|
||||||
})
|
})
|
||||||
|
|
||||||
register_plant_node(node_name)
|
register_plant_node(node_name)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -31,13 +31,13 @@ function farming.register_mgv6_decorations()
|
|||||||
register_plant("melon_8", 1, 20, "group:water", 1)
|
register_plant("melon_8", 1, 20, "group:water", 1)
|
||||||
register_plant("pumpkin_8", 1, 20, "group:water", 1)
|
register_plant("pumpkin_8", 1, 20, "group:water", 1)
|
||||||
register_plant("raspberry_4", 3, 10, "", -1)
|
register_plant("raspberry_4", 3, 10, "", -1)
|
||||||
register_plant("rhubarb_3", 3, 15, "group:tree", 1)
|
register_plant("rhubarb_3", 3, 15, "", -1)
|
||||||
register_plant("blueberry_4", 3, 10, "", -1)
|
register_plant("blueberry_4", 3, 10, "", -1)
|
||||||
register_plant("beanbush", 18, 35, "", -1)
|
register_plant("beanbush", 18, 35, "", -1)
|
||||||
register_plant("grapebush", 25, 45, "", -1)
|
register_plant("grapebush", 25, 45, "", -1)
|
||||||
end
|
end
|
||||||
|
|
||||||
-- v7 maps have a beach so plants growing near water is limited to 6- high
|
-- v7 maps have a beach so plants growing near water is limited to 6 high
|
||||||
function farming.register_mgv7_decorations()
|
function farming.register_mgv7_decorations()
|
||||||
register_plant("potato_3", 15, 40, "", -1)
|
register_plant("potato_3", 15, 40, "", -1)
|
||||||
register_plant("tomato_7", 5, 20, "", -1)
|
register_plant("tomato_7", 5, 20, "", -1)
|
||||||
@ -49,7 +49,7 @@ function farming.register_mgv7_decorations()
|
|||||||
register_plant("melon_8", 1, 6, "", -1)
|
register_plant("melon_8", 1, 6, "", -1)
|
||||||
register_plant("pumpkin_8", 1, 6, "", -1)
|
register_plant("pumpkin_8", 1, 6, "", -1)
|
||||||
register_plant("raspberry_4", 3, 10, "", -1)
|
register_plant("raspberry_4", 3, 10, "", -1)
|
||||||
register_plant("rhubarb_3", 3, 15, "group:tree", 1)
|
register_plant("rhubarb_3", 3, 15, "", -1)
|
||||||
register_plant("blueberry_4", 3, 10, "", -1)
|
register_plant("blueberry_4", 3, 10, "", -1)
|
||||||
register_plant("beanbush", 18, 35, "", -1)
|
register_plant("beanbush", 18, 35, "", -1)
|
||||||
register_plant("grapebush", 25, 45, "", -1)
|
register_plant("grapebush", 25, 45, "", -1)
|
||||||
|
3
soil.lua
3
soil.lua
@ -26,6 +26,7 @@ minetest.register_abm({
|
|||||||
interval = 15,
|
interval = 15,
|
||||||
chance = 4,
|
chance = 4,
|
||||||
catch_up = false,
|
catch_up = false,
|
||||||
|
|
||||||
action = function(pos, node)
|
action = function(pos, node)
|
||||||
|
|
||||||
pos.y = pos.y + 1
|
pos.y = pos.y + 1
|
||||||
@ -52,8 +53,10 @@ minetest.register_abm({
|
|||||||
if node.name == "farming:soil" then
|
if node.name == "farming:soil" then
|
||||||
minetest.set_node(pos, {name = "farming:soil_wet"})
|
minetest.set_node(pos, {name = "farming:soil_wet"})
|
||||||
end
|
end
|
||||||
|
|
||||||
elseif node.name == "farming:soil_wet" then
|
elseif node.name == "farming:soil_wet" then
|
||||||
minetest.set_node(pos, {name = "farming:soil"})
|
minetest.set_node(pos, {name = "farming:soil"})
|
||||||
|
|
||||||
elseif node.name == "farming:soil" then
|
elseif node.name == "farming:soil" then
|
||||||
minetest.set_node(pos, {name = "default:dirt"})
|
minetest.set_node(pos, {name = "default:dirt"})
|
||||||
end
|
end
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
local statistics = {}
|
local statistics = {}
|
||||||
|
|
||||||
local ROOT_2 = math.sqrt(2.0)
|
local ROOT_2 = math.sqrt(2.0)
|
||||||
|
|
||||||
-- Approximations for erf(x) and erfInv(x) from
|
-- Approximations for erf(x) and erfInv(x) from
|
||||||
-- https://en.wikipedia.org/wiki/Error_function
|
-- https://en.wikipedia.org/wiki/Error_function
|
||||||
|
|
||||||
local erf
|
local erf
|
||||||
local erf_inv
|
local erf_inv
|
||||||
|
|
||||||
@ -13,19 +13,26 @@ local C = 2.0/(math.pi * A)
|
|||||||
local D = 1.0 / A
|
local D = 1.0 / A
|
||||||
|
|
||||||
erf = function(x)
|
erf = function(x)
|
||||||
|
|
||||||
if x == 0 then return 0; end
|
if x == 0 then return 0; end
|
||||||
|
|
||||||
local xSq = x * x
|
local xSq = x * x
|
||||||
local aXSq = A * xSq
|
local aXSq = A * xSq
|
||||||
local v = math.sqrt(1.0 - math.exp(-xSq * (B + aXSq) / (1.0 + aXSq)))
|
local v = math.sqrt(1.0 - math.exp(-xSq * (B + aXSq) / (1.0 + aXSq)))
|
||||||
|
|
||||||
return (x > 0 and v) or -v
|
return (x > 0 and v) or -v
|
||||||
end
|
end
|
||||||
|
|
||||||
erf_inv = function(x)
|
erf_inv = function(x)
|
||||||
|
|
||||||
if x == 0 then return 0; end
|
if x == 0 then return 0; end
|
||||||
|
|
||||||
if x <= -1 or x >= 1 then return nil; end
|
if x <= -1 or x >= 1 then return nil; end
|
||||||
|
|
||||||
local y = math.log(1 - x * x)
|
local y = math.log(1 - x * x)
|
||||||
local u = C + 0.5 * y
|
local u = C + 0.5 * y
|
||||||
local v = math.sqrt(math.sqrt(u * u - D * y) - u)
|
local v = math.sqrt(math.sqrt(u * u - D * y) - u)
|
||||||
|
|
||||||
return (x > 0 and v) or -v
|
return (x > 0 and v) or -v
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -37,6 +44,7 @@ local poisson
|
|||||||
local cdf_table = {}
|
local cdf_table = {}
|
||||||
|
|
||||||
local function generate_cdf(lambda_index, lambda)
|
local function generate_cdf(lambda_index, lambda)
|
||||||
|
|
||||||
local max = math.ceil(4 * lambda)
|
local max = math.ceil(4 * lambda)
|
||||||
local pdf = math.exp(-lambda)
|
local pdf = math.exp(-lambda)
|
||||||
local cdf = pdf
|
local cdf = pdf
|
||||||
@ -56,6 +64,7 @@ for li = 1, 100 do
|
|||||||
end
|
end
|
||||||
|
|
||||||
poisson = function(lambda, max)
|
poisson = function(lambda, max)
|
||||||
|
|
||||||
if max < 2 then
|
if max < 2 then
|
||||||
return (math.random() < math.exp(-lambda) and 0) or 1
|
return (math.random() < math.exp(-lambda) and 0) or 1
|
||||||
elseif lambda >= 2 * max then
|
elseif lambda >= 2 * max then
|
||||||
@ -67,6 +76,7 @@ poisson = function(lambda, max)
|
|||||||
local cdfs = cdf_table[lambda_index]
|
local cdfs = cdf_table[lambda_index]
|
||||||
|
|
||||||
if cdfs then
|
if cdfs then
|
||||||
|
|
||||||
lambda = 0.25 * lambda_index
|
lambda = 0.25 * lambda_index
|
||||||
|
|
||||||
if u < cdfs[0] then return 0; end
|
if u < cdfs[0] then return 0; end
|
||||||
@ -74,9 +84,13 @@ poisson = function(lambda, max)
|
|||||||
if u >= cdfs[max - 1] then return max; end
|
if u >= cdfs[max - 1] then return max; end
|
||||||
|
|
||||||
if max > 4 then -- Binary search
|
if max > 4 then -- Binary search
|
||||||
|
|
||||||
local s = 0
|
local s = 0
|
||||||
|
|
||||||
while s + 1 < max do
|
while s + 1 < max do
|
||||||
|
|
||||||
local m = math.floor(0.5 * (s + max))
|
local m = math.floor(0.5 * (s + max))
|
||||||
|
|
||||||
if u < cdfs[m] then max = m; else s = m; end
|
if u < cdfs[m] then max = m; else s = m; end
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
@ -88,6 +102,7 @@ poisson = function(lambda, max)
|
|||||||
return max
|
return max
|
||||||
else
|
else
|
||||||
local x = lambda + math.sqrt(lambda) * std_normal(u)
|
local x = lambda + math.sqrt(lambda) * std_normal(u)
|
||||||
|
|
||||||
return (x < 0.5 and 0) or (x >= max - 0.5 and max) or math.floor(x + 0.5)
|
return (x < 0.5 and 0) or (x >= max - 0.5 and max) or math.floor(x + 0.5)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -102,14 +117,17 @@ statistics.erf_inv = erf_inv
|
|||||||
--
|
--
|
||||||
-- @return
|
-- @return
|
||||||
-- Any real number (actually between -3.0 and 3.0).
|
-- Any real number (actually between -3.0 and 3.0).
|
||||||
--
|
|
||||||
statistics.std_normal = function()
|
statistics.std_normal = function()
|
||||||
|
|
||||||
local u = math.random()
|
local u = math.random()
|
||||||
|
|
||||||
if u < 0.001 then
|
if u < 0.001 then
|
||||||
return -3.0
|
return -3.0
|
||||||
elseif u > 0.999 then
|
elseif u > 0.999 then
|
||||||
return 3.0
|
return 3.0
|
||||||
end
|
end
|
||||||
|
|
||||||
return std_normal(u)
|
return std_normal(u)
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -121,14 +139,17 @@ end
|
|||||||
-- The distribution standard deviation.
|
-- The distribution standard deviation.
|
||||||
-- @return
|
-- @return
|
||||||
-- Any real number (actually between -3*sigma and 3*sigma).
|
-- Any real number (actually between -3*sigma and 3*sigma).
|
||||||
--
|
|
||||||
statistics.normal = function(mu, sigma)
|
statistics.normal = function(mu, sigma)
|
||||||
|
|
||||||
local u = math.random()
|
local u = math.random()
|
||||||
|
|
||||||
if u < 0.001 then
|
if u < 0.001 then
|
||||||
return mu - 3.0 * sigma
|
return mu - 3.0 * sigma
|
||||||
elseif u > 0.999 then
|
elseif u > 0.999 then
|
||||||
return mu + 3.0 * sigma
|
return mu + 3.0 * sigma
|
||||||
end
|
end
|
||||||
|
|
||||||
return mu + sigma * std_normal(u)
|
return mu + sigma * std_normal(u)
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -140,10 +161,13 @@ end
|
|||||||
-- The distribution maximum.
|
-- The distribution maximum.
|
||||||
-- @return
|
-- @return
|
||||||
-- An integer between 0 and max (both inclusive).
|
-- An integer between 0 and max (both inclusive).
|
||||||
--
|
|
||||||
statistics.poisson = function(lambda, max)
|
statistics.poisson = function(lambda, max)
|
||||||
|
|
||||||
lambda, max = tonumber(lambda), tonumber(max)
|
lambda, max = tonumber(lambda), tonumber(max)
|
||||||
|
|
||||||
if not lambda or not max or lambda <= 0 or max < 1 then return 0; end
|
if not lambda or not max or lambda <= 0 or max < 1 then return 0; end
|
||||||
|
|
||||||
return poisson(lambda, max)
|
return poisson(lambda, max)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user