minetest-mm/mods/mobs_npc/functions.lua
2024-12-19 12:55:40 +01:00

436 lines
10 KiB
Lua

-- translation and mod check
local S = minetest.get_translator("mobs_npc")
local mcl = minetest.get_modpath("mcl_core") ~= nil
local def = minetest.get_modpath("default") ~= nil
-- show random message from list
mobs_npc.npc_talk = function(self, player, message_list)
local name = player:get_player_name() or ""
local messages = message_list or self.messages or {"??"}
if messages then
local msg = messages[math.random(#messages)]
minetest.chat_send_player(name, "<" .. (self.nametag or "") .. "> " .. msg)
end
end
-- drop random item from list
mobs_npc.drop_trade = function(self, player, item, item_list)
local w_inv = player:get_wielded_item()
if item ~= w_inv:get_name() then return end
local name = player:get_player_name()
if not mobs.is_creative(name) then
w_inv:take_item()
player:set_wielded_item(w_inv)
end
local pos = self.object:get_pos()
local drops = item_list
local drop = drops[math.random(#drops)]
local chance = 1
if type(drop) == "table" then
chance = drop[2] or 1
drop = drop[1]
end
if not minetest.registered_items[drop]
or math.random(chance) > 1 then
drop = mcl and "mcl_core:clay_lump" or "default:clay_lump"
end
local obj = minetest.add_item(pos, {name = drop})
local dir = player:get_look_dir()
obj:set_velocity({x = -dir.x, y = 1.5, z = -dir.z})
return true
end
-- check for simple_dialogs mod and setup
local context = {}
mobs_npc.useDialogs = "N"
minetest.register_on_leaveplayer(function(player)
context[player:get_player_name()] = nil
end)
if minetest.get_modpath("simple_dialogs") then
mobs_npc.useDialogs = "Y"
simple_dialogs.register_varloader(function(npcself, playername)
simple_dialogs.save_dialog_var(npcself, "NPCNAME", npcself.nametag, playername)
simple_dialogs.save_dialog_var(npcself, "STATE", npcself.state, playername)
simple_dialogs.save_dialog_var(npcself, "FOOD", npcself.food, playername)
simple_dialogs.save_dialog_var(npcself, "HEALTH", npcself.health, playername)
simple_dialogs.save_dialog_var(npcself, "owner", npcself.owner, playername)
end)
simple_dialogs.register_hook(function(npcself, playername,hook)
if hook.func == "TELEPORT" then
if npcself.owner then
--check to see if the player has 'bring' teleport privliges
local player_privs = minetest.get_player_privs(npcself.owner)
if player_privs["bring"] then
--validate x,y,z coords
if hook.parm and hook.parmcount and hook.parmcount > 2 then
local pos = {
x = tonumber(hook.parm[1]),
y = tonumber(hook.parm[2]),
z = tonumber(hook.parm[3])
}
if pos.x and pos.y and pos.z
and pos.x > -31500 and pos.x < 31500
and pos.y > -31500 and pos.y < 31500
and pos.z > -31500 and pos.z < 31500 then
local player = minetest.get_player_by_name(playername)
if player then
player:set_pos(pos) end
end
end
end
end
return "EXIT"
end
end)
end
-- Kilarin's formspec functions
function mobs_npc.get_controls_formspec(name, self)
self.id = set_npc_id(self) -- make sure id is set
local currentordermode = self.order
local npcId = self.id
local orderArray = {"wander", "stand", "follow"}
local currentorderidx = 1
for i = 1, 3 do --this seems like a clumsy way to do this
if orderArray[i] == currentordermode then
currentorderidx = i
break
end
end
-- Make npc controls formspec
local text = "NPC Controls"
local size = mobs_npc.useDialogs == "Y" and "size[15,10]" or "size[3.85,2.8]"
local formspec = {
size,
"label[0.375,0.5;", minetest.formspec_escape(text), "]",
"dropdown[0.375,1.25; 3,0.6;ordermode;wander,stand,follow;", currentorderidx, "]",
"button[0.375,2;3,0.8;exit;Exit]"
}
if mobs_npc.useDialogs == "Y" then
simple_dialogs.add_dialog_control_to_formspec(name, self, formspec, 0.375, 3.4)
end
table.concat(formspec, "")
--store npc id in local context so we can use it when the form is returned
context[name] = npcId
return table.concat(formspec, "")
end
-- receive and do orders given through form
minetest.register_on_player_receive_fields(function(player, formname, fields)
local pname = player:get_player_name()
if formname ~= "mobs_npc:controls" then
if context[pname] then context[pname] = nil end
return
end
local npcId = context[pname] or nil --get the npc id from local context
local npcself = get_npcself_from_id(npcId)
if npcself ~= nil then
if fields["exit"] then
minetest.close_formspec(pname, "mobs_npc:controls")
elseif fields["ordermode"] then
local pname = player:get_player_name()
npcself.order = fields["ordermode"]
if npcself.order == "wander" then
-- minetest.chat_send_player(pname, S("NPC will wander."))
elseif npcself.order == "stand" then
npcself.state = "stand"
npcself.attack = nil
npcself:set_animation("stand")
npcself:set_velocity(0)
-- minetest.chat_send_player(pname, S("NPC stands still."))
elseif npcself.order == "follow" then
-- minetest.chat_send_player(pname, S("NPC will follow you."))
end
end
if mobs_npc.useDialogs == "Y" then
simple_dialogs.process_simple_dialog_control_fields(pname, npcself, fields)
end
end
end)
-- check if npc has id set otherwise create one
function set_npc_id(npcself)
if not npcself.id then
npcself.id = (math.random(1, 1000) * math.random(1, 10000))
.. npcself.name .. (math.random(1, 1000) ^ 2)
end
return npcself.id
end
--this function finds an npcself in the luaentities list given an npcId
function get_npcself_from_id(npcId)
if npcId == nil then return nil end
for k, v in pairs(minetest.luaentities) do
if v.object and v.id and v.id == npcId then
return v
end
end
end
--This code comes almost exclusively from the trader and inventory of mobf, by Sapier.
--The copyright notice below is from mobf:
-------------------------------------------------------------------------------
-- 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 inventory.lua
--! @brief component containing mob inventory related functions
--! @copyright Sapier
--! @author Sapier
--! @date 2013-01-02
--
--! @defgroup Inventory Inventory subcomponent
--! @brief Component handling mob inventory
--! @ingroup framework_int
--! @{
--
-- Contact sapier a t gmx net
-------------------------------------------------------------------------------
-- This code has been heavily modified by isaiah658.
-- Trades are saved in entity metadata so they always stay the same after
-- initially being chosen. Also the formspec uses item image buttons instead of
-- inventory slots.
local function add_goods(self, race)
local trade_index = 1
local trades_already_added = {}
local trader_pool_size = 10
local item_pool_size = #race.items -- get number of items on list
self.trades = {}
if item_pool_size < trader_pool_size then
trader_pool_size = item_pool_size
end
for i = 1, trader_pool_size do
-- If there are more trades than the amount being added, they are
-- randomly selected. If they are equal, there is no reason to randomly
-- select them
local random_trade = nil
if item_pool_size == trader_pool_size then
random_trade = i
else
while random_trade == nil do
local num = math.random(item_pool_size)
if trades_already_added[num] == nil then
trades_already_added[num] = true
random_trade = num
end
end
end
if math.random(0, 100) > race.items[random_trade][3] then
self.trades[trade_index] = {
race.items[random_trade][1],
race.items[random_trade][2]
}
trade_index = trade_index + 1
end
end
end
function mobs_npc.shop_trade(self, clicker, race)
self.id = set_npc_id(self) -- make sure id is set
if not self.game_name then
self.game_name = tostring(race.names[math.random(1, #race.names)])
self.nametag = S("Trader @1", self.game_name)
self.object:set_properties({
nametag = self.nametag,
nametag_color = "#00FF00"
})
end
if self.trades == nil then
add_goods(self, race)
end
local player = clicker:get_player_name() or ""
minetest.chat_send_player(player,
S("[NPC] <Trader @1> Hello, @2, have a look at my wares.",
self.game_name, player))
-- Make formspec trade list
local formspec_trade_list = ""
local x, y
for i = 1, 10 do
if self.trades[i] and self.trades[i] ~= "" then
if i < 6 then
x = mcl and 1 or 0.5
y = i - 0.5
else
x = mcl and 5 or 4.5
y = i - 5.5
end
formspec_trade_list = formspec_trade_list
.. "item_image_button[".. x ..",".. y ..";1,1;"
.. self.trades[i][2] .. ";prices#".. i .."#".. self.id ..";]"
.. "item_image_button[".. x + 2 ..",".. y ..";1,1;"
.. self.trades[i][1] .. ";goods#".. i .."#".. self.id ..";]"
.. "image[".. x + 1 ..",".. y ..";1,1;gui_arrow_blank.png]"
end
end
local bg = mcl and mcl_vars.gui_bg_img or default.gui_bg_img or ""
local sl = mcl and mcl_vars.gui_slots or default.gui_slots or ""
local lc = mcl and "listcolors[#9d9d9d;#FFF7;#474747]" or ""
minetest.show_formspec(player, "mobs_npc:trade", "size[" .. (mcl and 9 or 8) .. ",10]"
.. bg
.. sl
.. "label[0.5,-0.1;" .. S("Trader @1's stock:", self.game_name) .. "]"
.. formspec_trade_list
.. lc
.. "list[current_player;main;0,6;" .. (mcl and 9 or 8) .. ",4;]"
)
end
minetest.register_on_player_receive_fields(function(player, formname, fields)
if formname ~= "mobs_npc:trade" then return end
if fields then
local trade = ""
for k, v in pairs(fields) do
trade = tostring(k)
end
local id = trade:split("#")[3]
local self = nil
if id ~= nil then
for k, v in pairs(minetest.luaentities) do
if v.object and v.id and v.id == id then
self = v
break
end
end
end
if self ~= nil then
local trade_number = tonumber(trade:split("#")[2])
if trade_number ~= nil and self.trades[trade_number] ~= nil then
local price = self.trades[trade_number][2]
local goods = self.trades[trade_number][1]
local inv = player:get_inventory()
if inv:contains_item("main", price) then
inv:remove_item("main", price)
local leftover = inv:add_item("main", goods)
if leftover:get_count() > 0 then
-- drop item(s) in front of player
local droppos = player:get_pos()
local dir = player:get_look_dir()
droppos.x = droppos.x + dir.x
droppos.z = droppos.z + dir.z
minetest.add_item(droppos, leftover)
end
end
end
end
end
end)