2016-12-18 18:32:39 +01:00
|
|
|
-- Trading code for Advanced NPC by Zorman2000
|
2016-12-06 02:22:33 +01:00
|
|
|
|
|
|
|
npc.trade = {}
|
|
|
|
|
|
|
|
npc.trade.CASUAL = "casual"
|
|
|
|
npc.trade.TRADER = "trader"
|
|
|
|
npc.trade.NONE = "none"
|
|
|
|
|
2016-12-15 02:03:51 +01:00
|
|
|
npc.trade.OFFER_BUY = "buy"
|
|
|
|
npc.trade.OFFER_SELL = "sell"
|
|
|
|
|
2017-02-09 17:38:14 +01:00
|
|
|
-- This variable establishes how much items a dedicated
|
|
|
|
-- trader will buy until retiring the offer
|
|
|
|
npc.trade.DEDICATED_MAX_BUY_AMOUNT = 5
|
|
|
|
|
2016-12-15 20:42:45 +01:00
|
|
|
-- This table holds all responses for trades
|
|
|
|
npc.trade.results = {
|
2016-12-16 02:51:06 +01:00
|
|
|
single_trade_offer = {},
|
2017-02-24 13:56:31 +01:00
|
|
|
trade_offers = {},
|
|
|
|
custom_trade_offer = {}
|
2016-12-15 20:42:45 +01:00
|
|
|
}
|
|
|
|
|
2017-02-23 13:27:25 +01:00
|
|
|
-- This is the text to be shown each time the NPC has more
|
|
|
|
-- than one custom trade options to choose from
|
|
|
|
npc.trade.CUSTOM_TRADES_PROMPT_TEXT = "Hi there, how can I help you today?"
|
|
|
|
|
2016-12-15 02:03:51 +01:00
|
|
|
-- Casual trader NPC dialogues definition
|
|
|
|
-- Casual buyer
|
|
|
|
npc.trade.CASUAL_TRADE_BUY_DIALOGUE = {
|
2016-12-29 04:10:06 +01:00
|
|
|
text = "I'm looking to buy some items, are you interested?",
|
2016-12-15 02:03:51 +01:00
|
|
|
casual_trade_type = npc.trade.OFFER_BUY,
|
2017-02-23 13:27:25 +01:00
|
|
|
dialogue_type = npc.dialogue.dialogue_type.casual_trade,
|
2016-12-15 02:03:51 +01:00
|
|
|
responses = {
|
|
|
|
[1] = {
|
2017-02-01 17:36:20 +01:00
|
|
|
text = "Sell",
|
2016-12-15 02:03:51 +01:00
|
|
|
action_type = "function",
|
|
|
|
response_id = 1,
|
|
|
|
action = function(self, player)
|
2016-12-16 02:51:06 +01:00
|
|
|
npc.trade.show_trade_offer_formspec(self, player, npc.trade.OFFER_BUY)
|
2016-12-15 02:03:51 +01:00
|
|
|
end
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
-- Casual seller
|
|
|
|
npc.trade.CASUAL_TRADE_SELL_DIALOGUE = {
|
|
|
|
text = "I have some items to sell, are you interested?",
|
2017-02-23 13:27:25 +01:00
|
|
|
dialogue_type = npc.dialogue.dialogue_type.casual_trade,
|
2016-12-15 02:03:51 +01:00
|
|
|
casual_trade_type = npc.trade.OFFER_SELL,
|
|
|
|
responses = {
|
|
|
|
[1] = {
|
2017-02-01 17:36:20 +01:00
|
|
|
text = "Buy",
|
2016-12-15 02:03:51 +01:00
|
|
|
action_type = "function",
|
|
|
|
response_id = 1,
|
|
|
|
action = function(self, player)
|
2016-12-16 02:51:06 +01:00
|
|
|
npc.trade.show_trade_offer_formspec(self, player, npc.trade.OFFER_SELL)
|
2016-12-15 02:03:51 +01:00
|
|
|
end
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-30 02:30:43 +01:00
|
|
|
-- Dedicated trade dialogue prompt
|
|
|
|
npc.trade.DEDICATED_TRADER_PROMPT = {
|
|
|
|
text = "Hello there, would you like to trade?",
|
2017-02-23 13:27:25 +01:00
|
|
|
dialogue_type = npc.dialogue.dialogue_type.dedicated_trade,
|
2017-01-30 02:30:43 +01:00
|
|
|
responses = {
|
|
|
|
[1] = {
|
|
|
|
text = "Buy",
|
|
|
|
action_type = "function",
|
|
|
|
response_id = 1,
|
|
|
|
action = function(self, player)
|
2017-02-02 03:16:31 +01:00
|
|
|
npc.trade.show_dedicated_trade_formspec(self, player, npc.trade.OFFER_SELL)
|
2017-01-30 02:30:43 +01:00
|
|
|
end
|
|
|
|
},
|
|
|
|
[2] = {
|
|
|
|
text = "Sell",
|
|
|
|
action_type = "function",
|
|
|
|
response_id = 2,
|
|
|
|
action = function(self, player)
|
2017-02-02 03:16:31 +01:00
|
|
|
npc.trade.show_dedicated_trade_formspec(self, player, npc.trade.OFFER_BUY)
|
2017-01-30 02:30:43 +01:00
|
|
|
end
|
2017-02-23 13:27:25 +01:00
|
|
|
},
|
|
|
|
[3] = {
|
|
|
|
text = "Other",
|
|
|
|
action_type = "function",
|
|
|
|
response_id = 3,
|
|
|
|
action = function(self, player)
|
|
|
|
local dialogue = npc.dialogue.create_custom_trade_options(self, player)
|
|
|
|
npc.dialogue.process_dialogue(self, dialogue, player:get_player_name())
|
|
|
|
end
|
2017-01-30 02:30:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-16 02:51:06 +01:00
|
|
|
function npc.trade.show_trade_offer_formspec(self, player, offer_type)
|
2016-12-15 20:42:45 +01:00
|
|
|
|
|
|
|
-- Strings for formspec, to include international support later
|
|
|
|
local prompt_string = " offers to buy from you"
|
|
|
|
local for_string = "for"
|
|
|
|
local buy_sell_string = "Sell"
|
|
|
|
|
|
|
|
-- Get offer. As this is casual trading, NPCs will only have
|
|
|
|
-- one trade offer
|
|
|
|
local trade_offer = self.trader_data.buy_offers[1]
|
|
|
|
if offer_type == npc.trade.OFFER_SELL then
|
|
|
|
trade_offer = self.trader_data.sell_offers[1]
|
|
|
|
prompt_string = " wants to sell to you"
|
|
|
|
buy_sell_string = "Buy"
|
|
|
|
end
|
|
|
|
|
|
|
|
local formspec = "size[8,4]"..
|
|
|
|
default.gui_bg..
|
|
|
|
default.gui_bg_img..
|
|
|
|
default.gui_slots..
|
|
|
|
"label[2,0.1;"..self.nametag..prompt_string.."]"..
|
|
|
|
"item_image_button[2,1.3;1.2,1.2;"..trade_offer.item..";item;]"..
|
|
|
|
"label[3.75,1.75;"..for_string.."]"..
|
2017-02-24 13:56:31 +01:00
|
|
|
"item_image_button[4.8,1.3;1.2,1.2;"..trade_offer.price[1]..";price;]"..
|
2016-12-15 20:42:45 +01:00
|
|
|
"button_exit[1,3.3;2.9,0.5;yes_option;"..buy_sell_string.."]"..
|
|
|
|
"button_exit[4.1,3.3;2.9,0.5;no_option;"..npc.dialogue.NEGATIVE_ANSWER_LABEL.."]"
|
|
|
|
|
|
|
|
-- Create entry into results table
|
2016-12-16 02:51:06 +01:00
|
|
|
npc.trade.results.single_trade_offer[player:get_player_name()] = {
|
|
|
|
trade_offer = trade_offer,
|
|
|
|
npc = self
|
2016-12-15 20:42:45 +01:00
|
|
|
}
|
|
|
|
-- Show formspec to player
|
2016-12-16 02:51:06 +01:00
|
|
|
minetest.show_formspec(player:get_player_name(), "advanced_npc:trade_offer", formspec)
|
2016-12-15 20:42:45 +01:00
|
|
|
end
|
|
|
|
|
2017-02-02 03:16:31 +01:00
|
|
|
|
|
|
|
function npc.trade.show_dedicated_trade_formspec(self, player, offers_type)
|
|
|
|
|
|
|
|
-- Choose the correct offers
|
|
|
|
local offers = self.trader_data.buy_offers
|
|
|
|
local menu_offer_type = "sell"
|
|
|
|
if offers_type == npc.trade.OFFER_SELL then
|
|
|
|
offers = self.trader_data.sell_offers
|
|
|
|
menu_offer_type = "buy"
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Create a grid with the items for trade offer
|
|
|
|
local max_columns = 4
|
|
|
|
local current_x = 0.2
|
|
|
|
local current_y = 0.5
|
|
|
|
local current_col = 1
|
|
|
|
local current_row = 1
|
|
|
|
local formspec = "size[8.9,8.2]"..
|
|
|
|
default.gui_bg..
|
|
|
|
default.gui_bg_img..
|
|
|
|
default.gui_slots..
|
|
|
|
"label[0.2,0.05;Click on the price button to "..menu_offer_type.." item]"
|
|
|
|
for i = 1, #offers do
|
2017-02-24 13:56:31 +01:00
|
|
|
local price_item_name = minetest.registered_items[npc.get_item_name(offers[i].price[1])].description
|
2017-02-23 13:27:25 +01:00
|
|
|
local count_label = ""
|
|
|
|
if npc.get_item_count(offers[i].item) > 1 then
|
|
|
|
count_label = "label["..(current_x + 1.35)..","..(current_y + 1)..";"..npc.get_item_count(offers[i].item).."]"
|
|
|
|
end
|
2017-02-02 03:16:31 +01:00
|
|
|
formspec = formspec..
|
|
|
|
"box["..current_x..","..current_y..";2,2.3;#212121]"..
|
|
|
|
"item_image["..(current_x + 0.45)..","..(current_y + 0.15)..";1.3,1.3;"..npc.get_item_name(offers[i].item).."]"..
|
2017-02-23 13:27:25 +01:00
|
|
|
count_label..
|
2017-02-24 13:56:31 +01:00
|
|
|
"item_image_button["..(current_x + 1.15)..","..(current_y + 1.4)..";1,1;"..offers[i].price[1]..";price"..i..";]"..
|
2017-02-02 03:16:31 +01:00
|
|
|
"label["..(current_x + 0.15)..","..(current_y + 1.7)..";Price]"
|
|
|
|
current_x = current_x + 2.1
|
|
|
|
current_col = current_col + 1
|
|
|
|
if current_col > 4 then
|
|
|
|
current_col = 1
|
|
|
|
current_x = 0.2
|
|
|
|
current_y = current_y + 2.4
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
formspec = formspec .. "button_exit[2.5,7.9;3.9,0.5;exit;Nevermind]"
|
|
|
|
|
2017-02-04 18:38:02 +01:00
|
|
|
-- Create entry into results table
|
|
|
|
npc.trade.results.trade_offers[player:get_player_name()] = {
|
|
|
|
offers_type = offers_type,
|
|
|
|
offers = offers,
|
|
|
|
npc = self
|
|
|
|
}
|
|
|
|
|
2017-02-02 03:16:31 +01:00
|
|
|
minetest.show_formspec(player:get_player_name(), "advanced_npc:dedicated_trading_offers", formspec)
|
|
|
|
|
|
|
|
end
|
|
|
|
|
2017-02-23 13:27:25 +01:00
|
|
|
-- For the moment, the trade offer for custom trade is always of sell type
|
|
|
|
function npc.trade.show_custom_trade_offer(self, player, offer)
|
|
|
|
local for_string = "for"
|
|
|
|
-- Create payments grid. Try to center it. When there are 4
|
|
|
|
-- payment options, a grid is to be displayed.
|
|
|
|
local price_count = #offer.price
|
|
|
|
local start_x = 2
|
|
|
|
local margin_x = 0
|
|
|
|
local start_y = 1.45
|
|
|
|
if price_count == 2 then
|
|
|
|
start_x = 1.5
|
|
|
|
margin_x = 0.3
|
|
|
|
elseif price_count == 3 then
|
|
|
|
start_x = 1.15
|
|
|
|
margin_x = 0.85
|
|
|
|
elseif price_count == 4 then
|
|
|
|
start_x = 1.5
|
|
|
|
start_y = 0.8
|
|
|
|
margin_x = 0.3
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Create payment grid
|
|
|
|
local price_grid = ""
|
|
|
|
for i = 1, #offer.price do
|
|
|
|
price_grid = price_grid.."item_image_button["..start_x..","..start_y..";1,1;"..offer.price[i]..";price"..i..";]"
|
|
|
|
if #offer.price == 4 and i == 2 then
|
|
|
|
start_x = 1.5
|
|
|
|
start_y = start_y + 1
|
|
|
|
else
|
|
|
|
start_x = start_x + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local formspec = "size[8,4]"..
|
|
|
|
default.gui_bg..
|
|
|
|
default.gui_bg_img..
|
|
|
|
default.gui_slots..
|
|
|
|
"label[2,0.1;"..self.nametag..": "..offer.dialogue_prompt.."]"..
|
|
|
|
price_grid..
|
|
|
|
"label["..(margin_x + 3.75)..",1.75;"..for_string.."]"..
|
|
|
|
"item_image_button["..(margin_x + 4.8)..",1.3;1.2,1.2;"..offer.item..";item;]"..
|
|
|
|
"button_exit[1,3.3;2.9,0.5;yes_option;"..offer.button_prompt.."]"..
|
|
|
|
"button_exit[4.1,3.3;2.9,0.5;no_option;"..npc.dialogue.NEGATIVE_ANSWER_LABEL.."]"
|
|
|
|
|
|
|
|
-- Create entry into results table
|
2017-02-24 13:56:31 +01:00
|
|
|
npc.trade.results.custom_trade_offer[player:get_player_name()] = {
|
|
|
|
trade_offer = offer,
|
2017-02-23 13:27:25 +01:00
|
|
|
npc = self
|
|
|
|
}
|
|
|
|
-- Show formspec to player
|
2017-02-24 13:56:31 +01:00
|
|
|
minetest.show_formspec(player:get_player_name(), "advanced_npc:custom_trade_offer", formspec)
|
2017-02-23 13:27:25 +01:00
|
|
|
end
|
|
|
|
|
2016-12-06 02:22:33 +01:00
|
|
|
function npc.trade.get_random_trade_status()
|
|
|
|
local chance = math.random(1,10)
|
|
|
|
|
|
|
|
if chance < 3 then
|
|
|
|
-- Non-trader
|
|
|
|
return npc.trade.NONE
|
|
|
|
elseif 3 <= chance and chance <= 7 then
|
|
|
|
-- Casual trader
|
|
|
|
return npc.trade.CASUAL
|
|
|
|
elseif chance > 7 then
|
|
|
|
-- Trader by profession
|
|
|
|
return npc.trade.TRADER
|
|
|
|
end
|
2016-12-15 02:03:51 +01:00
|
|
|
end
|
|
|
|
|
2017-02-01 17:36:20 +01:00
|
|
|
-- This function generates and stores on the NPC data trade
|
|
|
|
-- offers depending on the trader status.
|
|
|
|
function npc.trade.generate_trade_offers_by_status(self)
|
|
|
|
-- Get trader status
|
|
|
|
local status = self.trader_data.trader_status
|
|
|
|
-- Check what is the trader status
|
|
|
|
if status == npc.trade.NONE then
|
|
|
|
-- For none, clear all offers
|
|
|
|
self.trader_data.buy_offers = {}
|
|
|
|
self.trader_data.sell_offers = {}
|
|
|
|
elseif status == npc.trade.CASUAL then
|
|
|
|
-- For casual, generate one buy and one sell offer
|
|
|
|
self.trader_data.buy_offers = {
|
|
|
|
[1] = npc.trade.get_casual_trade_offer(self, npc.trade.OFFER_BUY)
|
|
|
|
}
|
|
|
|
self.trader_data.sell_offers = {
|
|
|
|
[1] = npc.trade.get_casual_trade_offer(self, npc.trade.OFFER_SELL)
|
|
|
|
}
|
|
|
|
elseif status == npc.trade.TRADER then
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Clear current offers
|
|
|
|
self.trader_data.buy_offers = {}
|
|
|
|
self.trader_data.sell_offers = {}
|
2017-02-01 17:36:20 +01:00
|
|
|
-- Get trade offers for a dedicated trader
|
|
|
|
local offers = npc.trade.get_dedicated_trade_offers(self)
|
|
|
|
-- Store buy offers
|
|
|
|
for i = 1, #offers.buy do
|
|
|
|
table.insert(self.trader_data.buy_offers, offers.buy[i])
|
|
|
|
end
|
|
|
|
-- Store sell offers
|
|
|
|
for i = 1, #offers.sell do
|
|
|
|
table.insert(self.trader_data.sell_offers, offers.sell[i])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-17 18:01:35 +01:00
|
|
|
-- Convenience method that retrieves all the currency
|
|
|
|
-- items that a NPC has on his/her inventory
|
|
|
|
function npc.trade.get_currencies_in_inventory(self)
|
|
|
|
local result = {}
|
2017-01-26 18:55:04 +01:00
|
|
|
local tier3 = npc.inventory_contains(self, npc.trade.prices.currency.tier3.string)
|
|
|
|
local tier2 = npc.inventory_contains(self, npc.trade.prices.currency.tier2.string)
|
|
|
|
local tier1 = npc.inventory_contains(self, npc.trade.prices.currency.tier1.string)
|
2016-12-17 18:01:35 +01:00
|
|
|
if tier3 ~= nil then
|
|
|
|
table.insert(result, {name = npc.get_item_name(tier3.item_string),
|
|
|
|
count = npc.get_item_count(tier3.item_string)} )
|
|
|
|
end
|
|
|
|
if tier2 ~= nil then
|
|
|
|
table.insert(result, {name = npc.get_item_name(tier2.item_string),
|
|
|
|
count = npc.get_item_count(tier2.item_string)} )
|
|
|
|
end
|
|
|
|
if tier1 ~= nil then
|
|
|
|
table.insert(result, {name = npc.get_item_name(tier1.item_string),
|
|
|
|
count = npc.get_item_count(tier1.item_string)} )
|
|
|
|
end
|
|
|
|
|
2017-03-23 15:20:08 +01:00
|
|
|
--minetest.log("Found currency in inventory: "..dump(result))
|
2016-12-17 18:01:35 +01:00
|
|
|
return result
|
|
|
|
end
|
|
|
|
|
2016-12-15 02:03:51 +01:00
|
|
|
-- This function will return an offer object, based
|
|
|
|
-- on the items the NPC has.
|
2016-12-15 20:42:45 +01:00
|
|
|
function npc.trade.get_casual_trade_offer(self, offer_type)
|
2016-12-17 18:01:35 +01:00
|
|
|
local result = {}
|
|
|
|
-- Check offer type
|
|
|
|
if offer_type == npc.trade.OFFER_BUY then
|
|
|
|
-- Create buy offer based on what the NPC can actually buy
|
|
|
|
local currencies = npc.trade.get_currencies_in_inventory(self)
|
|
|
|
-- Choose a random currency
|
|
|
|
local chosen_tier = currencies[math.random(#currencies)]
|
|
|
|
-- Get items for this currency
|
2017-01-26 18:55:04 +01:00
|
|
|
local buyable_items =
|
|
|
|
npc.trade.prices.get_items_for_currency_count(chosen_tier.name, chosen_tier.count, 0.5)
|
2016-12-17 18:01:35 +01:00
|
|
|
-- Select a random item from the buyable items
|
|
|
|
local item_set = {}
|
2017-01-26 18:55:04 +01:00
|
|
|
for item,price_info in pairs(buyable_items) do
|
2016-12-17 18:01:35 +01:00
|
|
|
table.insert(item_set, item)
|
|
|
|
end
|
|
|
|
local item = item_set[math.random(#item_set)]
|
|
|
|
-- Choose buying quantity. Since this is a buy offer, NPC will buy items
|
|
|
|
-- at half the price. Therefore, NPC will always ask for even quantities
|
|
|
|
-- so that the price count is always an integer number
|
2017-01-26 18:55:04 +01:00
|
|
|
local amount_to_buy = math.random(buyable_items[item].min_buyable_item_count, buyable_items[item].max_buyable_item_count)
|
|
|
|
-- Create trade offer
|
2017-03-23 15:20:08 +01:00
|
|
|
--minetest.log("Buyable item: "..dump(buyable_items[item]))
|
2017-01-30 02:08:34 +01:00
|
|
|
result = npc.trade.create_offer(npc.trade.OFFER_BUY, item, buyable_items[item].price, buyable_items[item].min_buyable_item_price, amount_to_buy)
|
2016-12-17 18:01:35 +01:00
|
|
|
else
|
2016-12-17 20:27:23 +01:00
|
|
|
-- Make sell offer, NPC will sell items to player at regular price
|
2016-12-17 18:01:35 +01:00
|
|
|
-- NPC will also offer items from their inventory
|
|
|
|
local sellable_items = {}
|
|
|
|
for i = 1, #self.inventory do
|
|
|
|
if self.inventory[i] ~= "" then
|
|
|
|
if npc.trade.prices.is_item_currency(self.inventory[i]) == false then
|
|
|
|
table.insert(sellable_items, self.inventory[i])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
-- Choose a random item from the sellable items
|
|
|
|
local item = sellable_items[math.random(#sellable_items)]
|
|
|
|
-- Choose how many of this item will be sold to player
|
|
|
|
local count = math.random(npc.get_item_count(item))
|
2017-01-26 18:55:04 +01:00
|
|
|
-- Create trade offer
|
2017-01-30 02:08:34 +01:00
|
|
|
result = npc.trade.create_offer(npc.trade.OFFER_SELL, npc.get_item_name(item), nil, nil, count)
|
2017-01-26 18:55:04 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
return result
|
|
|
|
end
|
|
|
|
|
2017-01-30 02:30:43 +01:00
|
|
|
-- The following function create buy and sell offers for dedicated traders,
|
|
|
|
-- based on the trader list and the source of items. Initially, it will only
|
|
|
|
-- be NPC inventories. In the future, it should support both NPC and chest
|
|
|
|
-- inventories,
|
2017-02-01 17:36:20 +01:00
|
|
|
function npc.trade.get_dedicated_trade_offers(self)
|
2017-01-31 03:51:16 +01:00
|
|
|
local offers = {
|
2017-02-01 17:36:20 +01:00
|
|
|
sell = {},
|
|
|
|
buy = {}
|
2017-01-31 03:51:16 +01:00
|
|
|
}
|
2017-01-30 02:30:43 +01:00
|
|
|
|
2017-02-01 17:36:20 +01:00
|
|
|
local trade_list = self.trader_data.trade_list.both
|
|
|
|
|
|
|
|
for item_name, trade_info in pairs(trade_list) do
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Abort if more than 12 buy or sell offers are made
|
|
|
|
if table.getn(offers.sell) >= 12 or table.getn(offers.buy) >= 12 then
|
|
|
|
break
|
|
|
|
end
|
2017-01-31 03:51:16 +01:00
|
|
|
-- For each item on the trader list, check if it is in the NPC inventory.
|
|
|
|
-- If it is, create a sell offer, else create a buy offer if possible.
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Also, avoid creating sell offers immediately if the item was just bought
|
2017-02-01 17:36:20 +01:00
|
|
|
local item = npc.inventory_contains(self, item_name)
|
2017-02-09 17:38:14 +01:00
|
|
|
if item ~= nil and trade_info.last_offer_type ~= npc.trade.OFFER_BUY then
|
|
|
|
|
|
|
|
-- Create sell offer for this item. Currently, traders will offer to sell only
|
2017-02-23 13:27:25 +01:00
|
|
|
-- one of their items to allow the fine control for players to buy what they want.
|
2017-02-09 17:38:14 +01:00
|
|
|
-- This requires, however, that the trade offers are re-generated everytime a
|
|
|
|
-- sell is made.
|
|
|
|
table.insert(offers.sell, npc.trade.create_offer(
|
|
|
|
npc.trade.OFFER_SELL,
|
|
|
|
item_name,
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
1)
|
|
|
|
)
|
|
|
|
-- Set last offer type
|
|
|
|
trade_info.last_offer_type = npc.trade.OFFER_SELL
|
|
|
|
|
2017-01-31 03:51:16 +01:00
|
|
|
else
|
2017-02-01 17:36:20 +01:00
|
|
|
-- Avoid flipping an item to the buy side if the stock was just depleted
|
|
|
|
if trade_info.last_offer_type ~= npc.trade.OFFER_SELL then
|
|
|
|
-- Create buy offer for this item
|
|
|
|
-- Only do if the NPC can actually afford the items.
|
|
|
|
local currencies = npc.trade.get_currencies_in_inventory(self)
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Check if currency isn't empty
|
|
|
|
if #currencies > 0 then
|
|
|
|
-- Choose a random currency
|
|
|
|
local chosen_tier = currencies[math.random(#currencies)]
|
|
|
|
-- Get items for this currency
|
|
|
|
local buyable_items =
|
|
|
|
npc.trade.prices.get_items_for_currency_count(chosen_tier.name, chosen_tier.count, 0.5)
|
|
|
|
-- Check if the item from trader list is present in the buyable items list
|
|
|
|
for buyable_item, price_info in pairs(buyable_items) do
|
|
|
|
if buyable_item == item_name then
|
|
|
|
-- If item found, create a buy offer for this item
|
|
|
|
-- Again, offers are created for one item only. Buy offers should be removed
|
|
|
|
-- after the NPC has bought a certain quantity, say, 5 items.
|
|
|
|
minetest.log("Item: "..item_name)
|
|
|
|
minetest.log("Trade info: "..dump(trade_info))
|
2017-03-23 15:20:08 +01:00
|
|
|
--minetest.log("Logic: "..dump(trade_info.item_bought_count == nil
|
|
|
|
-- or (trade_info.item_bought_count ~= nil and trade_info.item_bought_count <= npc.trade.DEDICATED_MAX_BUY_AMOUNT)))
|
2017-02-09 17:38:14 +01:00
|
|
|
if trade_info.item_bought_count == nil
|
|
|
|
or (trade_info.item_bought_count ~= nil and trade_info.item_bought_count <= npc.trade.DEDICATED_MAX_BUY_AMOUNT) then
|
|
|
|
-- Create trade offer for this item
|
|
|
|
table.insert(offers.buy, npc.trade.create_offer(
|
|
|
|
npc.trade.OFFER_BUY,
|
|
|
|
item_name,
|
|
|
|
price_info.price,
|
|
|
|
price_info.min_buyable_item_price,
|
|
|
|
price_info.min_buyable_item_count)
|
|
|
|
)
|
|
|
|
-- Set last offer type
|
|
|
|
trade_info.last_offer_type = npc.trade.OFFER_BUY
|
|
|
|
else
|
|
|
|
-- Clear the trade info for this item
|
|
|
|
trade_info.item_bought_count = 0
|
|
|
|
end
|
|
|
|
end
|
2017-02-01 17:36:20 +01:00
|
|
|
end
|
2017-01-31 03:51:16 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return offers
|
2017-01-30 02:30:43 +01:00
|
|
|
end
|
|
|
|
|
2017-01-26 18:55:04 +01:00
|
|
|
-- Creates a trade offer based on the offer type, given item and count. If
|
2017-01-30 02:08:34 +01:00
|
|
|
-- the offer is a "buy" offer, it is required to provide the price item and
|
|
|
|
-- the minimum price item count.
|
|
|
|
function npc.trade.create_offer(offer_type, item, price, min_price_item_count, count)
|
2017-01-26 18:55:04 +01:00
|
|
|
local result = {}
|
|
|
|
-- Check offer type
|
|
|
|
if offer_type == npc.trade.OFFER_BUY then
|
|
|
|
-- Get price for the given item
|
2017-01-30 02:08:34 +01:00
|
|
|
-- Create price itemstring
|
|
|
|
local price_string = price.tier.." "
|
|
|
|
..tostring( min_price_item_count * count )
|
|
|
|
|
|
|
|
-- Build the return object
|
2017-02-24 13:56:31 +01:00
|
|
|
-- Price is always an array, in this case of size 1
|
2017-01-30 02:08:34 +01:00
|
|
|
result = {
|
|
|
|
offer_type = offer_type,
|
|
|
|
item = item.." "..count,
|
2017-02-24 13:56:31 +01:00
|
|
|
price = {[1] = price_string}
|
2017-01-30 02:08:34 +01:00
|
|
|
}
|
2017-01-26 18:55:04 +01:00
|
|
|
else
|
|
|
|
-- Make sell offer, NPC will sell items to player at regular price
|
2016-12-17 18:01:35 +01:00
|
|
|
-- Get and calculate price for this object
|
2017-01-26 18:55:04 +01:00
|
|
|
local price_object = npc.trade.prices.table[item]
|
2016-12-17 20:27:23 +01:00
|
|
|
-- Check price object, if price < 1 then offer to sell for 1
|
|
|
|
if price_object.count < 1 then
|
|
|
|
price_object.count = 1
|
|
|
|
end
|
2016-12-17 18:01:35 +01:00
|
|
|
local price_string = price_object.tier.." "..tostring(price_object.count * count)
|
|
|
|
-- Build return object
|
2017-02-24 13:56:31 +01:00
|
|
|
-- Price is always an array, in this case of size 1
|
2016-12-17 18:01:35 +01:00
|
|
|
result = {
|
|
|
|
offer_type = offer_type,
|
|
|
|
item = npc.get_item_name(item).." "..count,
|
2017-02-24 13:56:31 +01:00
|
|
|
price = {[1] = price_string}
|
2016-12-17 18:01:35 +01:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
return result
|
2017-01-26 18:55:04 +01:00
|
|
|
|
2016-12-15 20:42:45 +01:00
|
|
|
end
|
|
|
|
|
2017-02-23 13:27:25 +01:00
|
|
|
-- A custom sell trade offer is a special type of trading the NPC can
|
|
|
|
-- have where a different prompt and multiple payment objects are
|
|
|
|
-- required from the player. A good example is offering to repair a sword,
|
|
|
|
-- where the player has to give an amount of currency and the sword to
|
|
|
|
-- repair in exchange to get a fully repaired sword.
|
|
|
|
-- For the moment being, only sell is supported.
|
|
|
|
function npc.trade.create_custom_sell_trade_offer(option_prompt, dialogue_prompt, button_prompt, item, payments)
|
|
|
|
return {
|
|
|
|
offer_type = npc.OFFER_SELL,
|
|
|
|
option_prompt = option_prompt,
|
|
|
|
dialogue_prompt = dialogue_prompt,
|
|
|
|
button_prompt = button_prompt,
|
|
|
|
item = item,
|
|
|
|
price = payments
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2017-01-26 18:55:04 +01:00
|
|
|
|
2017-02-09 17:38:14 +01:00
|
|
|
-- TODO: This method needs to be refactored to be able to manage
|
|
|
|
-- both NPC inventories and chest inventories.
|
|
|
|
-- Returns true if trade was possible, else returns false.
|
2016-12-16 02:51:06 +01:00
|
|
|
function npc.trade.perform_trade(self, player_name, offer)
|
|
|
|
|
|
|
|
local item_stack = ItemStack(offer.item)
|
2017-02-24 13:56:31 +01:00
|
|
|
-- Create item stacks for each price item
|
|
|
|
local price_stacks = {}
|
|
|
|
for i = 1, #offer.price do
|
|
|
|
table.insert(price_stacks, ItemStack(offer.price[i]))
|
|
|
|
end
|
2016-12-16 02:51:06 +01:00
|
|
|
local inv = minetest.get_inventory({type = "player", name = player_name})
|
|
|
|
|
|
|
|
-- Check if offer is a buy or sell
|
|
|
|
if offer.offer_type == npc.trade.OFFER_BUY then
|
|
|
|
-- If NPC is buying from player, then player loses item, gets price
|
|
|
|
-- Check player has the item being buyed
|
|
|
|
if inv:contains_item("main", item_stack) then
|
|
|
|
-- Check if there is enough room to add the price item to player
|
|
|
|
if inv:room_for_item("main", price_stack) then
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Remove item from player
|
2016-12-16 02:51:06 +01:00
|
|
|
inv:remove_item("main", item_stack)
|
2017-02-24 13:56:31 +01:00
|
|
|
-- Remove price item(s) from NPC
|
|
|
|
for i = 1, #price_stacks do
|
|
|
|
npc.take_item_from_inventory_itemstring(self, price_stacks[i])
|
|
|
|
end
|
2016-12-16 02:51:06 +01:00
|
|
|
-- Add item to NPC's inventory
|
|
|
|
npc.add_item_to_inventory_itemstring(self, offer.item)
|
|
|
|
-- Add price items to player
|
2017-02-24 13:56:31 +01:00
|
|
|
for i = 1, #price_stacks do
|
|
|
|
inv:add_item("main", price_stacks[i])
|
|
|
|
end
|
2016-12-16 02:51:06 +01:00
|
|
|
-- Send message to player
|
|
|
|
minetest.chat_send_player(player_name, "Thank you!")
|
2017-02-09 17:38:14 +01:00
|
|
|
return true
|
2016-12-16 02:51:06 +01:00
|
|
|
else
|
|
|
|
minetest.chat_send_player(player_name,
|
2017-01-26 18:55:04 +01:00
|
|
|
"Looks like you can't get what I'm giving you for payment!")
|
2017-02-09 17:38:14 +01:00
|
|
|
return false
|
2016-12-16 02:51:06 +01:00
|
|
|
end
|
|
|
|
else
|
|
|
|
minetest.chat_send_player(player_name, "Looks like you don't have what I want to buy...")
|
2017-02-09 17:38:14 +01:00
|
|
|
return false
|
2016-12-16 02:51:06 +01:00
|
|
|
end
|
|
|
|
else
|
|
|
|
-- If NPC is selling to the player, then player gives price and gets
|
2017-02-24 13:56:31 +01:00
|
|
|
-- item, NPC loses item and gets price.
|
2016-12-16 02:51:06 +01:00
|
|
|
-- Check NPC has the required item to pay
|
|
|
|
if inv:contains_item("main", price_stack) then
|
|
|
|
-- Check if there is enough room to add the item to player
|
|
|
|
if inv:room_for_item("main", item_stack) then
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Remove price item from player
|
2017-02-24 13:56:31 +01:00
|
|
|
for i = 1, #price_stacks do
|
|
|
|
inv:remove_item("main", price_stacks[i])
|
|
|
|
end
|
2017-02-09 17:38:14 +01:00
|
|
|
-- Remove sell item from NPC
|
|
|
|
npc.take_item_from_inventory_itemstring(self, offer.item)
|
2017-02-24 13:56:31 +01:00
|
|
|
-- Add price to NPC's inventory
|
|
|
|
for i = 1, #offer.price do
|
|
|
|
npc.add_item_to_inventory_itemstring(self, offer.price[i])
|
|
|
|
end
|
|
|
|
-- Add item items to player
|
2016-12-16 02:51:06 +01:00
|
|
|
inv:add_item("main", item_stack)
|
|
|
|
-- Send message to player
|
|
|
|
minetest.chat_send_player(player_name, "Thank you!")
|
2017-02-09 17:38:14 +01:00
|
|
|
return true
|
2016-12-16 02:51:06 +01:00
|
|
|
else
|
|
|
|
minetest.chat_send_player(player_name, "Looks like you can't carry anything else...")
|
2017-02-09 17:38:14 +01:00
|
|
|
return false
|
2016-12-16 02:51:06 +01:00
|
|
|
end
|
|
|
|
else
|
|
|
|
minetest.chat_send_player(player_name, "Looks like you don't have what I'm asking for!")
|
2017-02-09 17:38:14 +01:00
|
|
|
return false
|
2016-12-16 02:51:06 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-12-15 20:42:45 +01:00
|
|
|
-- Handler for chat formspec
|
|
|
|
minetest.register_on_player_receive_fields(function (player, formname, fields)
|
|
|
|
-- Additional checks for other forms should be handled here
|
2017-02-04 18:38:02 +01:00
|
|
|
-- Handle casual trade dialogue
|
2016-12-16 02:51:06 +01:00
|
|
|
if formname == "advanced_npc:trade_offer" then
|
2016-12-15 20:42:45 +01:00
|
|
|
local player_name = player:get_player_name()
|
|
|
|
|
|
|
|
if fields then
|
2016-12-16 02:51:06 +01:00
|
|
|
local player_response = npc.trade.results.single_trade_offer[player_name]
|
2017-01-19 01:34:02 +01:00
|
|
|
-- Unlock the action timer
|
|
|
|
npc.unlock_actions(player_response.npc)
|
|
|
|
|
2016-12-15 20:42:45 +01:00
|
|
|
if fields.yes_option then
|
2016-12-16 02:51:06 +01:00
|
|
|
npc.trade.perform_trade(player_response.npc, player_name, player_response.trade_offer)
|
2016-12-15 20:42:45 +01:00
|
|
|
elseif fields.no_option then
|
|
|
|
minetest.chat_send_player(player_name, "Talk to me if you change your mind!")
|
|
|
|
end
|
2017-01-19 01:34:02 +01:00
|
|
|
|
2016-12-15 20:42:45 +01:00
|
|
|
end
|
2017-02-04 18:38:02 +01:00
|
|
|
elseif formname == "advanced_npc:dedicated_trading_offers" then
|
|
|
|
local player_name = player:get_player_name()
|
2016-12-15 20:42:45 +01:00
|
|
|
|
2017-02-04 18:38:02 +01:00
|
|
|
if fields then
|
|
|
|
local player_response = npc.trade.results.trade_offers[player_name]
|
|
|
|
-- Unlock the action timer
|
|
|
|
npc.unlock_actions(player_response.npc)
|
|
|
|
|
|
|
|
local trade_offers = npc.trade.results.trade_offers[player_name].offers
|
|
|
|
-- Check which price was clicked
|
|
|
|
for i = 1, #trade_offers do
|
|
|
|
local price_button = "price"..tostring(i)
|
|
|
|
if fields[price_button] then
|
2017-02-09 17:38:14 +01:00
|
|
|
local trade_result = npc.trade.perform_trade(player_response.npc, player_name, trade_offers[i])
|
|
|
|
if trade_result == true then
|
|
|
|
-- Lock actions
|
|
|
|
npc.lock_actions(player_response.npc)
|
|
|
|
-- Account for buyed items
|
|
|
|
if player_response.offers_type == npc.trade.OFFER_BUY then
|
|
|
|
-- Increase the item bought count
|
|
|
|
local offer_item_name = npc.get_item_name(trade_offers[i].item)
|
|
|
|
minetest.log("Bought item name: "..dump(offer_item_name))
|
|
|
|
minetest.log(dump(player_response.npc.trader_data.trade_list.both[offer_item_name]))
|
|
|
|
-- Check if this item has been bought before
|
|
|
|
if player_response.npc.trader_data.trade_list.both[offer_item_name].item_bought_count == nil then
|
|
|
|
-- Set first count to 1
|
|
|
|
player_response.npc.trader_data.trade_list.both[offer_item_name].item_bought_count = 1
|
|
|
|
else
|
|
|
|
-- Increase count
|
|
|
|
player_response.npc.trader_data.trade_list.both[offer_item_name].item_bought_count
|
|
|
|
= player_response.npc.trader_data.trade_list.both[offer_item_name].item_bought_count + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
-- Re-generate trade offers
|
|
|
|
npc.trade.generate_trade_offers_by_status(player_response.npc)
|
|
|
|
-- Show refreshed formspec again to player
|
|
|
|
npc.trade.show_dedicated_trade_formspec(player_response.npc, player, player_response.offers_type)
|
|
|
|
return true
|
|
|
|
else
|
|
|
|
minetest.close_formspec(player_name, "advanced_npc:dedicated_trading_offers")
|
|
|
|
return false
|
|
|
|
end
|
2017-02-04 18:38:02 +01:00
|
|
|
--minetest.log("Player selected: "..dump(trade_offers[i]))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-02-24 13:56:31 +01:00
|
|
|
elseif formname == "advanced_npc:custom_trade_offer" then
|
|
|
|
-- Handle custom trade formspec
|
|
|
|
local player_name = player:get_player_name()
|
|
|
|
|
|
|
|
if fields then
|
|
|
|
local player_response = npc.trade.results.custom_trade_offer[player_name]
|
|
|
|
-- Unlock the action timer
|
|
|
|
npc.unlock_actions(player_response.npc)
|
|
|
|
|
|
|
|
if fields.yes_option then
|
|
|
|
npc.trade.perform_trade(player_response.npc, player_name, player_response.trade_offer)
|
|
|
|
elseif fields.no_option then
|
|
|
|
minetest.chat_send_player(player_name, "Talk to me if you change your mind!")
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
2017-02-04 18:38:02 +01:00
|
|
|
end
|
2016-12-15 20:42:45 +01:00
|
|
|
end)
|