2016-12-18 18:32:39 +01:00
|
|
|
-- Actions code for Advanced NPC by Zorman2000
|
|
|
|
---------------------------------------------------------------------------------------
|
|
|
|
-- Action functionality
|
|
|
|
---------------------------------------------------------------------------------------
|
2017-05-05 15:13:30 +02:00
|
|
|
-- The NPCs will be able to perform six fundamental actions that will allow
|
2016-12-18 18:32:39 +01:00
|
|
|
-- for them to perform any other kind of interaction in the world. These
|
|
|
|
-- fundamental actions are: place a node, dig a node, put items on an inventory,
|
|
|
|
-- take items from an inventory, find a node closeby (radius 3) and
|
|
|
|
-- walk a step on specific direction. These actions will be set on an action queue.
|
|
|
|
-- The queue will have the specific steps, in order, for the NPC to be able to do
|
|
|
|
-- something (example, go to a specific place and put a chest there). The
|
|
|
|
-- fundamental actions are added to the action queue to make a complete task for the NPC.
|
|
|
|
|
|
|
|
npc.actions = {}
|
|
|
|
|
2016-12-29 23:26:09 +01:00
|
|
|
-- Describes actions with doors or openable nodes
|
2017-01-06 13:57:42 +01:00
|
|
|
npc.actions.const = {
|
|
|
|
doors = {
|
|
|
|
action = {
|
|
|
|
OPEN = 1,
|
|
|
|
CLOSE = 2
|
|
|
|
},
|
|
|
|
state = {
|
|
|
|
OPEN = 1,
|
|
|
|
CLOSED = 2
|
|
|
|
}
|
|
|
|
},
|
|
|
|
beds = {
|
|
|
|
LAY = 1,
|
|
|
|
GET_UP = 2
|
|
|
|
},
|
|
|
|
sittable = {
|
|
|
|
SIT = 1,
|
|
|
|
GET_UP = 2
|
|
|
|
}
|
2016-12-29 23:26:09 +01:00
|
|
|
}
|
|
|
|
|
2017-05-05 15:13:30 +02:00
|
|
|
--npc.actions.one_nps_speed = 0.98
|
|
|
|
--npc.actions.one_half_nps_speed = 1.40
|
|
|
|
--npc.actions.two_nps_speed = 1.90'
|
|
|
|
npc.actions.one_nps_speed = 1
|
|
|
|
npc.actions.one_half_nps_speed = 1.5
|
|
|
|
npc.actions.two_nps_speed = 2
|
2017-01-16 18:16:42 +01:00
|
|
|
|
2017-01-13 13:12:13 +01:00
|
|
|
---------------------------------------------------------------------------------------
|
|
|
|
-- Actions
|
|
|
|
---------------------------------------------------------------------------------------
|
|
|
|
-- The following action alters the timer interval for executing actions, therefore
|
|
|
|
-- making waits and pauses possible, or increase timing when some actions want to
|
|
|
|
-- be performed faster, like walking.
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.set_interval(self, args)
|
|
|
|
local self_actions = args.self_actions
|
2017-01-13 13:12:13 +01:00
|
|
|
local new_interval = args.interval
|
|
|
|
local freeze_mobs_api = args.freeze
|
|
|
|
|
|
|
|
self.actions.action_interval = new_interval
|
|
|
|
return not freeze_mobs_api
|
|
|
|
end
|
|
|
|
|
|
|
|
-- The following action is for allowing the rest of mobs redo API to be executed
|
|
|
|
-- after this action ends. This is useful for times when no action is needed
|
|
|
|
-- and the NPC is allowed to roam freely.
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.freeze(self, args)
|
2017-01-13 13:12:13 +01:00
|
|
|
local freeze_mobs_api = args.freeze
|
|
|
|
minetest.log("Received: "..dump(freeze_mobs_api))
|
|
|
|
minetest.log("Returning: "..dump(not(freeze_mobs_api)))
|
|
|
|
return not(freeze_mobs_api)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- This action is to rotate to mob to a specifc direction. Currently, the code
|
|
|
|
-- contains also for diagonals, but remaining in the orthogonal domain is preferrable.
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.rotate(self, args)
|
2016-12-18 18:32:39 +01:00
|
|
|
local dir = args.dir
|
|
|
|
local yaw = 0
|
|
|
|
self.rotate = 0
|
|
|
|
if dir == npc.direction.north then
|
2016-12-29 04:10:06 +01:00
|
|
|
yaw = 0
|
2017-01-06 13:57:42 +01:00
|
|
|
elseif dir == npc.direction.north_east then
|
|
|
|
yaw = (7 * math.pi) / 4
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.east then
|
2016-12-29 04:10:06 +01:00
|
|
|
yaw = (3 * math.pi) / 2
|
2017-01-06 13:57:42 +01:00
|
|
|
elseif dir == npc.direction.south_east then
|
|
|
|
yaw = (5 * math.pi) / 4
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.south then
|
2016-12-29 04:10:06 +01:00
|
|
|
yaw = math.pi
|
2017-01-06 13:57:42 +01:00
|
|
|
elseif dir == npc.direction.south_west then
|
|
|
|
yaw = (3 * math.pi) / 4
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.west then
|
2016-12-29 04:10:06 +01:00
|
|
|
yaw = math.pi / 2
|
2017-01-06 13:57:42 +01:00
|
|
|
elseif dir == npc.direction.north_west then
|
|
|
|
yaw = math.pi / 4
|
2016-12-18 18:32:39 +01:00
|
|
|
end
|
|
|
|
self.object:setyaw(yaw)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- This function will make the NPC walk one step on a
|
|
|
|
-- specifc direction. One step means one node. It returns
|
|
|
|
-- true if it can move on that direction, and false if there is an obstacle
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.walk_step(self, args)
|
2016-12-18 18:32:39 +01:00
|
|
|
local dir = args.dir
|
2017-01-14 04:54:27 +01:00
|
|
|
local speed = args.speed
|
2017-05-05 15:13:30 +02:00
|
|
|
local target_pos = args.target_pos
|
2016-12-18 18:32:39 +01:00
|
|
|
local vel = {}
|
2017-01-14 04:54:27 +01:00
|
|
|
-- Set default node per seconds
|
|
|
|
if speed == nil then
|
2017-01-16 18:16:42 +01:00
|
|
|
speed = npc.actions.one_nps_speed
|
2017-01-14 04:54:27 +01:00
|
|
|
end
|
2017-05-05 15:13:30 +02:00
|
|
|
-- If there is a target position to reach, set it
|
|
|
|
if target_pos ~= nil then
|
|
|
|
self.actions.walking.target_pos = target_pos
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Set is_walking = true
|
|
|
|
self.actions.walking.is_walking = true
|
|
|
|
|
2016-12-18 18:32:39 +01:00
|
|
|
if dir == npc.direction.north then
|
2017-01-14 04:54:27 +01:00
|
|
|
vel = {x=0, y=0, z=speed}
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.east then
|
2017-01-14 04:54:27 +01:00
|
|
|
vel = {x=speed, y=0, z=0}
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.south then
|
2017-01-14 04:54:27 +01:00
|
|
|
vel = {x=0, y=0, z=-speed}
|
2016-12-18 18:32:39 +01:00
|
|
|
elseif dir == npc.direction.west then
|
2017-01-14 04:54:27 +01:00
|
|
|
vel = {x=-speed, y=0, z=0}
|
2016-12-18 18:32:39 +01:00
|
|
|
end
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Rotate NPC
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.actions.rotate(self, {dir=dir})
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Set velocity so that NPC walks
|
2016-12-18 18:32:39 +01:00
|
|
|
self.object:setvelocity(vel)
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Set walk animation
|
|
|
|
self.object:set_animation({
|
|
|
|
x = npc.ANIMATION_WALK_START,
|
|
|
|
y = npc.ANIMATION_WALK_END},
|
|
|
|
self.animation.speed_normal, 0)
|
2016-12-18 18:32:39 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- This action makes the NPC stand and remain like that
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.stand(self, args)
|
2017-01-09 17:34:38 +01:00
|
|
|
local pos = args.pos
|
|
|
|
local dir = args.dir
|
2017-05-05 15:13:30 +02:00
|
|
|
-- Set is_walking = true
|
|
|
|
self.actions.walking.is_walking = false
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Stop NPC
|
|
|
|
self.object:setvelocity({x=0, y=0, z=0})
|
2017-01-09 17:34:38 +01:00
|
|
|
-- If position given, set to that position
|
|
|
|
if pos ~= nil then
|
|
|
|
self.object:moveto(pos)
|
|
|
|
end
|
|
|
|
-- If dir given, set to that dir
|
|
|
|
if dir ~= nil then
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.actions.rotate(self, {dir=dir})
|
2017-01-09 17:34:38 +01:00
|
|
|
end
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Set stand animation
|
2017-01-09 17:34:38 +01:00
|
|
|
self.object:set_animation({
|
|
|
|
x = npc.ANIMATION_STAND_START,
|
|
|
|
y = npc.ANIMATION_STAND_END},
|
|
|
|
self.animation.speed_normal, 0)
|
2016-12-18 18:32:39 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- This action makes the NPC sit on the node where it is
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.sit(self, args)
|
2016-12-30 03:34:15 +01:00
|
|
|
local pos = args.pos
|
2017-01-09 17:34:38 +01:00
|
|
|
local dir = args.dir
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Stop NPC
|
|
|
|
self.object:setvelocity({x=0, y=0, z=0})
|
2017-01-09 17:34:38 +01:00
|
|
|
-- If position given, set to that position
|
2016-12-30 03:34:15 +01:00
|
|
|
if pos ~= nil then
|
2017-01-09 17:34:38 +01:00
|
|
|
self.object:moveto(pos)
|
|
|
|
end
|
|
|
|
-- If dir given, set to that dir
|
|
|
|
if dir ~= nil then
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.actions.rotate(self, {dir=dir})
|
2016-12-30 03:34:15 +01:00
|
|
|
end
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Set sit animation
|
|
|
|
self.object:set_animation({
|
|
|
|
x = npc.ANIMATION_SIT_START,
|
|
|
|
y = npc.ANIMATION_SIT_END},
|
|
|
|
self.animation.speed_normal, 0)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- This action makes the NPC lay on the node where it is
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.lay(self, args)
|
2016-12-30 03:34:15 +01:00
|
|
|
local pos = args.pos
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Stop NPC
|
|
|
|
self.object:setvelocity({x=0, y=0, z=0})
|
2016-12-30 03:34:15 +01:00
|
|
|
-- If position give, set to that position
|
|
|
|
if pos ~= nil then
|
2017-01-09 17:34:38 +01:00
|
|
|
self.object:moveto(pos)
|
2016-12-30 03:34:15 +01:00
|
|
|
end
|
2016-12-18 18:32:39 +01:00
|
|
|
-- Set sit animation
|
|
|
|
self.object:set_animation({
|
|
|
|
x = npc.ANIMATION_LAY_START,
|
|
|
|
y = npc.ANIMATION_LAY_END},
|
|
|
|
self.animation.speed_normal, 0)
|
|
|
|
end
|
2016-12-29 04:10:06 +01:00
|
|
|
|
|
|
|
-- Inventory functions for players and for nodes
|
|
|
|
-- This function is a convenience function to make it easy to put
|
|
|
|
-- and get items from another inventory (be it a player inv or
|
|
|
|
-- a node inv)
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.put_item_on_external_inventory(self, args)
|
2016-12-29 04:10:06 +01:00
|
|
|
local player = args.player
|
|
|
|
local pos = args.pos
|
|
|
|
local inv_list = args.inv_list
|
|
|
|
local item_name = args.item_name
|
|
|
|
local count = args.count
|
|
|
|
local is_furnace = args.is_furnace
|
|
|
|
local inv
|
|
|
|
if player ~= nil then
|
|
|
|
inv = minetest.get_inventory({type="player", name=player})
|
|
|
|
else
|
|
|
|
inv = minetest.get_inventory({type="node", pos=pos})
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Create ItemStack to put on external inventory
|
|
|
|
local item = ItemStack(item_name.." "..count)
|
|
|
|
-- Check if there is enough room to add the item on external invenotry
|
|
|
|
if inv:room_for_item(inv_list, item) then
|
|
|
|
-- Take item from NPC's inventory
|
|
|
|
if npc.take_item_from_inventory_itemstring(self, item) then
|
|
|
|
-- NPC doesn't have item and/or specified quantity
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
-- Add items to external inventory
|
|
|
|
inv:add_item(inv_list, item)
|
|
|
|
|
|
|
|
-- If this is a furnace, start furnace timer
|
|
|
|
if is_furnace == true then
|
|
|
|
minetest.get_node_timer(pos):start(1.0)
|
|
|
|
end
|
|
|
|
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
-- Not able to put on external inventory
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.take_item_from_external_inventory(self, args)
|
2016-12-29 04:10:06 +01:00
|
|
|
local player = args.player
|
|
|
|
local pos = args.pos
|
|
|
|
local inv_list = args.inv_list
|
|
|
|
local item_name = args.item_name
|
|
|
|
local count = args.count
|
|
|
|
local inv
|
|
|
|
if player ~= nil then
|
|
|
|
inv = minetest.get_inventory({type="player", name=player})
|
|
|
|
else
|
2017-01-14 04:54:27 +01:00
|
|
|
inv = minetest.get_inventory({type="node", pos=pos})
|
2016-12-29 04:10:06 +01:00
|
|
|
end
|
|
|
|
-- Create ItemSTack to take from external inventory
|
|
|
|
local item = ItemStack(item_name.." "..count)
|
|
|
|
-- Check if there is enough of the item to take
|
|
|
|
if inv:contains_item(inv_list, item) then
|
|
|
|
-- Add item to NPC's inventory
|
|
|
|
npc.add_item_to_inventory_itemstring(self, item)
|
|
|
|
-- Add items to external inventory
|
|
|
|
inv:remove_item(inv_list, item)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
-- Not able to put on external inventory
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.check_external_inventory_contains_item(self, args)
|
2017-01-06 13:57:42 +01:00
|
|
|
local player = args.player
|
|
|
|
local pos = args.pos
|
|
|
|
local inv_list = args.inv_list
|
|
|
|
local item_name = args.item_name
|
|
|
|
local count = args.count
|
|
|
|
local inv
|
|
|
|
if player ~= nil then
|
|
|
|
inv = minetest.get_inventory({type="player", name=player})
|
|
|
|
else
|
2017-01-14 04:54:27 +01:00
|
|
|
inv = minetest.get_inventory({type="node", pos=pos})
|
2017-01-06 13:57:42 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Create ItemStack for checking the external inventory
|
|
|
|
local item = ItemStack(item_name.." "..count)
|
|
|
|
-- Check if inventory contains item
|
|
|
|
return inv:contains_item(inv_list, item)
|
|
|
|
end
|
|
|
|
|
2017-01-09 17:34:38 +01:00
|
|
|
-- TODO: Refactor this function so that it uses a table to check
|
|
|
|
-- for doors instead of having separate logic for each door type
|
|
|
|
function npc.actions.get_openable_node_state(node, npc_dir)
|
2017-01-06 13:57:42 +01:00
|
|
|
minetest.log("Node name: "..dump(node.name))
|
|
|
|
local state = npc.actions.const.doors.state.CLOSED
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Check for default doors and gates
|
2016-12-30 01:02:25 +01:00
|
|
|
local a_i1, a_i2 = string.find(node.name, "_a")
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Check for cottages gates
|
2017-01-06 13:57:42 +01:00
|
|
|
local open_i1, open_i2 = string.find(node.name, "_close")
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Check for cottages half door
|
|
|
|
local half_door_is_closed = false
|
|
|
|
if node.name == "cottages:half_door" then
|
|
|
|
half_door_is_closed = (node.param2 + 2) % 4 == npc_dir
|
|
|
|
end
|
|
|
|
if a_i1 == nil and open_i1 == nil and not half_door_is_closed then
|
2017-01-06 13:57:42 +01:00
|
|
|
state = npc.actions.const.doors.state.OPEN
|
2016-12-30 01:02:25 +01:00
|
|
|
end
|
2017-01-06 13:57:42 +01:00
|
|
|
minetest.log("Door state: "..dump(state))
|
2016-12-30 01:02:25 +01:00
|
|
|
return state
|
|
|
|
end
|
|
|
|
|
2017-01-09 17:34:38 +01:00
|
|
|
-- This function is used to open or close openable nodes.
|
|
|
|
-- Currently supported openable nodes are: any doors using the
|
|
|
|
-- default doors API, and the cottages mod gates and doors.
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.use_door(self, args)
|
2016-12-29 23:26:09 +01:00
|
|
|
local pos = args.pos
|
|
|
|
local action = args.action
|
2017-01-09 17:34:38 +01:00
|
|
|
local dir = args.dir
|
2016-12-29 23:26:09 +01:00
|
|
|
local node = minetest.get_node(pos)
|
2017-01-09 17:34:38 +01:00
|
|
|
local state = npc.actions.get_openable_node_state(node, dir)
|
2016-12-30 01:02:25 +01:00
|
|
|
|
2016-12-29 23:26:09 +01:00
|
|
|
local clicker = self.object
|
|
|
|
if action ~= state then
|
|
|
|
minetest.registered_nodes[node.name].on_rightclick(pos, node, clicker, nil, nil)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-29 04:10:06 +01:00
|
|
|
|
|
|
|
---------------------------------------------------------------------------------------
|
|
|
|
-- Tasks functionality
|
|
|
|
---------------------------------------------------------------------------------------
|
|
|
|
-- Tasks are operations that require many actions to perform. Basic tasks, like
|
|
|
|
-- walking from one place to another, operating a furnace, storing or taking
|
2016-12-30 03:34:15 +01:00
|
|
|
-- items from a chest, are provided here.
|
2016-12-29 04:10:06 +01:00
|
|
|
|
|
|
|
-- This function allows a NPC to use a furnace using only items from
|
|
|
|
-- its own inventory. Fuel is not provided. Once the furnace is finished
|
|
|
|
-- with the fuel items the NPC will take whatever was cooked and whatever
|
|
|
|
-- remained to cook. The function received the position of the furnace
|
|
|
|
-- to use, and the item to cook in furnace. Item is an itemstring
|
2017-03-10 01:41:44 +01:00
|
|
|
function npc.actions.use_furnace(self, args)
|
|
|
|
local pos = args.pos
|
|
|
|
local item = args.item
|
|
|
|
local freeze = args.freeze
|
2017-01-14 04:54:27 +01:00
|
|
|
-- Define which items are usable as fuels. The NPC
|
|
|
|
-- will mainly use this as fuels to avoid getting useful
|
|
|
|
-- items (such as coal lumps) for burning
|
|
|
|
local fuels = {"default:leaves",
|
|
|
|
"default:pine_needles",
|
|
|
|
"default:tree",
|
|
|
|
"default:acacia_tree",
|
|
|
|
"default:aspen_tree",
|
|
|
|
"default:jungletree",
|
|
|
|
"default:pine_tree",
|
|
|
|
"default:coalblock",
|
|
|
|
"farming:straw"}
|
|
|
|
|
|
|
|
-- Check if NPC has item to cook
|
|
|
|
local src_item = npc.inventory_contains(self, npc.get_item_name(item))
|
|
|
|
if src_item == nil then
|
|
|
|
-- Unable to cook item that is not in inventory
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2016-12-29 04:10:06 +01:00
|
|
|
-- Check if NPC has a fuel item
|
2017-01-14 04:54:27 +01:00
|
|
|
for i = 1,9 do
|
2016-12-29 04:10:06 +01:00
|
|
|
local fuel_item = npc.inventory_contains(self, fuels[i])
|
2017-01-14 04:54:27 +01:00
|
|
|
|
|
|
|
if fuel_item ~= nil then
|
|
|
|
-- Get fuel item's burn time
|
|
|
|
local fuel_time =
|
|
|
|
minetest.get_craft_result({method="fuel", width=1, items={ItemStack(fuel_item.item_string)}}).time
|
|
|
|
local total_fuel_time = fuel_time * npc.get_item_count(fuel_item.item_string)
|
|
|
|
minetest.log("Fuel time: "..dump(fuel_time))
|
|
|
|
|
|
|
|
-- Get item to cook's cooking time
|
|
|
|
local cook_result =
|
|
|
|
minetest.get_craft_result({method="cooking", width=1, items={ItemStack(src_item.item_string)}})
|
|
|
|
local total_cook_time = cook_result.time * npc.get_item_count(item)
|
|
|
|
minetest.log("Cook: "..dump(cook_result))
|
|
|
|
|
|
|
|
minetest.log("Total cook time: "..total_cook_time
|
|
|
|
..", total fuel burn time: "..dump(total_fuel_time))
|
|
|
|
|
|
|
|
-- Check if there is enough fuel to cook all items
|
|
|
|
if total_cook_time > total_fuel_time then
|
|
|
|
-- Don't have enough fuel to cook item. Return the difference
|
|
|
|
-- so it may help on trying to acquire the fuel later.
|
|
|
|
-- NOTE: Yes, returning here means that NPC could probably have other
|
|
|
|
-- items usable as fuels and ignore them. This should be ok for now,
|
|
|
|
-- considering that fuel items are ordered in a way where cheaper, less
|
|
|
|
-- useless items come first, saving possible valuable items.
|
|
|
|
return cook_result.time - fuel_time
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Calculate how much fuel is needed
|
|
|
|
local fuel_amount = total_cook_time / fuel_time
|
|
|
|
if fuel_amount < 1 then
|
|
|
|
fuel_amount = 1
|
|
|
|
end
|
|
|
|
|
|
|
|
minetest.log("Amount of fuel needed: "..fuel_amount)
|
2016-12-29 04:10:06 +01:00
|
|
|
|
|
|
|
-- Put this item on the fuel inventory list of the furnace
|
|
|
|
local args = {
|
|
|
|
player = nil,
|
|
|
|
pos = pos,
|
|
|
|
inv_list = "fuel",
|
|
|
|
item_name = npc.get_item_name(fuel_item.item_string),
|
2017-01-14 04:54:27 +01:00
|
|
|
count = fuel_amount
|
2016-12-29 04:10:06 +01:00
|
|
|
}
|
|
|
|
npc.add_action(self, npc.actions.put_item_on_external_inventory, args)
|
|
|
|
-- Put the item that we want to cook on the furnace
|
|
|
|
args = {
|
|
|
|
player = nil,
|
|
|
|
pos = pos,
|
|
|
|
inv_list = "src",
|
|
|
|
item_name = npc.get_item_name(src_item.item_string),
|
2017-01-14 04:54:27 +01:00
|
|
|
count = npc.get_item_count(item),
|
2016-12-29 04:10:06 +01:00
|
|
|
is_furnace = true
|
|
|
|
}
|
|
|
|
npc.add_action(self, npc.actions.put_item_on_external_inventory, args)
|
|
|
|
|
2017-01-14 04:54:27 +01:00
|
|
|
-- Now, set NPC to wait until furnace is done.
|
|
|
|
minetest.log("Setting wait action for "..dump(total_cook_time))
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.set_interval, {interval=total_cook_time, freeze=freeze})
|
2017-01-14 04:54:27 +01:00
|
|
|
|
|
|
|
-- Reset timer
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.set_interval, {interval=1, freeze=true})
|
2017-01-06 13:57:42 +01:00
|
|
|
|
2017-01-14 04:54:27 +01:00
|
|
|
-- If freeze is false, then we will have to find the way back to the furnace
|
|
|
|
-- once cooking is done.
|
|
|
|
if freeze == false then
|
|
|
|
minetest.log("Adding walk to position to wandering: "..dump(pos))
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_task(self, npc.actions.walk_to_pos, {end_pos=pos, walkable={}})
|
2017-01-14 04:54:27 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Take cooked items back
|
|
|
|
args = {
|
2017-01-20 02:42:46 +01:00
|
|
|
|
2017-01-14 04:54:27 +01:00
|
|
|
player = nil,
|
|
|
|
pos = pos,
|
|
|
|
inv_list = "dst",
|
|
|
|
item_name = cook_result.item:get_name(),
|
|
|
|
count = npc.get_item_count(item),
|
|
|
|
is_furnace = false
|
|
|
|
}
|
2017-05-05 15:13:30 +02:00
|
|
|
minetest.log("Taking item back: "..minetest.pos_to_string(pos))
|
2017-01-14 04:54:27 +01:00
|
|
|
npc.add_action(self, npc.actions.take_item_from_external_inventory, args)
|
|
|
|
|
|
|
|
minetest.log("Inventory: "..dump(self.inventory))
|
2017-01-06 13:57:42 +01:00
|
|
|
|
2016-12-29 04:10:06 +01:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
-- Couldn't use the furnace due to lack of items
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2016-12-30 03:34:15 +01:00
|
|
|
-- This function makes the NPC lay or stand up from a bed. The
|
|
|
|
-- pos is the location of the bed, action can be lay or get up
|
|
|
|
function npc.actions.use_bed(self, pos, action)
|
2017-01-09 17:34:38 +01:00
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
minetest.log(dump(node))
|
|
|
|
local dir = minetest.facedir_to_dir(node.param2)
|
2016-12-30 03:34:15 +01:00
|
|
|
|
2017-01-06 13:57:42 +01:00
|
|
|
if action == npc.actions.const.beds.LAY then
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Get position
|
2017-01-10 20:21:45 +01:00
|
|
|
local bed_pos = npc.actions.nodes.beds[node.name].get_lay_pos(pos, dir)
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Sit down on bed, rotate to correct direction
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.sit, {pos=bed_pos, dir=(node.param2 + 2) % 4})
|
2016-12-30 03:34:15 +01:00
|
|
|
-- Lay down
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.lay, {})
|
2016-12-30 03:34:15 +01:00
|
|
|
else
|
|
|
|
-- Calculate position to get up
|
2017-01-10 20:21:45 +01:00
|
|
|
local bed_pos_y = npc.actions.nodes.beds[node.name].get_lay_pos(pos, dir).y
|
|
|
|
local bed_pos = {x = pos.x, y = bed_pos_y, z = pos.z}
|
2016-12-30 03:34:15 +01:00
|
|
|
-- Sit up
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.sit, {pos=bed_pos})
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Initialize direction: Default is front of bottom of bed
|
|
|
|
local dir = (node.param2 + 2) % 4
|
|
|
|
-- Find empty node around node
|
2017-01-10 20:21:45 +01:00
|
|
|
-- Take into account that mats are close to the floor, so y adjustmen is zero
|
|
|
|
local y_adjustment = -1
|
|
|
|
if npc.actions.nodes.beds[node.name].type == "mat" then
|
|
|
|
y_adjustment = 0
|
|
|
|
end
|
|
|
|
local empty_nodes = npc.places.find_node_orthogonally(bed_pos, {"air", "cottages:bench"}, y_adjustment)
|
2017-01-09 17:34:38 +01:00
|
|
|
if empty_nodes ~= nil then
|
|
|
|
-- Get direction to the empty node
|
|
|
|
dir = npc.actions.get_direction(bed_pos, empty_nodes[1].pos)
|
|
|
|
end
|
|
|
|
-- Calculate position to get out of bed
|
|
|
|
local pos_out_of_bed =
|
|
|
|
{x=empty_nodes[1].pos.x, y=empty_nodes[1].pos.y + 1, z=empty_nodes[1].pos.z}
|
|
|
|
-- Account for benches if they are present to avoid standing over them
|
|
|
|
if empty_nodes[1].name == "cottages:bench" then
|
|
|
|
pos_out_of_bed = {x=empty_nodes[1].pos.x, y=empty_nodes[1].pos.y + 1, z=empty_nodes[1].pos.z}
|
|
|
|
if empty_nodes[1].param2 == 0 then
|
|
|
|
pos_out_of_bed.z = pos_out_of_bed.z - 0.3
|
|
|
|
elseif empty_nodes[1].param2 == 1 then
|
|
|
|
pos_out_of_bed.x = pos_out_of_bed.x - 0.3
|
|
|
|
elseif empty_nodes[1].param2 == 2 then
|
|
|
|
pos_out_of_bed.z = pos_out_of_bed.z + 0.3
|
|
|
|
elseif empty_nodes[1].param2 == 3 then
|
|
|
|
pos_out_of_bed.x = pos_out_of_bed.x + 0.3
|
|
|
|
end
|
|
|
|
end
|
|
|
|
-- Stand out of bed
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.stand, {pos=pos_out_of_bed, dir=dir})
|
2016-12-30 03:34:15 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-01-06 13:57:42 +01:00
|
|
|
-- This function makes the NPC lay or stand up from a bed. The
|
|
|
|
-- pos is the location of the bed, action can be lay or get up
|
2017-03-10 01:41:44 +01:00
|
|
|
function npc.actions.use_sittable(self, args)
|
|
|
|
local pos = args.pos
|
|
|
|
local action = args.action
|
2017-01-06 13:57:42 +01:00
|
|
|
local node = minetest.get_node(pos)
|
|
|
|
|
|
|
|
if action == npc.actions.const.sittable.SIT then
|
2017-05-05 15:13:30 +02:00
|
|
|
-- minetest.log("On sittable: Node: "..dump(node))
|
|
|
|
-- minetest.log("On sittable: Sittable stuff: "..dump(npc.actions.nodes.sittable[node.name]))
|
|
|
|
-- minetest.log("On sittable: HAHA: "..dump(node))
|
2017-01-06 13:57:42 +01:00
|
|
|
-- Calculate position depending on bench
|
2017-01-11 13:54:50 +01:00
|
|
|
local sit_pos = npc.actions.nodes.sittable[node.name].get_sit_pos(pos, node.param2)
|
2017-01-06 13:57:42 +01:00
|
|
|
-- Sit down on bench/chair/stairs
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.sit, {pos=sit_pos, dir=(node.param2 + 2) % 4})
|
2017-01-06 13:57:42 +01:00
|
|
|
else
|
2017-01-11 13:54:50 +01:00
|
|
|
-- Find empty areas around chair
|
|
|
|
local dir = node.param2 + 2 % 4
|
|
|
|
local empty_nodes = npc.places.find_node_orthogonally(pos, {"air"}, 0)
|
|
|
|
if empty_nodes ~= nil then
|
|
|
|
-- Get direction to the empty node
|
|
|
|
dir = npc.actions.get_direction(pos, empty_nodes[1].pos)
|
|
|
|
end
|
|
|
|
-- Calculate position to get out of bed
|
|
|
|
local pos_out_of_sittable =
|
|
|
|
{x=empty_nodes[1].pos.x, y=empty_nodes[1].pos.y + 1, z=empty_nodes[1].pos.z}
|
2017-01-06 13:57:42 +01:00
|
|
|
-- Stand
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.stand, {pos=pos_out_of_sittable, dir=dir})
|
2017-01-06 13:57:42 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- This function returns the direction enum
|
|
|
|
-- for the moving from v1 to v2
|
|
|
|
function npc.actions.get_direction(v1, v2)
|
|
|
|
local dir = vector.subtract(v2, v1)
|
|
|
|
if dir.x ~= 0 then
|
|
|
|
if dir.x > 0 then
|
|
|
|
return npc.direction.east
|
|
|
|
else
|
|
|
|
return npc.direction.west
|
|
|
|
end
|
|
|
|
elseif dir.z ~= 0 then
|
|
|
|
if dir.z > 0 then
|
|
|
|
return npc.direction.north
|
|
|
|
else
|
|
|
|
return npc.direction.south
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-12-29 04:10:06 +01:00
|
|
|
|
2016-12-30 03:34:15 +01:00
|
|
|
-- This function can be used to make the NPC walk from one
|
2017-01-09 17:34:38 +01:00
|
|
|
-- position to another. If the optional parameter walkable_nodes
|
|
|
|
-- is included, which is a table of node names, these nodes are
|
|
|
|
-- going to be considered walkable for the algorithm to find a
|
|
|
|
-- path.
|
2017-01-20 02:42:46 +01:00
|
|
|
function npc.actions.walk_to_pos(self, args)
|
2017-01-16 18:16:42 +01:00
|
|
|
-- Get arguments for this task
|
2017-01-14 04:54:27 +01:00
|
|
|
local end_pos = args.end_pos
|
|
|
|
local walkable_nodes = args.walkable
|
2016-12-29 04:10:06 +01:00
|
|
|
|
2017-01-14 04:54:27 +01:00
|
|
|
-- Round start_pos to make sure it can find start and end
|
|
|
|
local start_pos = vector.round(self.object:getpos())
|
|
|
|
-- Use y of end_pos (this can only be done assuming flat terrain)
|
|
|
|
start_pos.y = self.object:getpos().y
|
|
|
|
minetest.log("Walk to pos: Using start position: "..dump(start_pos))
|
2016-12-29 04:10:06 +01:00
|
|
|
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Set walkable nodes to empty if the parameter hasn't been used
|
|
|
|
if walkable_nodes == nil then
|
|
|
|
walkable_nodes = {}
|
|
|
|
end
|
|
|
|
|
2016-12-29 21:09:30 +01:00
|
|
|
-- Find path
|
2017-01-09 17:34:38 +01:00
|
|
|
local path = pathfinder.find_path(start_pos, end_pos, 20, walkable_nodes)
|
2016-12-29 04:10:06 +01:00
|
|
|
|
|
|
|
if path ~= nil then
|
2017-05-05 15:13:30 +02:00
|
|
|
minetest.log("[advanced_npc] Found path to node: "..minetest.pos_to_string(end_pos))
|
|
|
|
-- Store path
|
|
|
|
self.actions.walking.path = path
|
2016-12-29 21:09:30 +01:00
|
|
|
|
2017-01-16 18:16:42 +01:00
|
|
|
-- Local variables
|
2017-01-06 13:57:42 +01:00
|
|
|
local door_opened = false
|
2017-01-16 18:16:42 +01:00
|
|
|
local speed = npc.actions.two_nps_speed
|
|
|
|
|
|
|
|
-- Set the action timer interval to half second. This is to account for
|
|
|
|
-- the increased speed when walking.
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.set_interval, {interval=0.5, freeze=true})
|
2017-01-06 13:57:42 +01:00
|
|
|
|
2017-05-05 15:13:30 +02:00
|
|
|
-- Set the initial last and target positions
|
|
|
|
self.actions.walking.target_pos = path[1].pos
|
|
|
|
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Add steps to path
|
2016-12-29 04:10:06 +01:00
|
|
|
for i = 1, #path do
|
2017-01-09 17:34:38 +01:00
|
|
|
-- Do not add an extra step if reached the goal node
|
2017-01-06 13:57:42 +01:00
|
|
|
if (i+1) == #path then
|
2016-12-29 21:09:30 +01:00
|
|
|
-- Add direction to last node
|
|
|
|
local dir = npc.actions.get_direction(path[i].pos, end_pos)
|
|
|
|
-- Add stand animation at end
|
2017-05-05 15:13:30 +02:00
|
|
|
npc.add_action(self, npc.actions.stand, {dir = dir})
|
2016-12-29 21:09:30 +01:00
|
|
|
break
|
|
|
|
end
|
|
|
|
-- Get direction to move from path[i] to path[i+1]
|
2016-12-29 04:10:06 +01:00
|
|
|
local dir = npc.actions.get_direction(path[i].pos, path[i+1].pos)
|
2016-12-30 01:02:25 +01:00
|
|
|
-- Check if next node is a door, if it is, open it, then walk
|
2017-01-06 13:57:42 +01:00
|
|
|
if path[i+1].type == pathfinder.node_types.openable then
|
2016-12-30 01:02:25 +01:00
|
|
|
-- Check if door is already open
|
|
|
|
local node = minetest.get_node(path[i+1].pos)
|
2017-01-09 17:34:38 +01:00
|
|
|
if npc.actions.get_openable_node_state(node, dir) == npc.actions.const.doors.state.CLOSED then
|
2017-01-06 13:57:42 +01:00
|
|
|
minetest.log("Opening action to open door")
|
2016-12-30 01:02:25 +01:00
|
|
|
-- Stop to open door, this avoids misplaced movements later on
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.stand, {dir=dir})
|
2016-12-30 01:02:25 +01:00
|
|
|
-- Open door
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.use_door, {pos=path[i+1].pos, dir=dir, action=npc.actions.const.doors.action.OPEN})
|
2017-01-06 13:57:42 +01:00
|
|
|
|
|
|
|
door_opened = true
|
2016-12-30 01:02:25 +01:00
|
|
|
end
|
|
|
|
end
|
2017-05-05 15:13:30 +02:00
|
|
|
|
2016-12-29 04:10:06 +01:00
|
|
|
-- Add walk action to action queue
|
2017-05-05 15:13:30 +02:00
|
|
|
npc.add_action(self, npc.actions.walk_step, {dir = dir, speed = speed, target_pos = path[i+1].pos})
|
2017-01-06 13:57:42 +01:00
|
|
|
|
|
|
|
if door_opened then
|
|
|
|
-- Stop to close door, this avoids misplaced movements later on
|
2017-01-14 04:54:27 +01:00
|
|
|
local x_adj, z_adj = 0, 0
|
|
|
|
if dir == 0 then
|
|
|
|
z_adj = 0.1
|
|
|
|
elseif dir == 1 then
|
|
|
|
x_adj = 0.1
|
|
|
|
elseif dir == 2 then
|
|
|
|
z_adj = -0.1
|
|
|
|
elseif dir == 3 then
|
|
|
|
x_adj = -0.1
|
|
|
|
end
|
|
|
|
local pos_on_close = {x=path[i+1].pos.x + x_adj, y=path[i+1].pos.y + 1, z=path[i+1].pos.z + z_adj}
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.stand, {dir=(dir + 2)% 4, pos=pos_on_close})
|
2017-01-06 13:57:42 +01:00
|
|
|
-- Close door
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.use_door, {pos=path[i+1].pos, action=npc.actions.const.doors.action.CLOSE})
|
2017-01-06 13:57:42 +01:00
|
|
|
|
|
|
|
door_opened = false
|
|
|
|
end
|
|
|
|
|
2016-12-29 04:10:06 +01:00
|
|
|
end
|
2017-01-16 18:16:42 +01:00
|
|
|
|
|
|
|
-- Return the action interval to default interval of 1 second
|
|
|
|
-- By default, always freeze.
|
2017-01-20 02:42:46 +01:00
|
|
|
npc.add_action(self, npc.actions.set_interval, {interval=1, freeze=true})
|
2017-01-16 18:16:42 +01:00
|
|
|
|
2017-01-06 13:57:42 +01:00
|
|
|
else
|
|
|
|
minetest.log("Unable to find path.")
|
2016-12-29 04:10:06 +01:00
|
|
|
end
|
2017-01-16 18:16:42 +01:00
|
|
|
end
|