forked from mtcontrib/exchange_shop
First version 🐈
This commit is contained in:
commit
89d08b907b
17
README.txt
Normal file
17
README.txt
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
Exchange Shop
|
||||||
|
===============
|
||||||
|
|
||||||
|
This mod adds an improved ("currency" compatible) shop to your world.
|
||||||
|
|
||||||
|
Features:
|
||||||
|
* 4 buyer and 4 seller slots
|
||||||
|
* Much storage capacity
|
||||||
|
* Custom shop title settable
|
||||||
|
* pipeworks compatibilty
|
||||||
|
|
||||||
|
Optional dependencies:
|
||||||
|
* currency
|
||||||
|
* bitchange
|
||||||
|
* wrench
|
||||||
|
|
||||||
|
License: CC0 (for everything)
|
97
currency_migrate.lua
Normal file
97
currency_migrate.lua
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
-- Combine stacks into a new list
|
||||||
|
local function compress_list(list)
|
||||||
|
local items = {}
|
||||||
|
local new_list = {}
|
||||||
|
for i, stack in pairs(list or {}) do
|
||||||
|
if not stack:is_empty() then
|
||||||
|
if stack:get_stack_max() == 1 then
|
||||||
|
table.insert(new_list, stack)
|
||||||
|
else
|
||||||
|
items[stack:get_name()] = (items[stack:get_name()] or 0)
|
||||||
|
+ stack:get_count()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
for name, count in pairs(items) do
|
||||||
|
local max = ItemStack(name):get_stack_max()
|
||||||
|
|
||||||
|
repeat
|
||||||
|
local take = math.min(max, count)
|
||||||
|
local stack = ItemStack(name)
|
||||||
|
stack:set_count(take)
|
||||||
|
table.insert(new_list, stack)
|
||||||
|
count = count - take
|
||||||
|
until count == 0
|
||||||
|
end
|
||||||
|
return new_list
|
||||||
|
end
|
||||||
|
|
||||||
|
local function list_add_list(inv, list_name, list)
|
||||||
|
local leftover_list = {}
|
||||||
|
for i, stack in pairs(list or {}) do
|
||||||
|
local leftover = inv:add_item(list_name, stack)
|
||||||
|
if not leftover:is_empty() then
|
||||||
|
table.insert(leftover_list, leftover)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if #leftover_list > 0 then
|
||||||
|
minetest.log("warning", "[exchange_shop] List " .. list_name
|
||||||
|
.. " is full. Possible item loss!")
|
||||||
|
end
|
||||||
|
return leftover_list
|
||||||
|
end
|
||||||
|
|
||||||
|
local function migrate_shop_node(pos, node)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local owner = meta:get_string("owner")
|
||||||
|
local title = meta:get_string("infotext")
|
||||||
|
local inv = meta:get_inventory()
|
||||||
|
local def = minetest.registered_nodes[exchange_shop.shopname]
|
||||||
|
|
||||||
|
-- Create new slots
|
||||||
|
def.on_construct(pos)
|
||||||
|
meta:set_string("owner", owner)
|
||||||
|
meta:set_string("infotext", title)
|
||||||
|
|
||||||
|
list_add_list(inv, "custm", inv:get_list("customers_gave"))
|
||||||
|
inv:set_size("customers_gave", 0)
|
||||||
|
|
||||||
|
local new_owner_gives = compress_list(inv:get_list("owner_gives"))
|
||||||
|
local new_owner_wants = compress_list(inv:get_list("owner_wants"))
|
||||||
|
local dst_gives = "cust_og"
|
||||||
|
local dst_wants = "cust_ow"
|
||||||
|
if #new_owner_gives > 4 or #new_owner_wants > 4 then
|
||||||
|
-- Not enough space (from 6 slots to 4)
|
||||||
|
-- redirect everything to the stock
|
||||||
|
dst_gives = "stock"
|
||||||
|
dst_wants = "custm"
|
||||||
|
end
|
||||||
|
list_add_list(inv, dst_gives, new_owner_gives)
|
||||||
|
list_add_list(inv, dst_wants, new_owner_wants)
|
||||||
|
|
||||||
|
inv:set_size("owner_gives", 0)
|
||||||
|
inv:set_size("owner_takes", 0)
|
||||||
|
|
||||||
|
node.name = exchange_shop.shopname
|
||||||
|
minetest.swap_node(pos, node)
|
||||||
|
end
|
||||||
|
|
||||||
|
minetest.register_lbm({
|
||||||
|
label = "currency shop to exchange shop migration",
|
||||||
|
name = "exchange_shop:currency_migrate",
|
||||||
|
nodenames = { "currency:shop" },
|
||||||
|
run_at_every_load = true, -- TODO this for testing only
|
||||||
|
action = migrate_shop_node
|
||||||
|
})
|
||||||
|
|
||||||
|
-- Clean up garbage
|
||||||
|
minetest.register_on_joinplayer(function(player)
|
||||||
|
local inv = player:get_inventory()
|
||||||
|
for i, name in pairs({"customer_gives", "customer_gets"}) do
|
||||||
|
if inv:get_size(name) > 0 then
|
||||||
|
local leftover = list_add_list(inv, "main", inv:get_list(name))
|
||||||
|
list_add_list(inv, "craft", leftover)
|
||||||
|
inv:set_size(name, 0)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end)
|
19
currency_override.lua
Normal file
19
currency_override.lua
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
local def = table.copy(minetest.registered_nodes["currency:shop"])
|
||||||
|
def.groups.not_in_creative_inventory = 1
|
||||||
|
|
||||||
|
minetest.override_item("currency:shop", {
|
||||||
|
groups = def.groups,
|
||||||
|
on_construct = function() end,
|
||||||
|
after_place_node = function(pos, ...)
|
||||||
|
local node = minetest.get_node(pos)
|
||||||
|
node.name = exchange_shop.shopname
|
||||||
|
minetest.swap_node(pos, node)
|
||||||
|
|
||||||
|
local new_def = minetest.registered_nodes[exchange_shop.shopname]
|
||||||
|
if new_def.on_construct then
|
||||||
|
new_def.on_construct(pos)
|
||||||
|
end
|
||||||
|
new_def.after_place_node(pos, unpack({...}))
|
||||||
|
|
||||||
|
end
|
||||||
|
})
|
3
depends.txt
Normal file
3
depends.txt
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
currency?
|
||||||
|
bitchange?
|
||||||
|
wrench?
|
27
init.lua
Normal file
27
init.lua
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
exchange_shop = {}
|
||||||
|
exchange_shop.storage_size = 5 * 4
|
||||||
|
exchange_shop.shopname = "exchange_shop:shop"
|
||||||
|
|
||||||
|
local modpath = minetest.get_modpath("exchange_shop")
|
||||||
|
local has_currency = minetest.get_modpath("currency")
|
||||||
|
local has_bitchange = minetest.get_modpath("bitchange")
|
||||||
|
local migrate_currency = true -- TODO testing!
|
||||||
|
local slow_migrate_currency = false
|
||||||
|
|
||||||
|
|
||||||
|
if has_bitchange then
|
||||||
|
minetest.register_alias("exchange_shop:shop", "bitchange:shop")
|
||||||
|
exchange_shop.shopname = "bitchange:shop"
|
||||||
|
else
|
||||||
|
dofile(modpath .. "/shop_functions.lua")
|
||||||
|
dofile(modpath .. "/shop.lua")
|
||||||
|
end
|
||||||
|
|
||||||
|
if has_currency then
|
||||||
|
if migrate_currency then
|
||||||
|
dofile(modpath .. "/currency_migrate.lua")
|
||||||
|
end
|
||||||
|
if slow_migrate_currency then
|
||||||
|
dofile(modpath .. "/currency_override.lua")
|
||||||
|
end
|
||||||
|
end
|
2
mod.conf
Normal file
2
mod.conf
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
name = exchange_shop
|
||||||
|
optional_depends = currency, bitchange, wrench
|
267
shop.lua
Executable file
267
shop.lua
Executable file
@ -0,0 +1,267 @@
|
|||||||
|
--[[
|
||||||
|
Exchange Shop
|
||||||
|
|
||||||
|
This code is based on the idea of Dan Duncombe's exchange shop
|
||||||
|
https://web.archive.org/web/20160403113102/https://forum.minetest.net/viewtopic.php?id=7002
|
||||||
|
]]
|
||||||
|
|
||||||
|
|
||||||
|
local shop_positions = {}
|
||||||
|
|
||||||
|
local function get_exchange_shop_formspec(mode, pos, title)
|
||||||
|
local name = "nodemeta:"..pos.x..","..pos.y..","..pos.z
|
||||||
|
|
||||||
|
local function listring(src)
|
||||||
|
return "listring[".. name ..";" .. src .. "]" ..
|
||||||
|
"listring[current_player;main]"
|
||||||
|
end
|
||||||
|
if mode == "customer" then
|
||||||
|
-- customer
|
||||||
|
return (
|
||||||
|
"size[8,9;]"..
|
||||||
|
"label[0,0;Exchange shop]"..
|
||||||
|
"label[1,0.5;Owner needs:]"..
|
||||||
|
"list["..name..";cust_ow;1,1;2,2;]"..
|
||||||
|
"button[3,2.4;2,1;exchange;Exchange]"..
|
||||||
|
"label[5,0.5;Owner gives:]"..
|
||||||
|
"list["..name..";cust_og;5,1;2,2;]"..
|
||||||
|
"label[0.7,3.5;Ejected items:]"..
|
||||||
|
"label[0.7,3.8;(Remove me!)]"..
|
||||||
|
"list["..name..";cust_ej;3,3.5;4,1;]"..
|
||||||
|
"list[current_player;main;0,5;8,4;]"..
|
||||||
|
listring("cust_ej")
|
||||||
|
)
|
||||||
|
end
|
||||||
|
if mode == "owner_custm"
|
||||||
|
or mode == "owner_stock" then
|
||||||
|
-- owner
|
||||||
|
local formspec = (
|
||||||
|
"size[11,10;]"..
|
||||||
|
"label[0.3,0.1;Title:]"..
|
||||||
|
"field[1.5,0.5;3,0.5;title;;"..title.."]"..
|
||||||
|
"field_close_on_enter[title;false]"..
|
||||||
|
"button[4.1,0.2;1,0.5;set_title;Set]"..
|
||||||
|
"label[0,0.7;You need:]"..
|
||||||
|
"list["..name..";cust_ow;0,1.2;2,2;]"..
|
||||||
|
"label[3,0.7;You give:]"..
|
||||||
|
"list["..name..";cust_og;3,1.2;2,2;]"..
|
||||||
|
"label[0,3.5;Ejected items: (Remove me!)]"..
|
||||||
|
"list["..name..";custm_ej;0,4;4,1;]"..
|
||||||
|
"label[6,0;You are viewing:]"..
|
||||||
|
"label[6,0.3;(Click to switch)]"..
|
||||||
|
listring("custm_ej")
|
||||||
|
)
|
||||||
|
|
||||||
|
if mode == "owner_custm" then
|
||||||
|
formspec = (formspec..
|
||||||
|
"button[8.5,0.2;2.5,0.5;view_stock;Customers stock]"..
|
||||||
|
"list["..name..";custm;6,1;5,4;]"..
|
||||||
|
listring("custm"))
|
||||||
|
else
|
||||||
|
formspec = (formspec..
|
||||||
|
"button[8.5,0.2;2.5,0.5;view_custm;Your stock]"..
|
||||||
|
"list["..name..";stock;6,1;5,4;]"..
|
||||||
|
listring("stock"))
|
||||||
|
end
|
||||||
|
return (formspec..
|
||||||
|
"label[1,5;Use (E) + (Right click) for customer interface]"..
|
||||||
|
"list[current_player;main;1,6;8,4;]")
|
||||||
|
end
|
||||||
|
return ""
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
minetest.register_on_player_receive_fields(function(sender, formname, fields)
|
||||||
|
if formname ~= "exchange_shop:shop_formspec" then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local player_name = sender:get_player_name()
|
||||||
|
local pos = shop_positions[player_name]
|
||||||
|
if not pos then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
if (fields.quit and fields.quit ~= "") or
|
||||||
|
minetest.get_node(pos).name ~= "exchange_shop:shop" then
|
||||||
|
shop_positions[player_name] = nil
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local title = meta:get_string("title")
|
||||||
|
local shop_owner = meta:get_string("owner")
|
||||||
|
|
||||||
|
if fields.title and exchange_shop.has_access(meta, player_name) then
|
||||||
|
-- Limit title length
|
||||||
|
fields.title = fields.title:sub(1, 80)
|
||||||
|
if title ~= fields.title then
|
||||||
|
if fields.title ~= "" then
|
||||||
|
meta:set_string("infotext", "'" .. fields.title
|
||||||
|
.. "' (owned by " .. shop_owner .. ")")
|
||||||
|
else
|
||||||
|
meta:set_string("infotext", "Exchange shop (owned by "
|
||||||
|
.. shop_owner ..")")
|
||||||
|
end
|
||||||
|
meta:set_string("title", minetest.formspec_escape(fields.title))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if fields.exchange then
|
||||||
|
local shop_inv = meta:get_inventory()
|
||||||
|
local player_inv = sender:get_inventory()
|
||||||
|
if shop_inv:is_empty("cust_ow")
|
||||||
|
and shop_inv:is_empty("cust_og") then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local err_msg = exchange_shop.exchange_action(player_inv, shop_inv)
|
||||||
|
-- Throw error message
|
||||||
|
if err_msg then
|
||||||
|
minetest.chat_send_player(player_name, minetest.colorize("#F33",
|
||||||
|
"Exchange shop: " .. err_msg))
|
||||||
|
end
|
||||||
|
elseif exchange_shop.has_access(meta, player_name) then
|
||||||
|
local mode
|
||||||
|
if fields.view_custm then
|
||||||
|
mode = "owner_custm"
|
||||||
|
elseif fields.view_stock then
|
||||||
|
mode = "owner_stock"
|
||||||
|
else
|
||||||
|
return
|
||||||
|
end
|
||||||
|
minetest.show_formspec(player_name, "exchange_shop:shop_formspec",
|
||||||
|
get_exchange_shop_formspec(mode, pos, title))
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
|
minetest.register_node(exchange_shop.shopname, {
|
||||||
|
description = "Exchange Shop",
|
||||||
|
tiles = {
|
||||||
|
"shop_top.png", "shop_top.png",
|
||||||
|
"shop_side.png","shop_side.png",
|
||||||
|
"shop_side.png", "shop_front.png"
|
||||||
|
},
|
||||||
|
paramtype2 = "facedir",
|
||||||
|
groups = {choppy=2, oddly_breakable_by_hand=2,
|
||||||
|
tubedevice=1, tubedevice_receiver=1},
|
||||||
|
tube = {
|
||||||
|
insert_object = function(pos, node, stack, direction)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local inv = meta:get_inventory()
|
||||||
|
return inv:add_item("stock", stack)
|
||||||
|
end,
|
||||||
|
can_insert = function(pos, node, stack, direction)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local inv = meta:get_inventory()
|
||||||
|
return inv:room_for_item("stock", stack)
|
||||||
|
end,
|
||||||
|
input_inventory = "custm",
|
||||||
|
connect_sides = {left=1, right=1, back=1, top=1, bottom=1}
|
||||||
|
},
|
||||||
|
sounds = default.node_sound_wood_defaults(),
|
||||||
|
after_place_node = function(pos, placer)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local owner = placer:get_player_name()
|
||||||
|
meta:set_string("owner", owner)
|
||||||
|
meta:set_string("infotext", "Exchange shop (owned by "
|
||||||
|
.. owner .. ")")
|
||||||
|
end,
|
||||||
|
on_construct = function(pos)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
meta:set_string("infotext", "Exchange shop (constructing)")
|
||||||
|
meta:set_string("owner", "")
|
||||||
|
local inv = meta:get_inventory()
|
||||||
|
inv:set_size("stock", exchange_shop.storage_size) -- needed stock for exchanges
|
||||||
|
inv:set_size("custm", exchange_shop.storage_size) -- stock of the customers exchanges
|
||||||
|
inv:set_size("custm_ej", 4) -- ejected items if shop has no inventory room
|
||||||
|
inv:set_size("cust_ow", 2*2) -- owner wants
|
||||||
|
inv:set_size("cust_og", 2*2) -- owner gives
|
||||||
|
inv:set_size("cust_ej", 4) -- ejected items if player has no inventory room
|
||||||
|
end,
|
||||||
|
can_dig = function(pos,player)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local inv = meta:get_inventory()
|
||||||
|
if inv:is_empty("stock") and inv:is_empty("custm")
|
||||||
|
and inv:is_empty("cust_ow") and inv:is_empty("custm_ej")
|
||||||
|
and inv:is_empty("cust_og") and inv:is_empty("cust_ej") then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
minetest.chat_send_player(player:get_player_name(),
|
||||||
|
"Cannot dig exchange shop: one or multiple stocks are in use.")
|
||||||
|
return false
|
||||||
|
end,
|
||||||
|
on_rightclick = function(pos, node, clicker, itemstack)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
local player_name = clicker:get_player_name()
|
||||||
|
|
||||||
|
local mode = "customer"
|
||||||
|
if exchange_shop.has_access(meta, player_name) and
|
||||||
|
not clicker:get_player_control().aux1 then
|
||||||
|
mode = "owner_custm"
|
||||||
|
end
|
||||||
|
shop_positions[player_name] = pos
|
||||||
|
minetest.show_formspec(player_name, "exchange_shop:shop_formspec",
|
||||||
|
get_exchange_shop_formspec(mode, pos, meta:get_string("title")))
|
||||||
|
end,
|
||||||
|
allow_metadata_inventory_move = function(pos, from_list, from_index, to_list, to_index, count, player)
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
if exchange_shop.has_access(meta, player:get_player_name()) then
|
||||||
|
return count
|
||||||
|
end
|
||||||
|
return 0
|
||||||
|
end,
|
||||||
|
allow_metadata_inventory_put = function(pos, listname, index, stack, player)
|
||||||
|
if player:get_player_name() == ":pipeworks" then
|
||||||
|
return stack:get_count()
|
||||||
|
end
|
||||||
|
if listname == "custm" then
|
||||||
|
minetest.chat_send_player(player:get_player_name(),
|
||||||
|
"Exchange shop: Please press 'Customers stock' and insert your items there.")
|
||||||
|
return 0
|
||||||
|
end
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
if exchange_shop.has_access(meta, player:get_player_name())
|
||||||
|
and listname ~= "cust_ej"
|
||||||
|
and listname ~= "custm_ej" then
|
||||||
|
return stack:get_count()
|
||||||
|
end
|
||||||
|
return 0
|
||||||
|
end,
|
||||||
|
allow_metadata_inventory_take = function(pos, listname, index, stack, player)
|
||||||
|
if player:get_player_name() == ":pipeworks" then
|
||||||
|
return stack:get_count()
|
||||||
|
end
|
||||||
|
local meta = minetest.get_meta(pos)
|
||||||
|
if exchange_shop.has_access(meta, player:get_player_name())
|
||||||
|
or listname == "cust_ej" then
|
||||||
|
return stack:get_count()
|
||||||
|
end
|
||||||
|
return 0
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
|
minetest.register_craft({
|
||||||
|
output = "exchange_shop:shop",
|
||||||
|
recipe = {
|
||||||
|
{"default:sign_wall"},
|
||||||
|
{"default:chest_locked"},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
minetest.register_on_leaveplayer(function(player)
|
||||||
|
shop_positions[player:get_player_name()] = nil
|
||||||
|
end)
|
||||||
|
|
||||||
|
if minetest.get_modpath("wrench") and wrench then
|
||||||
|
local STRING = wrench.META_TYPE_STRING
|
||||||
|
wrench:register_node("exchange_shop:shop", {
|
||||||
|
lists = {"stock", "custm", "custm_ej", "cust_ow", "cust_og", "cust_ej"},
|
||||||
|
metas = {
|
||||||
|
owner = STRING,
|
||||||
|
infotext = STRING,
|
||||||
|
title = STRING,
|
||||||
|
},
|
||||||
|
owned = true
|
||||||
|
})
|
||||||
|
end
|
162
shop_functions.lua
Normal file
162
shop_functions.lua
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
function exchange_shop.has_access(meta, player_name)
|
||||||
|
local owner = meta:get_string("owner")
|
||||||
|
if player_name == owner or owner == "" then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
local privs = minetest.get_player_privs(player_name)
|
||||||
|
return privs.server or privs.protection_bypass
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- Tool wear aware replacement for contains_item.
|
||||||
|
function exchange_shop.list_contains_item(inv, listname, stack)
|
||||||
|
local count = stack:get_count()
|
||||||
|
if count == 0 then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
local list = inv:get_list(listname)
|
||||||
|
local name = stack:get_name()
|
||||||
|
local wear = stack:get_wear()
|
||||||
|
for _, list_stack in pairs(list) do
|
||||||
|
if list_stack:get_name() == name and
|
||||||
|
list_stack:get_wear() <= wear then
|
||||||
|
if list_stack:get_count() >= count then
|
||||||
|
return true
|
||||||
|
else
|
||||||
|
count = count - list_stack:get_count()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Tool wear aware replacement for remove_item.
|
||||||
|
function exchange_shop.list_remove_item(inv, listname, stack)
|
||||||
|
local wanted = stack:get_count()
|
||||||
|
if wanted == 0 then
|
||||||
|
return stack
|
||||||
|
end
|
||||||
|
|
||||||
|
local list = inv:get_list(listname)
|
||||||
|
local name = stack:get_name()
|
||||||
|
local wear = stack:get_wear()
|
||||||
|
local remaining = wanted
|
||||||
|
local removed_wear = 0
|
||||||
|
|
||||||
|
for index, list_stack in pairs(list) do
|
||||||
|
if list_stack:get_name() == name and
|
||||||
|
list_stack:get_wear() <= wear then
|
||||||
|
local taken_stack = list_stack:take_item(remaining)
|
||||||
|
inv:set_stack(listname, index, list_stack)
|
||||||
|
|
||||||
|
removed_wear = math.max(removed_wear, taken_stack:get_wear())
|
||||||
|
remaining = remaining - taken_stack:get_count()
|
||||||
|
if remaining == 0 then
|
||||||
|
break
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Todo: Also remove kebab
|
||||||
|
local removed_stack = ItemStack(name)
|
||||||
|
removed_stack:set_count(wanted - remaining)
|
||||||
|
removed_stack:set_wear(removed_wear)
|
||||||
|
return removed_stack
|
||||||
|
end
|
||||||
|
|
||||||
|
function exchange_shop.exchange_action(player_inv, shop_inv)
|
||||||
|
if not shop_inv:is_empty("cust_ej")
|
||||||
|
or not shop_inv:is_empty("custm_ej") then
|
||||||
|
return "One or multiple ejection fields are filled. "..
|
||||||
|
"Please empty them or contact the shop owner."
|
||||||
|
end
|
||||||
|
local owner_wants = shop_inv:get_list("cust_ow")
|
||||||
|
local owner_gives = shop_inv:get_list("cust_og")
|
||||||
|
|
||||||
|
-- Check validness of stack "owner wants"
|
||||||
|
for i1, item1 in pairs(owner_wants) do
|
||||||
|
local name1 = item1:get_name()
|
||||||
|
for i2, item2 in pairs(owner_wants) do
|
||||||
|
if name1 == "" then
|
||||||
|
break
|
||||||
|
end
|
||||||
|
if i1 ~= i2 and name1 == item2:get_name() then
|
||||||
|
return "The field 'Owner needs' can not contain multiple "..
|
||||||
|
"times the same items. Please contact the shop owner."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Check validness of stack "owner gives"
|
||||||
|
for i1, item1 in pairs(owner_gives) do
|
||||||
|
local name1 = item1:get_name()
|
||||||
|
for i2, item2 in pairs(owner_gives) do
|
||||||
|
if name1 == "" then
|
||||||
|
break
|
||||||
|
end
|
||||||
|
if i1 ~= i2 and name1 == item2:get_name() then
|
||||||
|
return "The field 'Owner gives' can not contain multiple "..
|
||||||
|
"times the same items. Please contact the shop owner."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Check for space in the shop
|
||||||
|
for i, item in pairs(owner_wants) do
|
||||||
|
if not shop_inv:room_for_item("custm", item) then
|
||||||
|
return "The stock in this shop is full. "..
|
||||||
|
"Please contact the shop owner."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local list_contains_item = exchange_shop.list_contains_item
|
||||||
|
|
||||||
|
-- Check availability of the shop's items
|
||||||
|
for i, item in pairs(owner_gives) do
|
||||||
|
if not list_contains_item(shop_inv, "stock", item) then
|
||||||
|
return "This shop is sold out."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Check for space in the player's inventory
|
||||||
|
for i, item in pairs(owner_gives) do
|
||||||
|
if not player_inv:room_for_item("main", item) then
|
||||||
|
return "You do not have enough space in your inventory."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Check availability of the player's items
|
||||||
|
for i, item in pairs(owner_wants) do
|
||||||
|
if not list_contains_item(player_inv, "main", item) then
|
||||||
|
return "You do not have the required items."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local list_remove_item = exchange_shop.list_remove_item
|
||||||
|
|
||||||
|
-- Conditions are ok: (try to) exchange now
|
||||||
|
local fully_exchanged = true
|
||||||
|
for i, item in pairs(owner_wants) do
|
||||||
|
local stack = list_remove_item(player_inv, "main", item)
|
||||||
|
if shop_inv:room_for_item("custm", stack) then
|
||||||
|
shop_inv:add_item("custm", stack)
|
||||||
|
else
|
||||||
|
-- Move to ejection field
|
||||||
|
shop_inv:add_item("custm_ej", stack)
|
||||||
|
fully_exchanged = false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
for i, item in pairs(owner_gives) do
|
||||||
|
local stack = list_remove_item(shop_inv, "stock", item)
|
||||||
|
if player_inv:room_for_item("main", stack) then
|
||||||
|
player_inv:add_item("main", stack)
|
||||||
|
else
|
||||||
|
-- Move to ejection field
|
||||||
|
shop_inv:add_item("cust_ej", stack)
|
||||||
|
fully_exchanged = false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if not fully_exchanged then
|
||||||
|
return "Warning! Stacks are overflowing somewhere!"
|
||||||
|
end
|
||||||
|
end
|
BIN
textures/shop_front.png
Normal file
BIN
textures/shop_front.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 738 B |
BIN
textures/shop_side.png
Normal file
BIN
textures/shop_side.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 672 B |
BIN
textures/shop_top.png
Normal file
BIN
textures/shop_top.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 652 B |
Loading…
Reference in New Issue
Block a user