432 lines
14 KiB
Lua
432 lines
14 KiB
Lua
local S = minetest.get_translator("unified_inventory")
|
|
local NS = function(s) return s end
|
|
local F = minetest.formspec_escape
|
|
local ui = unified_inventory
|
|
|
|
minetest.register_privilege("creative", {
|
|
description = S("Can use the creative inventory"),
|
|
give_to_singleplayer = false,
|
|
})
|
|
|
|
minetest.register_privilege("ui_full", {
|
|
description = S("Forces Unified Inventory to be displayed in Full mode if Lite mode is configured globally"),
|
|
give_to_singleplayer = false,
|
|
})
|
|
|
|
local trash = minetest.create_detached_inventory("trash", {
|
|
--allow_put = function(inv, listname, index, stack, player)
|
|
-- if ui.is_creative(player:get_player_name()) then
|
|
-- return stack:get_count()
|
|
-- else
|
|
-- return 0
|
|
-- end
|
|
--end,
|
|
on_put = function(inv, listname, index, stack, player)
|
|
inv:set_stack(listname, index, nil)
|
|
local player_name = player:get_player_name()
|
|
minetest.sound_play("trash", {to_player=player_name, gain = 1.0})
|
|
end,
|
|
})
|
|
trash:set_size("main", 1)
|
|
|
|
-- ui.register_button("craft", {
|
|
-- type = "image",
|
|
-- image = "ui_craft_icon.png",
|
|
-- tooltip = S("Crafting Grid")
|
|
-- })
|
|
|
|
ui.register_button("clear_inv", {
|
|
type = "image",
|
|
image = "ui_trash_icon.png",
|
|
tooltip = S("Clear inventory"),
|
|
action = function(player)
|
|
local player_name = player:get_player_name()
|
|
if not ui.is_creative(player_name) then
|
|
minetest.chat_send_player(player_name,
|
|
S("This button has been disabled outside"
|
|
.." of creative mode to prevent"
|
|
.." accidental inventory trashing."
|
|
.."\nUse the trash slot instead."))
|
|
ui.set_inventory_formspec(player, ui.current_page[player_name])
|
|
return
|
|
end
|
|
player:get_inventory():set_list("main", {})
|
|
minetest.chat_send_player(player_name, S('Inventory cleared!'))
|
|
minetest.sound_play("trash_all",
|
|
{to_player=player_name, gain = 1.0})
|
|
end,
|
|
condition = function(player)
|
|
return ui.is_creative(player:get_player_name())
|
|
end,
|
|
})
|
|
|
|
ui.register_page("craft", {
|
|
get_formspec = function(player, perplayer_formspec)
|
|
|
|
local formheaderx = perplayer_formspec.form_header_x
|
|
local formheadery = perplayer_formspec.form_header_y
|
|
local craftx = perplayer_formspec.craft_x
|
|
local crafty = perplayer_formspec.craft_y
|
|
|
|
local player_name = player:get_player_name()
|
|
local formspec = {
|
|
perplayer_formspec.standard_inv_bg,
|
|
}
|
|
local formspec2 = {
|
|
perplayer_formspec.craft_grid,
|
|
"label["..formheaderx..","..formheadery..";" ..F(S("Crafting")).."]",
|
|
"listcolors[#00000000;#00000000]",
|
|
"listring[current_name;craft]",
|
|
"listring[current_player;main]",
|
|
string.format("label[%f,%f;%s]", craftx + 6.45, crafty + 2.4, F(S("Trash:"))),
|
|
ui.make_trash_slot(craftx + 6.25, crafty + 2.5),
|
|
ui.single_slot(craftx - 2.5, crafty + 2.5),
|
|
string.format("label[%f,%f;%s]", craftx - 2.3, crafty + 2.4,F(S("Refill:"))),
|
|
string.format("list[detached:%srefill;main;%f,%f;1,1;]", F(player_name), craftx - 2.5 + ui.list_img_offset, crafty + 2.5 + ui.list_img_offset),
|
|
}
|
|
local n=#formspec+1
|
|
|
|
if ui.is_creative(player_name) then
|
|
--copy formspec2 to formspec
|
|
for i=1,#formspec2 do
|
|
formspec[n]=formspec2[i]
|
|
n = n+1
|
|
end
|
|
end
|
|
return {formspec=table.concat(formspec)}
|
|
end,
|
|
})
|
|
|
|
-- stack_image_button(): generate a form button displaying a stack of items
|
|
--
|
|
-- The specified item may be a group. In that case, the group will be
|
|
-- represented by some item in the group, along with a flag indicating
|
|
-- that it's a group. If the group contains only one item, it will be
|
|
-- treated as if that item had been specified directly.
|
|
|
|
local function stack_image_button(x, y, w, h, buttonname_prefix, item)
|
|
local name = item:get_name()
|
|
local count = item:get_count()
|
|
local wear = item:get_wear()
|
|
local description = item:get_meta():get_string("description")
|
|
local show_is_group = false
|
|
local displayitem = name.." "..count.." "..wear
|
|
local selectitem = name
|
|
if name:sub(1, 6) == "group:" then
|
|
local group_name = name:sub(7)
|
|
local group_item = ui.get_group_item(group_name)
|
|
show_is_group = not group_item.sole
|
|
displayitem = group_item.item or "unknown"
|
|
selectitem = group_item.sole and displayitem or name
|
|
end
|
|
local label = show_is_group and "G" or ""
|
|
-- Unique id to prevent tooltip being overridden
|
|
local id = string.format("%i%i_", x*10, y*10)
|
|
local buttonname = F(id..buttonname_prefix..ui.mangle_for_formspec(selectitem))
|
|
local button = string.format("item_image_button[%f,%f;%f,%f;%s;%s;%s]",
|
|
x, y, w, h,
|
|
F(displayitem), buttonname, label)
|
|
if show_is_group then
|
|
local groupstring, andcount = ui.extract_groupnames(name)
|
|
local grouptip
|
|
if andcount == 1 then
|
|
grouptip = S("Any item belonging to the @1 group", groupstring)
|
|
elseif andcount > 1 then
|
|
grouptip = S("Any item belonging to the groups @1", groupstring)
|
|
end
|
|
grouptip = F(grouptip)
|
|
if andcount >= 1 then
|
|
button = button .. string.format("tooltip[%s;%s]", buttonname, grouptip)
|
|
end
|
|
elseif description ~= "" then
|
|
button = button .. string.format("tooltip[%s;%s]", buttonname, F(description))
|
|
end
|
|
return button
|
|
end
|
|
|
|
local recipe_text = {
|
|
recipe = NS("Recipe @1 of @2"),
|
|
usage = NS("Usage @1 of @2"),
|
|
}
|
|
local no_recipe_text = {
|
|
recipe = S("No recipes"),
|
|
usage = S("No usages"),
|
|
}
|
|
local role_text = {
|
|
recipe = S("Result"),
|
|
usage = S("Ingredient"),
|
|
}
|
|
local next_alt_text = {
|
|
recipe = S("Show next recipe"),
|
|
usage = S("Show next usage"),
|
|
}
|
|
local prev_alt_text = {
|
|
recipe = S("Show previous recipe"),
|
|
usage = S("Show previous usage"),
|
|
}
|
|
local other_dir = {
|
|
recipe = "usage",
|
|
usage = "recipe",
|
|
}
|
|
|
|
ui.register_page("craftguide", {
|
|
get_formspec = function(player, perplayer_formspec)
|
|
|
|
local craftguidex = perplayer_formspec.craft_guide_x
|
|
local craftguidey = perplayer_formspec.craft_guide_y
|
|
local craftguidearrowx = perplayer_formspec.craft_guide_arrow_x
|
|
local craftguideresultx = perplayer_formspec.craft_guide_result_x
|
|
local formheaderx = perplayer_formspec.form_header_x
|
|
local formheadery = perplayer_formspec.form_header_y
|
|
local give_x = perplayer_formspec.give_btn_x
|
|
|
|
local player_name = player:get_player_name()
|
|
local player_privs = minetest.get_player_privs(player_name)
|
|
|
|
local formspec = {
|
|
perplayer_formspec.standard_inv_bg,
|
|
"label["..formheaderx..","..formheadery..";" .. F(S("Crafting Guide")) .. "]",
|
|
"listcolors[#00000000;#00000000]"
|
|
}
|
|
|
|
local item_name = ui.current_item[player_name]
|
|
if not item_name then
|
|
return { formspec = table.concat(formspec) }
|
|
end
|
|
|
|
local n = 4
|
|
|
|
local item_name_shown
|
|
if minetest.registered_items[item_name]
|
|
and minetest.registered_items[item_name].description then
|
|
item_name_shown = S("@1 (@2)",
|
|
minetest.registered_items[item_name].description, item_name)
|
|
else
|
|
item_name_shown = item_name
|
|
end
|
|
|
|
local dir = ui.current_craft_direction[player_name]
|
|
local rdir = dir == "recipe" and "usage" or "recipe"
|
|
|
|
local crafts = ui.crafts_for[dir][item_name]
|
|
local alternate = ui.alternate[player_name]
|
|
local alternates, craft
|
|
if crafts and #crafts > 0 then
|
|
alternates = #crafts
|
|
craft = crafts[alternate]
|
|
end
|
|
local has_give = player_privs.give or ui.is_creative(player_name)
|
|
|
|
formspec[n] = string.format("image[%f,%f;%f,%f;ui_crafting_arrow.png]",
|
|
craftguidearrowx, craftguidey, ui.imgscale, ui.imgscale)
|
|
|
|
formspec[n+1] = string.format("textarea[%f,%f;10,1;;%s: %s;]",
|
|
perplayer_formspec.craft_guide_resultstr_x, perplayer_formspec.craft_guide_resultstr_y,
|
|
F(role_text[dir]), item_name_shown)
|
|
n = n + 2
|
|
|
|
local giveme_form = table.concat({
|
|
"label[".. (give_x+0.1)..",".. (craftguidey + 2.7) .. ";" .. F(S("Give me:")) .. "]",
|
|
"button["..(give_x)..",".. (craftguidey + 2.9) .. ";0.75,0.5;craftguide_giveme_1;1]",
|
|
"button["..(give_x+0.8)..",".. (craftguidey + 2.9) .. ";0.75,0.5;craftguide_giveme_10;10]",
|
|
"button["..(give_x+1.6)..",".. (craftguidey + 2.9) .. ";0.75,0.5;craftguide_giveme_99;99]"
|
|
})
|
|
|
|
if not craft then
|
|
-- No craft recipes available for this item.
|
|
formspec[n] = string.format("label[%f,%f;%s]", craftguidex+2.5, craftguidey+1.5, F(no_recipe_text[dir]))
|
|
local no_pos = dir == "recipe" and (craftguidex+2.5) or craftguideresultx
|
|
local item_pos = dir == "recipe" and craftguideresultx or (craftguidex+2.5)
|
|
formspec[n+1] = "image["..no_pos..","..craftguidey..";1.2,1.2;ui_no.png]"
|
|
formspec[n+2] = stack_image_button(item_pos, craftguidey, 1.2, 1.2,
|
|
"item_button_" .. other_dir[dir] .. "_", ItemStack(item_name))
|
|
if has_give then
|
|
formspec[n+3] = giveme_form
|
|
end
|
|
return { formspec = table.concat(formspec) }
|
|
else
|
|
formspec[n] = stack_image_button(craftguideresultx, craftguidey, 1.2, 1.2,
|
|
"item_button_" .. rdir .. "_", ItemStack(craft.output))
|
|
n = n + 1
|
|
end
|
|
|
|
local craft_type = ui.registered_craft_types[craft.type] or
|
|
ui.craft_type_defaults(craft.type, {})
|
|
if craft_type.icon then
|
|
formspec[n] = string.format("image[%f,%f;%f,%f;%s]",
|
|
craftguidearrowx+0.35, craftguidey, 0.5, 0.5, craft_type.icon)
|
|
n = n + 1
|
|
end
|
|
formspec[n] = string.format("label[%f,%f;%s]", craftguidearrowx + 0.15, craftguidey + 1.4, F(craft_type.description))
|
|
n = n + 1
|
|
|
|
local display_size = craft_type.dynamic_display_size
|
|
and craft_type.dynamic_display_size(craft)
|
|
or { width = craft_type.width, height = craft_type.height }
|
|
local craft_width = craft_type.get_shaped_craft_width
|
|
and craft_type.get_shaped_craft_width(craft)
|
|
or display_size.width
|
|
|
|
-- This keeps recipes aligned to the right,
|
|
-- so that they're close to the arrow.
|
|
local xoffset = craftguidex+3.75
|
|
local bspc = 1.25
|
|
-- Offset factor for crafting grids with side length > 4
|
|
local of = (3/math.max(3, math.max(display_size.width, display_size.height)))
|
|
local od = 0
|
|
-- Minimum grid size at which size optimization measures kick in
|
|
local mini_craft_size = 6
|
|
if display_size.width >= mini_craft_size then
|
|
od = math.max(1, display_size.width - 2)
|
|
xoffset = xoffset - 0.1
|
|
end
|
|
-- Size modifier factor
|
|
local sf = math.min(1, of * (1.05 + 0.05*od))
|
|
-- Button size
|
|
local bsize = 1.2 * sf
|
|
|
|
if display_size.width >= mini_craft_size then -- it's not a normal 3x3 grid
|
|
bsize = 0.8 * sf
|
|
end
|
|
if (bsize > 0.35 and display_size.width) then
|
|
for y = 1, display_size.height do
|
|
for x = 1, display_size.width do
|
|
local item
|
|
if craft and x <= craft_width then
|
|
item = craft.items[(y-1) * craft_width + x]
|
|
end
|
|
-- Flipped x, used to build formspec buttons from right to left
|
|
local fx = display_size.width - (x-1)
|
|
-- x offset, y offset
|
|
local xof = ((fx-1) * of + of) * bspc
|
|
local yof = ((y-1) * of + 1) * bspc
|
|
if item then
|
|
formspec[n] = stack_image_button(
|
|
xoffset - xof, craftguidey - 1.25 + yof, bsize, bsize,
|
|
"item_button_recipe_",
|
|
ItemStack(item))
|
|
else
|
|
-- Fake buttons just to make grid
|
|
formspec[n] = string.format("image_button[%f,%f;%f,%f;ui_blank_image.png;;]",
|
|
xoffset - xof, craftguidey - 1.25 + yof, bsize, bsize)
|
|
end
|
|
n = n + 1
|
|
end
|
|
end
|
|
else
|
|
-- Error
|
|
formspec[n] = string.format("label[2,%f;%s]",
|
|
craftguidey, F(S("This recipe is too@nlarge to be displayed.")))
|
|
n = n + 1
|
|
end
|
|
|
|
if craft_type.uses_crafting_grid and display_size.width <= 3 then
|
|
formspec[n] = "label["..(give_x+0.1)..",".. (craftguidey + 1.7) .. ";" .. F(S("To craft grid:")) .. "]"
|
|
formspec[n+1] = "button[".. (give_x)..",".. (craftguidey + 1.9) .. ";0.75,0.5;craftguide_craft_1;1]"
|
|
formspec[n+2] = "button[".. (give_x+0.8)..",".. (craftguidey + 1.9) .. ";0.75,0.5;craftguide_craft_10;10]"
|
|
formspec[n+3] = "button[".. (give_x+1.6)..",".. (craftguidey + 1.9) .. ";0.75,0.5;craftguide_craft_max;" .. F(S("All")) .. "]"
|
|
n = n + 4
|
|
end
|
|
|
|
if has_give then
|
|
formspec[n] = giveme_form
|
|
n = n + 1
|
|
end
|
|
|
|
if alternates and alternates > 1 then
|
|
formspec[n] = string.format("label[%f,%f;%s]",
|
|
craftguidex+4, craftguidey + 2.3, F(S(recipe_text[dir], alternate, alternates)))
|
|
formspec[n+1] = string.format("image_button[%f,%f;1.1,1.1;ui_left_icon.png;alternate_prev;]",
|
|
craftguidearrowx+0.2, craftguidey + 2.6)
|
|
formspec[n+2] = string.format("image_button[%f,%f;1.1,1.1;ui_right_icon.png;alternate;]",
|
|
craftguidearrowx+1.35, craftguidey + 2.6)
|
|
formspec[n+3] = "tooltip[alternate_prev;" .. F(prev_alt_text[dir]) .. "]"
|
|
formspec[n+4] = "tooltip[alternate;" .. F(next_alt_text[dir]) .. "]"
|
|
end
|
|
|
|
return { formspec = table.concat(formspec) }
|
|
end,
|
|
})
|
|
|
|
local function craftguide_giveme(player, formname, fields)
|
|
local player_name = player:get_player_name()
|
|
local player_privs = minetest.get_player_privs(player_name)
|
|
if not player_privs.give and
|
|
not ui.is_creative(player_name) then
|
|
minetest.log("action", "[unified_inventory] Denied give action to player " ..
|
|
player_name)
|
|
return
|
|
end
|
|
|
|
local amount
|
|
for k, v in pairs(fields) do
|
|
amount = k:match("craftguide_giveme_(.*)")
|
|
if amount then break end
|
|
end
|
|
|
|
amount = tonumber(amount) or 0
|
|
if amount == 0 then return end
|
|
|
|
local output = ui.current_item[player_name]
|
|
if (not output) or (output == "") then return end
|
|
|
|
local player_inv = player:get_inventory()
|
|
|
|
player_inv:add_item("main", {name = output, count = amount})
|
|
end
|
|
|
|
local function craftguide_craft(player, formname, fields)
|
|
local amount
|
|
for k, v in pairs(fields) do
|
|
amount = k:match("craftguide_craft_(.*)")
|
|
if amount then break end
|
|
end
|
|
if not amount then return end
|
|
|
|
amount = tonumber(amount) or -1 -- fallback for "all"
|
|
if amount == 0 or amount < -1 or amount > 99 then return end
|
|
|
|
local player_name = player:get_player_name()
|
|
|
|
local output = ui.current_item[player_name] or ""
|
|
if output == "" then return end
|
|
|
|
local crafts = ui.crafts_for[
|
|
ui.current_craft_direction[player_name]][output] or {}
|
|
if #crafts == 0 then return end
|
|
|
|
local alternate = ui.alternate[player_name]
|
|
|
|
local craft = crafts[alternate]
|
|
if not craft.width then
|
|
if not craft.output then
|
|
minetest.log("warning", "[unified_inventory] Craft has no output.")
|
|
else
|
|
minetest.log("warning", ("[unified_inventory] Craft for '%s' has no width."):format(craft.output))
|
|
end
|
|
return
|
|
end
|
|
if craft.width > 3 then return end
|
|
|
|
ui.craftguide_match_craft(player, "main", "craft", craft, amount)
|
|
|
|
ui.set_inventory_formspec(player, "craft")
|
|
end
|
|
|
|
minetest.register_on_player_receive_fields(function(player, formname, fields)
|
|
if formname ~= "" then
|
|
return
|
|
end
|
|
|
|
for k, v in pairs(fields) do
|
|
if k:match("craftguide_craft_") then
|
|
craftguide_craft(player, formname, fields)
|
|
return
|
|
end
|
|
if k:match("craftguide_giveme_") then
|
|
craftguide_giveme(player, formname, fields)
|
|
return
|
|
end
|
|
end
|
|
end)
|