8acddab74f
When applying bonemeal to eg. farm crops, these have a chance to grow in response to the application of bone meal. When a node can be bonemealed, the applied bone meal item should always be spent after using it, regardless of the results. Currently this does not work correctly, if the result of bonemealing has no effect on the node, the used bone meal item is not spent. This commit fixes the behavior of the bone meal item to always be taken when used on a node that defines a `_mcl_on_bonemealing()` callback. The nodes that implement the callback imay use the handler's return value only to signal if the bonemealing was succesful, not to signal if it was at all possible. For this reason, some nodes need to be made more strictly conforming to the API. * Always take the used bone meal item (if user is not in creative mode), regardless of whether the bonemealed node's handler returned `true`. * Make dispensers spawn particles after succesful bonemealing. * Trivial comment fix. * Ripe cocoa pod cannot be bonemealed. * Update API.md to describe the stricter API semantics.
161 lines
5.3 KiB
Lua
161 lines
5.3 KiB
Lua
local S = minetest.get_translator(minetest.get_current_modname())
|
|
|
|
local longdesc = S(
|
|
"Bone meal is a white dye and also useful as a fertilizer to " ..
|
|
"speed up the growth of many plants."
|
|
)
|
|
local usagehelp = S(
|
|
"Rightclick a sheep to turn its wool white. Rightclick a plant " ..
|
|
"to speed up its growth. Note that not all plants can be " ..
|
|
"fertilized like this. When you rightclick a grass block, tall " ..
|
|
"grass and flowers will grow all over the place."
|
|
)
|
|
|
|
mcl_bone_meal = {}
|
|
|
|
-- Bone meal particle api:
|
|
|
|
--- Spawns bone meal particles.
|
|
-- pos: where the particles spawn
|
|
-- def: particle spawner parameters, see minetest.add_particlespawner() for
|
|
-- details on these parameters.
|
|
--
|
|
function mcl_bone_meal.add_bone_meal_particle(pos, def)
|
|
if not def then
|
|
def = {}
|
|
end
|
|
minetest.add_particlespawner({
|
|
amount = def.amount or 10,
|
|
time = def.time or 0.1,
|
|
minpos = def.minpos or vector.subtract(pos, 0.5),
|
|
maxpos = def.maxpos or vector.add(pos, 0.5),
|
|
minvel = def.minvel or vector.new(-0.01, 0.01, -0.01),
|
|
maxvel = def.maxvel or vector.new(0.01, 0.01, 0.01),
|
|
minacc = def.minacc or vector.new(0, 0, 0),
|
|
maxacc = def.maxacc or vector.new(0, 0, 0),
|
|
minexptime = def.minexptime or 1,
|
|
maxexptime = def.maxexptime or 4,
|
|
minsize = def.minsize or 0.7,
|
|
maxsize = def.maxsize or 2.4,
|
|
texture = "mcl_particles_bonemeal.png^[colorize:#00EE00:125", -- TODO: real MC color
|
|
glow = def.glow or 1,
|
|
})
|
|
end
|
|
|
|
-- Begin legacy bone meal API.
|
|
--
|
|
-- Compatibility code for legacy users of the old bone meal API.
|
|
-- This code will be removed at some time in the future.
|
|
--
|
|
mcl_bone_meal.bone_meal_callbacks = {}
|
|
|
|
-- Shims for the old API are still available in mcl_dye and defer to
|
|
-- the real functions in mcl_bone_meal.
|
|
--
|
|
function mcl_bone_meal.register_on_bone_meal_apply(func)
|
|
minetest.log("warning", "register_on_bone_meal_apply(func) is deprecated. Read mcl_bone_meal/API.md!")
|
|
table.insert(mcl_bone_meal.bone_meal_callbacks, func)
|
|
end
|
|
|
|
-- Legacy registered users of the old API are handled through this function.
|
|
--
|
|
local function apply_bone_meal(pointed_thing, placer)
|
|
for _, func in pairs(mcl_bone_meal.bone_meal_callbacks) do
|
|
if func(pointed_thing, placer) then
|
|
return true
|
|
end
|
|
end
|
|
|
|
local pos = pointed_thing.under
|
|
local n = minetest.get_node(pos)
|
|
if n.name == "" then return false end
|
|
|
|
-- Wheat, Potato, Carrot, Pumpkin Stem, Melon Stem: Advance by 2-5 stages
|
|
if string.find(n.name, "mcl_farming:sweet_berry_bush_") then
|
|
mcl_dye.add_bone_meal_particle(pos)
|
|
if n.name == "mcl_farming:sweet_berry_bush_3" then
|
|
return minetest.add_item(vector.offset(pos,math.random()-0.5,math.random()-0.5,math.random()-0.5),"mcl_farming:sweet_berry")
|
|
else
|
|
return mcl_farming:grow_plant("plant_sweet_berry_bush", pos, n, 0, true)
|
|
end
|
|
return true
|
|
--[[
|
|
Here for when Bonemeal becomes an api, there's code if needed for handling applying to bamboo.
|
|
-- Handle applying bonemeal to bamboo.
|
|
elseif mcl_bamboo.is_bamboo(n.name) then
|
|
local success = mcl_bamboo.grow_bamboo(pos, true)
|
|
if success then
|
|
mcl_dye.add_bone_meal_particle(pos)
|
|
end
|
|
return success
|
|
--]]
|
|
end
|
|
|
|
return false
|
|
end
|
|
-- End legacy bone meal API
|
|
|
|
minetest.register_craftitem("mcl_bone_meal:bone_meal", {
|
|
inventory_image = "mcl_bone_meal.png",
|
|
description = S("Bone Meal"),
|
|
_tt_help = S("Speeds up plant growth"),
|
|
_doc_items_longdesc = longdesc,
|
|
_doc_items_usagehelp = usagehelp,
|
|
stack_max = 64,
|
|
groups = {craftitem=1, dye=1, basecolor_white=1, excolor_white=1, unicolor_white=1},
|
|
on_place = function(itemstack, placer, pointed_thing)
|
|
local pos = pointed_thing.under
|
|
local node = minetest.get_node(pos)
|
|
local ndef = minetest.registered_nodes[node.name]
|
|
-- Use pointed node's on_rightclick function first, if present.
|
|
if placer and not placer:get_player_control().sneak then
|
|
if ndef and ndef.on_rightclick then
|
|
return ndef.on_rightclick(pos, node, placer, itemstack, pointed_thing) or itemstack
|
|
end
|
|
end
|
|
-- If the pointed node can be bonemealed, let it handle the processing.
|
|
if ndef and ndef._mcl_on_bonemealing then
|
|
if ndef._mcl_on_bonemealing(pointed_thing, placer) then
|
|
mcl_bone_meal.add_bone_meal_particle(pos)
|
|
end
|
|
if not minetest.is_creative_enabled(placer:get_player_name()) then
|
|
itemstack:take_item()
|
|
end
|
|
-- Otherwise try the legacy API.
|
|
elseif apply_bone_meal(pointed_thing, placer) and
|
|
not minetest.is_creative_enabled(placer:get_player_name()) then
|
|
itemstack:take_item()
|
|
end
|
|
return itemstack
|
|
end,
|
|
_on_dispense = function(itemstack, pos, droppos, dropnode, dropdir)
|
|
local pointed_thing
|
|
if dropnode.name == "air" then
|
|
pointed_thing = {above = droppos, under = vector.offset(droppos, 0, -1 ,0)}
|
|
else
|
|
pointed_thing = {above = pos, under = droppos}
|
|
end
|
|
local node = minetest.get_node(pointed_thing.under)
|
|
local ndef = minetest.registered_nodes[node.name]
|
|
-- If the pointed node can be bonemealed, let it handle the processing.
|
|
if ndef and ndef._mcl_on_bonemealing then
|
|
if ndef._mcl_on_bonemealing(pointed_thing, nil) then
|
|
mcl_bone_meal.add_bone_meal_particle(pos)
|
|
end
|
|
itemstack:take_item()
|
|
else
|
|
-- Otherwise try the legacy API.
|
|
if apply_bone_meal(pointed_thing, nil) then
|
|
itemstack:take_item()
|
|
end
|
|
end
|
|
return itemstack
|
|
end,
|
|
_dispense_into_walkable = true
|
|
})
|
|
|
|
minetest.register_craft({
|
|
output = "mcl_bone_meal:bone_meal 3",
|
|
recipe = {{"mcl_mobitems:bone"}},
|
|
})
|