mobf_core/mobf/harvesting.lua
sapier 44902f7e0a Fix not all mobs rotateable by punching
Fix crash due to inconsistent mob state
Fix mobs not correctly dealing damage do other mobs
2014-07-14 22:36:57 +02:00

251 lines
7.8 KiB
Lua

-------------------------------------------------------------------------------
-- Mob Framework Mod by Sapier
--
-- You may copy, use, modify or do nearly anything except removing this
-- copyright notice.
-- And of course you are NOT allow to pretend you have written it.
--
--! @file harvesting.lua
--! @brief component for all harvesting related mob features
--! @copyright Sapier
--! @author Sapier
--! @date 2012-08-09
--
--! @defgroup harvesting Harvesting subcomponent
--! @brief Component handling harvesting
--! @ingroup framework_int
--! @{
-- Contact sapier a t gmx net
-------------------------------------------------------------------------------
mobf_assert_backtrace(harvesting == nil)
--! @class harvesting
--! @brief harvesting features
harvesting = {}
--!@}
-------------------------------------------------------------------------------
-- @function [parent=#harvesting] init_dynamic_data(entity,now)
--
--! @brief initialize dynamic data required by harvesting
--! @memberof harvesting
--
--! @param entity mob to initialize harvest dynamic data
--! @param now current time
-------------------------------------------------------------------------------
function harvesting.init_dynamic_data(entity,now)
dbg_mobf.harvesting_lvl1("MOBF: " .. entity.data.name
.. " initializing harvesting dynamic data")
local data = {
ts_last = now,
}
entity.dynamic_data.harvesting = data
end
-------------------------------------------------------------------------------
-- @function [parent=#harvesting] callback(entity,player,now)
--
--! @brief callback handler for harvest by player
--! @memberof harvesting
--
--! @param entity mob being harvested
--! @param player player harvesting
--! @param now the current time
--! @return true/false if handled by harvesting or not
-------------------------------------------------------------------------------
function harvesting.callback(entity,player,now)
dbg_mobf.harvesting_lvl1("MOBF: harvest function called")
local now = mobf_get_current_time()
--handle catching of mob
if entity.data.catching ~= nil and
entity.data.catching.tool ~= "" then
if (entity.dynamic_data.spawning.player_spawned and
entity.dynamic_data.spawning.spawner == nil) then
dbg_mobf.harvesting_lvl1("MOBF: mob flagged as player spanwned but no spawner set!")
entity.dynamic_data.spawning.player_spawned = false
end
--grief protection
if minetest.world_setting_get("mobf_grief_protection") and
entity.dynamic_data.spawning.player_spawned and
entity.dynamic_data.spawning.spawner ~= player:get_player_name() then
dbg_mobf.harvesting_lvl1("MOBF: anti gief triggered catching aborted")
return true
end
-- what's wielded by player
local tool = player:get_wielded_item()
if tool:get_name() == entity.data.catching.tool then
dbg_mobf.harvesting_lvl1("MOBF: player wearing ".. entity.data.catching.tool)
--check if player has enough room
local inventory_add_result = nil
if entity.data.generic.addoncatch ~= nil then
inventory_add_result = player:get_inventory():add_item("main",
entity.data.generic.addoncatch.." 1")
dbg_mobf.harvesting_lvl2(
"MOBF: adding specified oncatch item: " ..
entity.data.generic.addoncatch)
else
inventory_add_result = player:get_inventory():add_item("main",
entity.data.modname ..":"..entity.data.name.." 1")
dbg_mobf.harvesting_lvl2(
"MOBF: adding automatic oncatch item: " ..
entity.data.modname ..":"..entity.data.name)
end
if not inventory_add_result:is_empty() then
minetest.chat_send_player(player:get_player_name(),
"You don't have any room left in inventory!")
return true
end
--play catch sound
if entity.data.sound ~= nil then
sound.play(entity.object:getpos(),entity.data.sound.catch);
end
if entity.data.catching.consumed == true then
if player:get_inventory():contains_item("main",entity.data.catching.tool.." 1") then
dbg_mobf.harvesting_lvl2("MOBF: removing: "
.. entity.data.catching.tool.." 1")
player:get_inventory():remove_item("main",
entity.data.catching.tool.." 1")
else
mobf_bug_warning(LOGLEVEL_ERROR,"MOBF: BUG!!! player is"
.. " wearing a item he doesn't have in inventory!!!")
end
end
spawning.remove(entity, "cought")
return true
end
end
--handle harvestable mobs, check if player is wearing correct tool
if entity.data.harvest ~= nil then
dbg_mobf.harvesting_lvl1("MOBF: trying to harvest harvestable mob")
if (entity.data.harvest.tool ~= "") then
local tool = player:get_wielded_item()
if tool ~= nil then
dbg_mobf.harvesting_lvl1("MOBF: Player is wearing >"
.. tool:get_name() .. "< required is >".. entity.data.harvest.tool
.. "< wear: " .. tool:get_wear())
if (tool:get_name() ~= entity.data.harvest.tool) then
--player is wearing wrong tool do an attack
return false
else
--tool is completely consumed
if entity.data.harvest.tool_consumed == true then
if player:get_inventory():contains_item("main",entity.data.harvest.tool.." 1") == false then
dbg_mobf.harvesting_lvl1("MOBF: Player doesn't have"
.. " at least 1 of ".. entity.data.harvest.tool)
--handled but not ok so don't attack
return true
end
else
--damage tool
local tool_wear = tool:get_wear()
dbg_mobf.harvesting_lvl1("MOBF: tool " .. tool:get_name()
.. " wear: " .. tool_wear)
-- damage used tool
if tool_wear ~= nil and
entity.data.harvest.max_tool_usage ~= nil then
local todamage = (65535/entity.data.harvest.max_tool_usage)
dbg_mobf.harvesting_lvl1("MOBF: tool damage calculated: "
.. todamage);
if tool:add_wear(todamage) ~= true then
dbg_mobf.harvesting_lvl3("MOBF: Tried to damage non tool item "
.. tool:get_name() .. "!");
end
player:set_wielded_item(tool)
end
end
end
else
--player isn't wearing a tool so this has to be an attack
return false
end
else
--no havest tool defined so this has to be an attack
return false
end
--transformation and harvest delay is exclusive
--harvest delay mode
if entity.data.harvest.min_delay < 0 or
entity.dynamic_data.harvesting.ts_last + entity.data.harvest.min_delay < now then
--TODO check if player has enough room
player:get_inventory():add_item("main", entity.data.harvest.result.." 1")
--check if tool is consumed by action
if entity.data.harvest.tool_consumed then
dbg_mobf.harvesting_lvl2("MOBF: removing "
..entity.data.harvest.tool.." 1")
player:get_inventory():remove_item("main",entity.data.harvest.tool.." 1")
end
else
dbg_mobf.harvesting_lvl1("MOBF: " .. entity.data.name
.. " not ready to be harvested")
end
-- check if mob is transformed by harvest
if entity.data.harvest.transforms_to ~= "" then
local transformed = spawning.replace_entity(entity,
entity.data.harvest.transforms_to)
else
entity.dynamic_data.harvesting.ts_last = mobf_get_current_time()
end
--play harvest sound
if entity.data.sound ~= nil then
sound.play(entity.object:getpos(),entity.data.sound.harvest);
end
--harvest done
return true
end
return false
end
-------------------------------------------------------------------------------
-- @function transform(entity)
--
--! @brief self transform callback for mob
--! @ingroup harvesting
--
--! @param entity mob calling
--! @param now current time
-------------------------------------------------------------------------------
function transform(entity,now)
--check if it's a transformable mob
if entity.data.auto_transform ~= nil then
if now - entity.dynamic_data.spawning.original_spawntime
> entity.data.auto_transform.delay then
spawning.replace_entity(entity,entity.data.auto_transform.result)
return false
end
end
end