2019-09-19 14:00:26 +02:00
|
|
|
local S = minetest.get_translator("unified_inventory")
|
2018-04-02 13:33:36 +02:00
|
|
|
local F = minetest.formspec_escape
|
2021-03-07 14:18:17 +01:00
|
|
|
local ui = unified_inventory
|
2014-06-21 12:44:31 +02:00
|
|
|
|
2014-06-13 11:40:52 +02:00
|
|
|
-- This pair of encoding functions is used where variable text must go in
|
|
|
|
-- button names, where the text might contain formspec metacharacters.
|
|
|
|
-- We can escape button names for the formspec, to avoid screwing up
|
|
|
|
-- form structure overall, but they then don't get de-escaped, and so
|
|
|
|
-- the input we get back from the button contains the formspec escaping.
|
|
|
|
-- This is a game engine bug, and in the anticipation that it might be
|
|
|
|
-- fixed some day we don't want to rely on it. So for safety we apply
|
|
|
|
-- an encoding that avoids all formspec metacharacters.
|
2021-03-05 16:58:18 +01:00
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.mangle_for_formspec(str)
|
2014-06-13 11:40:52 +02:00
|
|
|
return string.gsub(str, "([^A-Za-z0-9])", function (c) return string.format("_%d_", string.byte(c)) end)
|
|
|
|
end
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.demangle_for_formspec(str)
|
2014-06-13 11:40:52 +02:00
|
|
|
return string.gsub(str, "_([0-9]+)_", function (v) return string.char(v) end)
|
|
|
|
end
|
2013-09-29 00:15:37 +02:00
|
|
|
|
2021-03-05 16:58:18 +01:00
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.get_per_player_formspec(player_name)
|
2021-03-08 10:51:54 +01:00
|
|
|
local draw_lite_mode = ui.lite_mode and not minetest.check_player_privs(player_name, {ui_full=true})
|
2015-10-05 10:24:01 +02:00
|
|
|
|
2021-03-08 10:56:52 +01:00
|
|
|
return table.copy(draw_lite_mode and ui.style_lite or ui.style_full), draw_lite_mode
|
2015-10-05 10:24:01 +02:00
|
|
|
end
|
|
|
|
|
2021-04-05 20:07:14 +02:00
|
|
|
local function formspec_button(ui_peruser, name, image, offset, pos, scale, label)
|
|
|
|
local element = 'image_button'
|
|
|
|
if minetest.registered_items[image] then
|
|
|
|
element = 'item_image_button'
|
|
|
|
end
|
|
|
|
local spc = (1-scale)*ui_peruser.btn_size/2
|
|
|
|
local size = ui_peruser.btn_size*scale
|
|
|
|
return string.format("%s[%f,%f;%f,%f;%s;%s;]", element,
|
|
|
|
(offset.x or offset[1]) + ( ui_peruser.btn_spc * (pos.x or pos[1]) ) + spc,
|
|
|
|
(offset.y or offset[2]) + ( ui_peruser.btn_spc * (pos.y or pos[2]) ) + spc,
|
|
|
|
size, size, image, name) ..
|
|
|
|
string.format("tooltip[%s;%s]", name, F(label or name))
|
|
|
|
end
|
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.get_formspec(player, page)
|
2015-10-05 10:24:01 +02:00
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
if not player then
|
|
|
|
return ""
|
|
|
|
end
|
2015-10-05 10:24:01 +02:00
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
local player_name = player:get_player_name()
|
2021-03-07 14:18:17 +01:00
|
|
|
local ui_peruser,draw_lite_mode = ui.get_per_player_formspec(player_name)
|
2015-10-05 10:24:01 +02:00
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
ui.current_page[player_name] = page
|
|
|
|
local pagedef = ui.pages[page]
|
2013-09-29 00:15:37 +02:00
|
|
|
|
2019-03-31 12:19:08 +02:00
|
|
|
if not pagedef then
|
|
|
|
return "" -- Invalid page name
|
|
|
|
end
|
2021-05-25 19:31:34 +02:00
|
|
|
|
2015-08-16 16:40:49 +02:00
|
|
|
local formspec = {
|
2021-03-31 06:52:30 +02:00
|
|
|
"formspec_version[4]",
|
|
|
|
"size["..ui_peruser.formw..","..ui_peruser.formh.."]",
|
2019-03-31 12:19:08 +02:00
|
|
|
pagedef.formspec_prepend and "" or "no_prepend[]",
|
2021-03-31 06:52:30 +02:00
|
|
|
ui.standard_background
|
2015-08-16 16:40:49 +02:00
|
|
|
}
|
2015-06-28 09:47:03 +02:00
|
|
|
|
2021-03-31 06:52:30 +02:00
|
|
|
local n = 5
|
2015-06-28 09:47:03 +02:00
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
local perplayer_formspec = ui.get_per_player_formspec(player_name)
|
2015-10-05 10:24:01 +02:00
|
|
|
local fsdata = pagedef.get_formspec(player, perplayer_formspec)
|
|
|
|
|
2015-08-16 16:40:49 +02:00
|
|
|
formspec[n] = fsdata.formspec
|
|
|
|
n = n+1
|
2013-09-29 00:15:37 +02:00
|
|
|
|
2015-06-28 09:47:03 +02:00
|
|
|
local button_row = 0
|
|
|
|
local button_col = 0
|
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
-- Main buttons
|
2015-10-05 10:24:01 +02:00
|
|
|
|
|
|
|
local filtered_inv_buttons = {}
|
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
for i, def in pairs(ui.buttons) do
|
2018-04-02 13:33:36 +02:00
|
|
|
if not (draw_lite_mode and def.hide_lite) then
|
2015-10-05 10:24:01 +02:00
|
|
|
table.insert(filtered_inv_buttons, def)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
for i, def in pairs(filtered_inv_buttons) do
|
2015-06-28 09:47:03 +02:00
|
|
|
|
2015-10-05 10:24:01 +02:00
|
|
|
if draw_lite_mode and i > 4 then
|
2015-06-28 09:47:03 +02:00
|
|
|
button_row = 1
|
|
|
|
button_col = 1
|
|
|
|
end
|
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
if def.type == "image" then
|
2016-11-06 02:28:45 +01:00
|
|
|
if (def.condition == nil or def.condition(player) == true) then
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
formspec[n] = string.format("image_button[%f,%f;%f,%f;%s;%s;]",
|
|
|
|
ui_peruser.main_button_x + ui_peruser.btn_spc * (i - 1) - button_col * ui_peruser.btn_spc * 4,
|
|
|
|
ui_peruser.main_button_y + button_row * ui_peruser.btn_spc,
|
|
|
|
ui_peruser.btn_size,ui_peruser.btn_size,
|
|
|
|
F(def.image),
|
|
|
|
F(def.name))
|
|
|
|
formspec[n+1] = "tooltip["..F(def.name)..";"..(def.tooltip or "").."]"
|
|
|
|
n = n+2
|
2016-11-06 02:28:45 +01:00
|
|
|
else
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
formspec[n] = string.format("image[%f,%f;%f,%f;%s^[colorize:#808080:alpha]",
|
|
|
|
ui_peruser.main_button_x + ui_peruser.btn_spc * (i - 1) - button_col * ui_peruser.btn_spc * 4,
|
|
|
|
ui_peruser.main_button_y + button_row * ui_peruser.btn_spc,
|
|
|
|
ui_peruser.btn_size,ui_peruser.btn_size,def.image)
|
|
|
|
n = n+1
|
2016-11-06 02:28:45 +01:00
|
|
|
end
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-10-03 05:25:07 +02:00
|
|
|
if fsdata.draw_inventory ~= false then
|
|
|
|
-- Player inventory
|
2015-08-16 16:40:49 +02:00
|
|
|
formspec[n] = "listcolors[#00000000;#00000000]"
|
2021-03-05 16:58:18 +01:00
|
|
|
formspec[n+1] = ui_peruser.standard_inv
|
2015-08-16 16:40:49 +02:00
|
|
|
n = n+2
|
2013-10-03 05:25:07 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
if fsdata.draw_item_list == false then
|
2015-08-16 16:40:49 +02:00
|
|
|
return table.concat(formspec, "")
|
2013-10-03 05:25:07 +02:00
|
|
|
end
|
|
|
|
|
2021-04-05 20:07:14 +02:00
|
|
|
-- Category filters
|
|
|
|
|
|
|
|
local categories_pos = { ui_peruser.page_x, ui_peruser.page_y-ui_peruser.btn_spc-0.5 }
|
|
|
|
local categories_scroll_pos = { ui_peruser.page_x, ui_peruser.form_header_y-(draw_lite_mode and 0 or 0.2) }
|
|
|
|
|
|
|
|
formspec[n] = string.format("background9[%f,%f;%f,%f;%s;false;3]",
|
|
|
|
ui_peruser.page_x-0.1, categories_scroll_pos[2],
|
|
|
|
(ui_peruser.btn_spc * ui_peruser.pagecols) + 0.13, 1.4+(draw_lite_mode and 0 or 0.2),
|
|
|
|
"ui_smallbg_9_sliced.png")
|
|
|
|
n = n + 1
|
|
|
|
|
|
|
|
formspec[n] = string.format("label[%f,%f;%s]", ui_peruser.page_x, ui_peruser.form_header_y+(draw_lite_mode and 0.3 or 0.2), "Category:")
|
|
|
|
n = n + 1
|
|
|
|
|
|
|
|
local scroll_offset = 0
|
|
|
|
local category_count = #unified_inventory.category_list
|
|
|
|
if category_count > ui_peruser.pagecols then
|
|
|
|
scroll_offset = unified_inventory.current_category_scroll[player_name]
|
|
|
|
end
|
|
|
|
|
|
|
|
for index, category in ipairs(unified_inventory.category_list) do
|
|
|
|
local column = index - scroll_offset
|
|
|
|
if column > 0 and column <= ui_peruser.pagecols then
|
|
|
|
local scale = 0.8
|
|
|
|
if unified_inventory.current_category[player_name] == category.name then
|
|
|
|
scale = 1
|
|
|
|
end
|
|
|
|
formspec[n] = formspec_button(ui_peruser, "category_"..category.name, category.symbol, categories_pos, {column-1, 0}, scale, category.label)
|
|
|
|
n = n + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if category_count > ui_peruser.pagecols and scroll_offset > 0 then
|
|
|
|
-- prev
|
|
|
|
formspec[n] = formspec_button(ui_peruser, "prev_category", "ui_left_icon.png", categories_scroll_pos, {ui_peruser.pagecols - 2, 0}, 0.8, S("Scroll categories left"))
|
|
|
|
n = n + 1
|
|
|
|
end
|
|
|
|
if category_count > ui_peruser.pagecols and category_count - scroll_offset > ui_peruser.pagecols then
|
|
|
|
-- next
|
|
|
|
formspec[n] = formspec_button(ui_peruser, "next_category", "ui_right_icon.png", categories_scroll_pos, {ui_peruser.pagecols - 1, 0}, 0.8, S("Scroll categories right"))
|
|
|
|
n = n + 1
|
|
|
|
end
|
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
-- Search box
|
2016-11-29 20:45:59 +01:00
|
|
|
formspec[n] = "field_close_on_enter[searchbox;false]"
|
2015-06-28 09:47:03 +02:00
|
|
|
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
formspec[n+1] = string.format("field[%f,%f;%f,%f;searchbox;;%s]",
|
|
|
|
ui_peruser.page_buttons_x, ui_peruser.page_buttons_y,
|
|
|
|
ui_peruser.searchwidth - 0.1, ui_peruser.btn_size,
|
|
|
|
F(ui.current_searchbox[player_name]))
|
|
|
|
formspec[n+2] = string.format("image_button[%f,%f;%f,%f;ui_search_icon.png;searchbutton;]",
|
|
|
|
ui_peruser.page_buttons_x + ui_peruser.searchwidth, ui_peruser.page_buttons_y,
|
|
|
|
ui_peruser.btn_size,ui_peruser.btn_size)
|
|
|
|
formspec[n+3] = "tooltip[searchbutton;" ..F(S("Search")) .. "]"
|
|
|
|
formspec[n+4] = string.format("image_button[%f,%f;%f,%f;ui_reset_icon.png;searchresetbutton;]",
|
|
|
|
ui_peruser.page_buttons_x + ui_peruser.searchwidth + ui_peruser.btn_spc,
|
|
|
|
ui_peruser.page_buttons_y,
|
|
|
|
ui_peruser.btn_size, ui_peruser.btn_size)
|
|
|
|
formspec[n+5] = "tooltip[searchresetbutton;"..F(S("Reset search and display everything")).."]"
|
|
|
|
|
|
|
|
n = n + 6
|
2021-03-05 16:58:18 +01:00
|
|
|
|
|
|
|
-- Controls to flip items pages
|
|
|
|
|
|
|
|
local btnlist = {
|
2021-03-09 20:26:36 +01:00
|
|
|
{ "ui_skip_backward_icon.png", "start_list", S("First page") },
|
|
|
|
{ "ui_doubleleft_icon.png", "rewind3", S("Back three pages") },
|
|
|
|
{ "ui_left_icon.png", "rewind1", S("Back one page") },
|
|
|
|
{ "ui_right_icon.png", "forward1", S("Forward one page") },
|
|
|
|
{ "ui_doubleright_icon.png", "forward3", S("Forward three pages") },
|
|
|
|
{ "ui_skip_forward_icon.png", "end_list", S("Last page") },
|
2021-03-05 16:58:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if draw_lite_mode then
|
|
|
|
btnlist[5] = nil
|
|
|
|
btnlist[2] = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
local bn = 0
|
|
|
|
for _, b in pairs(btnlist) do
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
formspec[n] = string.format("image_button[%f,%f;%f,%f;%s;%s;]",
|
|
|
|
ui_peruser.page_buttons_x + ui_peruser.btn_spc*bn,
|
|
|
|
ui_peruser.page_buttons_y + ui_peruser.btn_spc,
|
|
|
|
ui_peruser.btn_size, ui_peruser.btn_size,
|
|
|
|
b[1],b[2])
|
2021-03-09 20:26:36 +01:00
|
|
|
formspec[n+1] = "tooltip["..b[2]..";"..F(b[3]).."]"
|
2021-03-05 16:58:18 +01:00
|
|
|
bn = bn + 1
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
n = n + 2
|
2015-06-28 09:47:03 +02:00
|
|
|
end
|
|
|
|
|
2018-04-02 13:33:36 +02:00
|
|
|
local no_matches = S("No matching items")
|
2015-10-05 10:24:01 +02:00
|
|
|
if draw_lite_mode then
|
2018-04-02 13:33:36 +02:00
|
|
|
no_matches = S("No matches.")
|
2015-06-28 09:47:03 +02:00
|
|
|
end
|
2013-09-29 00:15:37 +02:00
|
|
|
|
|
|
|
-- Items list
|
2021-03-07 14:18:17 +01:00
|
|
|
if #ui.filtered_items_list[player_name] == 0 then
|
2021-03-05 16:58:18 +01:00
|
|
|
formspec[n] = "label["..ui_peruser.page_x..","..(ui_peruser.page_y+0.15)..";" .. F(no_matches) .. "]"
|
2014-04-29 21:08:04 +02:00
|
|
|
else
|
2021-03-07 14:18:17 +01:00
|
|
|
local dir = ui.active_search_direction[player_name]
|
|
|
|
local list_index = ui.current_index[player_name]
|
2020-03-25 18:46:35 +01:00
|
|
|
local page2 = math.floor(list_index / (ui_peruser.items_per_page) + 1)
|
2014-04-29 21:08:04 +02:00
|
|
|
local pagemax = math.floor(
|
2021-03-07 14:18:17 +01:00
|
|
|
(#ui.filtered_items_list[player_name] - 1)
|
2015-10-05 10:24:01 +02:00
|
|
|
/ (ui_peruser.items_per_page) + 1)
|
|
|
|
for y = 0, ui_peruser.pagerows - 1 do
|
|
|
|
for x = 0, ui_peruser.pagecols - 1 do
|
2021-03-07 14:18:17 +01:00
|
|
|
local name = ui.filtered_items_list[player_name][list_index]
|
2018-12-12 20:18:53 +01:00
|
|
|
local item = minetest.registered_items[name]
|
|
|
|
if item then
|
2016-08-07 02:14:46 +02:00
|
|
|
-- Clicked on current item: Flip crafting direction
|
2021-03-07 14:18:17 +01:00
|
|
|
if name == ui.current_item[player_name] then
|
|
|
|
local cdir = ui.current_craft_direction[player_name]
|
2016-08-07 02:14:46 +02:00
|
|
|
if cdir == "recipe" then
|
|
|
|
dir = "usage"
|
|
|
|
elseif cdir == "usage" then
|
|
|
|
dir = "recipe"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
-- Default: use active search direction by default
|
2021-03-07 14:18:17 +01:00
|
|
|
dir = ui.active_search_direction[player_name]
|
2016-08-07 02:14:46 +02:00
|
|
|
end
|
2018-12-12 20:18:53 +01:00
|
|
|
|
|
|
|
local button_name = "item_button_" .. dir .. "_"
|
2021-03-07 14:18:17 +01:00
|
|
|
.. ui.mangle_for_formspec(name)
|
2021-03-07 12:09:13 +01:00
|
|
|
formspec[n] = ("item_image_button[%f,%f;%f,%f;%s;%s;]"):format(
|
|
|
|
ui_peruser.page_x + x * ui_peruser.btn_spc,
|
|
|
|
ui_peruser.page_y + y * ui_peruser.btn_spc,
|
|
|
|
ui_peruser.btn_size, ui_peruser.btn_size,
|
2018-12-12 20:18:53 +01:00
|
|
|
name, button_name
|
|
|
|
)
|
|
|
|
formspec[n + 1] = ("tooltip[%s;%s \\[%s\\]]"):format(
|
|
|
|
button_name, minetest.formspec_escape(item.description),
|
|
|
|
item.mod_origin or "??"
|
|
|
|
)
|
|
|
|
n = n + 2
|
2015-06-28 09:47:03 +02:00
|
|
|
list_index = list_index + 1
|
|
|
|
end
|
2014-04-29 21:08:04 +02:00
|
|
|
end
|
|
|
|
end
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
formspec[n] = string.format("label[%f,%f;%s: %s]",
|
2021-04-05 20:07:14 +02:00
|
|
|
ui_peruser.page_buttons_x + ui_peruser.btn_spc * (draw_lite_mode and 1 or 2),
|
|
|
|
ui_peruser.page_buttons_y + 0.1 + ui_peruser.btn_spc * 2,
|
Multiple related changes to string handling
1) Convert most formspec elements to use string.format(), when the
result would be more readable, or less messy, or at least makes the line
shorter, assuming it looked like it really needed it to begin with.
2) Convert all long `foo..","..bar..";"..baz..bleh..` types of excessive
string concatenation into tables that then get concated only once, when
their containing functions return the final formspec string.
3) In some places in the code, such tables were already being used, and
were named "formspec", while others were named "fs". I settled on just
one name, "formspec", as it's more readable, if longer.
4) There was a mix of styles of adding items to those tables:
* Some places used line after line of `t[#t + 1] = foo/bar/baz`.
* Others places used the form `t[1] = foo, t[2] = bar, ...`.
* Still others used the form `t[n] = foo, t[n+1] = bar...`,
with `n` being increased or reset every so often.
Most of them should now be of the third form, with a few of the second.
2021-03-07 16:37:20 +01:00
|
|
|
F(S("Page")), S("@1 of @2",page2,pagemax))
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
2015-08-16 16:40:49 +02:00
|
|
|
n= n+1
|
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
if ui.activefilter[player_name] ~= "" then
|
2021-04-05 20:07:14 +02:00
|
|
|
formspec[n] = string.format("label[%f,%f;%s: %s]",
|
|
|
|
ui_peruser.page_x, ui_peruser.page_y - 0.25,
|
|
|
|
F(S("Filter")), F(ui.activefilter[player_name]))
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
2015-08-16 16:40:49 +02:00
|
|
|
return table.concat(formspec, "")
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.set_inventory_formspec(player, page)
|
2013-09-29 00:15:37 +02:00
|
|
|
if player then
|
2021-03-07 14:18:17 +01:00
|
|
|
player:set_inventory_formspec(ui.get_formspec(player, page))
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-04-05 20:07:14 +02:00
|
|
|
local function valid_def(def)
|
|
|
|
return (not def.groups.not_in_creative_inventory
|
|
|
|
or def.groups.not_in_creative_inventory == 0)
|
|
|
|
and def.description
|
|
|
|
and def.description ~= ""
|
|
|
|
end
|
|
|
|
|
2013-09-29 00:15:37 +02:00
|
|
|
--apply filter to the inventory list (create filtered copy of full one)
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.apply_filter(player, filter, search_dir)
|
2015-08-16 16:40:49 +02:00
|
|
|
if not player then
|
|
|
|
return false
|
|
|
|
end
|
2014-01-19 19:27:52 +01:00
|
|
|
local player_name = player:get_player_name()
|
2013-09-29 00:15:37 +02:00
|
|
|
local lfilter = string.lower(filter)
|
2014-05-01 21:44:16 +02:00
|
|
|
local ffilter
|
2014-01-19 17:34:44 +01:00
|
|
|
if lfilter:sub(1, 6) == "group:" then
|
|
|
|
local groups = lfilter:sub(7):split(",")
|
2014-05-01 21:44:16 +02:00
|
|
|
ffilter = function(name, def)
|
|
|
|
for _, group in ipairs(groups) do
|
2015-08-16 16:40:49 +02:00
|
|
|
if not def.groups[group]
|
|
|
|
or def.groups[group] <= 0 then
|
2014-05-01 21:44:16 +02:00
|
|
|
return false
|
2014-01-19 17:34:44 +01:00
|
|
|
end
|
|
|
|
end
|
2014-05-01 21:44:16 +02:00
|
|
|
return true
|
2014-01-19 17:34:44 +01:00
|
|
|
end
|
|
|
|
else
|
2020-07-15 20:21:06 +02:00
|
|
|
local lang = minetest.get_player_information(player_name).lang_code
|
2014-05-01 21:44:16 +02:00
|
|
|
ffilter = function(name, def)
|
|
|
|
local lname = string.lower(name)
|
|
|
|
local ldesc = string.lower(def.description)
|
2020-07-15 20:21:06 +02:00
|
|
|
local llocaldesc = minetest.get_translated_string
|
|
|
|
and string.lower(minetest.get_translated_string(lang, def.description))
|
2014-06-13 09:34:29 +02:00
|
|
|
return string.find(lname, lfilter, 1, true) or string.find(ldesc, lfilter, 1, true)
|
2020-07-15 20:21:06 +02:00
|
|
|
or llocaldesc and string.find(llocaldesc, lfilter, 1, true)
|
2014-05-01 21:44:16 +02:00
|
|
|
end
|
|
|
|
end
|
2021-03-07 14:18:17 +01:00
|
|
|
ui.filtered_items_list[player_name]={}
|
2021-04-05 20:07:14 +02:00
|
|
|
local category = ui.current_category[player_name] or 'all'
|
|
|
|
if category == 'all' then
|
|
|
|
for name, def in pairs(minetest.registered_items) do
|
|
|
|
if valid_def(def)
|
|
|
|
and ffilter(name, def) then
|
|
|
|
table.insert(ui.filtered_items_list[player_name], name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
elseif category == 'uncategorized' then
|
|
|
|
for name, def in pairs(minetest.registered_items) do
|
|
|
|
if (not ui.find_category(name))
|
|
|
|
and valid_def(def)
|
|
|
|
and ffilter(name, def) then
|
|
|
|
table.insert(ui.filtered_items_list[player_name], name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
else
|
|
|
|
for name,exists in pairs(ui.registered_category_items[category]) do
|
|
|
|
local def = minetest.registered_items[name]
|
|
|
|
if exists and def
|
|
|
|
and valid_def(def)
|
|
|
|
and ffilter(name, def) then
|
|
|
|
table.insert(ui.filtered_items_list[player_name], name)
|
|
|
|
end
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
|
|
|
end
|
2021-03-07 14:18:17 +01:00
|
|
|
table.sort(ui.filtered_items_list[player_name])
|
|
|
|
ui.filtered_items_list_size[player_name] = #ui.filtered_items_list[player_name]
|
|
|
|
ui.current_index[player_name] = 1
|
|
|
|
ui.activefilter[player_name] = filter
|
|
|
|
ui.active_search_direction[player_name] = search_dir
|
|
|
|
ui.set_inventory_formspec(player,
|
|
|
|
ui.current_page[player_name])
|
2013-09-29 00:15:37 +02:00
|
|
|
end
|
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.items_in_group(groups)
|
2013-09-29 00:15:37 +02:00
|
|
|
local items = {}
|
|
|
|
for name, item in pairs(minetest.registered_items) do
|
|
|
|
for _, group in pairs(groups:split(',')) do
|
|
|
|
if item.groups[group] then
|
|
|
|
table.insert(items, name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return items
|
|
|
|
end
|
2014-05-22 21:24:10 +02:00
|
|
|
|
2021-03-07 14:18:17 +01:00
|
|
|
function ui.sort_inventory(inv)
|
2014-05-22 21:24:10 +02:00
|
|
|
local inlist = inv:get_list("main")
|
|
|
|
local typecnt = {}
|
|
|
|
local typekeys = {}
|
|
|
|
for _, st in ipairs(inlist) do
|
|
|
|
if not st:is_empty() then
|
|
|
|
local n = st:get_name()
|
|
|
|
local w = st:get_wear()
|
|
|
|
local m = st:get_metadata()
|
|
|
|
local k = string.format("%s %05d %s", n, w, m)
|
|
|
|
if not typecnt[k] then
|
|
|
|
typecnt[k] = {
|
|
|
|
name = n,
|
|
|
|
wear = w,
|
|
|
|
metadata = m,
|
|
|
|
stack_max = st:get_stack_max(),
|
|
|
|
count = 0,
|
|
|
|
}
|
|
|
|
table.insert(typekeys, k)
|
|
|
|
end
|
|
|
|
typecnt[k].count = typecnt[k].count + st:get_count()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
table.sort(typekeys)
|
|
|
|
local outlist = {}
|
|
|
|
for _, k in ipairs(typekeys) do
|
|
|
|
local tc = typecnt[k]
|
|
|
|
while tc.count > 0 do
|
|
|
|
local c = math.min(tc.count, tc.stack_max)
|
|
|
|
table.insert(outlist, ItemStack({
|
|
|
|
name = tc.name,
|
|
|
|
wear = tc.wear,
|
|
|
|
metadata = tc.metadata,
|
|
|
|
count = c,
|
|
|
|
}))
|
|
|
|
tc.count = tc.count - c
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if #outlist > #inlist then return end
|
|
|
|
while #outlist < #inlist do
|
|
|
|
table.insert(outlist, ItemStack(nil))
|
|
|
|
end
|
|
|
|
inv:set_list("main", outlist)
|
|
|
|
end
|