minetest/mods/craftguide/init.lua
2020-07-12 22:31:30 -04:00

378 lines
9.5 KiB
Lua

local player_data = {}
local init_items = {}
local recipes_cache = {}
local usages_cache = {}
local group_stereotypes = {
dye = "dye:white",
wool = "wool:white",
coal = "default:coal_lump",
vessel = "vessels:glass_bottle",
flower = "flowers:dandelion_yellow"
}
local function table_replace(t, val, new)
for k, v in pairs(t) do
if v == val then
t[k] = new
end
end
end
local function item_in_recipe(item, recipe)
for _, recipe_item in pairs(recipe.items) do
if recipe_item == item then
return true
end
end
return false
end
local function extract_groups(str)
if str:sub(1, 6) == "group:" then
return str:sub(7):split()
end
end
local function item_has_groups(item_groups, groups)
for _, group in ipairs(groups) do
if not item_groups[group] then
return false
end
end
return true
end
-- If item can be used in recipe because recipe takes a `group:` item that item
-- matches, return a copy of recipe with the `group:` item replaced with item.
local function groups_item_in_recipe(item, recipe)
local item_groups = minetest.registered_items[item].groups
for _, recipe_item in pairs(recipe.items) do
local groups = extract_groups(recipe_item)
if groups and item_has_groups(item_groups, groups) then
local usage = table.copy(recipe)
table_replace(usage.items, recipe_item, item)
return usage
end
end
end
local function get_item_usages(item)
local usages = {}
for _, recipes in pairs(recipes_cache) do
for _, recipe in ipairs(recipes) do
if item_in_recipe(item, recipe) then
table.insert(usages, recipe)
else
recipe = groups_item_in_recipe(item, recipe)
if recipe then
table.insert(usages, recipe)
end
end
end
end
return #usages > 0 and usages
end
minetest.register_on_mods_loaded(function()
for name, def in pairs(minetest.registered_items) do
if def.groups.not_in_craft_guide ~= 1 and def.description ~= "" then
recipes_cache[name] = minetest.get_all_craft_recipes(name)
end
end
for name, def in pairs(minetest.registered_items) do
if def.groups.not_in_craft_guide ~= 1 and def.description ~= "" then
usages_cache[name] = get_item_usages(name)
if recipes_cache[name] or usages_cache[name] then
table.insert(init_items, name)
end
end
end
table.sort(init_items)
end)
local function groups_to_item(groups)
if #groups == 1 then
local group = groups[1]
if group_stereotypes[group] then
return group_stereotypes[group]
elseif minetest.registered_items["default:"..group] then
return "default:"..group
end
end
for name, def in pairs(minetest.registered_items) do
if item_has_groups(def.groups, groups) then
return name
end
end
return ":unknown"
end
local function get_burntime(item)
return minetest.get_craft_result({method="fuel", width=1, items={item}}).time
end
local function get_tooltip(item, groups, burntime)
local tooltip
if groups then
local groupstr = {}
for _, group in ipairs(groups) do
table.insert(groupstr, minetest.colorize("yellow", group))
end
groupstr = table.concat(groupstr, ", ")
tooltip = "Any item belonging to the group(s): "..groupstr
else
local itemdef = minetest.registered_items[item]
tooltip = itemdef and itemdef.description or "Unknown Item"
end
if burntime > 0 then
tooltip = tooltip.."\nBurning time: "..minetest.colorize("yellow", burntime)
end
return ("tooltip[%s;%s]"):format(item, tooltip)
end
local function get_recipe_formspec(data)
local fs = {}
local recipe = data.recipes[data.rnum]
local width = recipe.width
local cooktime, shapeless
if recipe.method == "cooking" then
cooktime, width = width, 1
elseif width == 0 then
shapeless = true
width = #recipe.items <= 4 and 2 or math.min(3, #recipe.items)
end
table.insert(fs, ("label[5.5,6.6;%s %d of %d]")
:format(data.show_usages and "Usage" or "Recipe", data.rnum, #data.recipes))
if #data.recipes > 1 then
table.insert(fs, [[
image_button[5.5,7.2;0.8,0.8;craftguide_prev_icon.png;recipe_prev;]
image_button[6.2,7.2;0.8,0.8;craftguide_next_icon.png;recipe_next;]
]])
end
local rows = math.ceil(table.maxn(recipe.items) / width)
if width > 3 or rows > 3 then
table.insert(fs, "label[0,6.6;Recipe is too big to be displayed.]")
return table.concat(fs)
end
for i, item in pairs(recipe.items) do
local x = (i - 1) % width + 3 - width
local y = math.ceil(i / width + 6 - math.min(2, rows)) + 0.6
local burntime = get_burntime(item)
local groups = extract_groups(item)
if groups then
item = groups_to_item(groups)
end
table.insert(fs, ("item_image_button[%d,%f;1.05,1.05;%s;%s;%s]")
:format(x, y, item, item, groups and "\nG" or ""))
if groups or burntime > 0 then
table.insert(fs, get_tooltip(item, groups, burntime))
end
end
if shapeless or recipe.method == "cooking" then
table.insert(fs, ("image[3.2,6.1;0.5,0.5;craftguide_%s.png]")
:format(shapeless and "shapeless" or "furnace"))
local tooltip = shapeless and "Shapeless" or
"Cooking time: "..minetest.colorize("yellow", cooktime)
table.insert(fs, "tooltip[3.2,6.1;0.5,0.5;"..tooltip.."]")
end
table.insert(fs, "image[3,6.6;1,1;gui_furnace_arrow_bg.png^[transformR270]")
local output_name = recipe.output:match("%S*")
table.insert(fs, ("item_image_button[4,6.6;1.05,1.05;%s;%s;]")
:format(recipe.output, output_name))
local burntime = get_burntime(output_name)
if burntime > 0 then
table.insert(fs, get_tooltip(output_name, nil, burntime))
end
return table.concat(fs)
end
local function get_formspec(name)
local data = player_data[name]
data.pagemax = math.max(1, math.ceil(#data.items / 32))
local fs = {}
table.insert(fs, ("field[0.3,0.32;2.5,1;filter;;%s]")
:format(minetest.formspec_escape(data.filter)))
table.insert(fs, ("label[6.2,0.22;%s / %d]")
:format(minetest.colorize("yellow", data.pagenum), data.pagemax))
table.insert(fs, [[
image_button[2.4,0.12;0.8,0.8;craftguide_search_icon.png;search;]
image_button[3.05,0.12;0.8,0.8;craftguide_clear_icon.png;clear;]
image_button[5.4,0.12;0.8,0.8;craftguide_prev_icon.png;prev;]
image_button[7.2,0.12;0.8,0.8;craftguide_next_icon.png;next;]
tooltip[search;Search]
tooltip[clear;Reset]
tooltip[prev;Previous page]
tooltip[next;Next page]
field_close_on_enter[filter;false]
]])
if #data.items == 0 then
table.insert(fs, "label[3,2;No items to show.]")
else
local first_item = (data.pagenum - 1) * 32
for i = first_item, first_item + 31 do
local item = data.items[i + 1]
if not item then
break
end
local x = i % 8
local y = (i % 32 - x) / 8 + 1
table.insert(fs, ("item_image_button[%d,%d;1.05,1.05;%s;%s_inv;]")
:format(x, y, item, item))
end
end
if data.recipes then
if #data.recipes > 0 then
table.insert(fs, get_recipe_formspec(data))
elseif data.show_usages then
table.insert(fs, "label[2,6.6;No usages.\nClick again to show recipes.]")
else
table.insert(fs, "label[2,6.6;No recipes.\nClick again to show usages.]")
end
end
return table.concat(fs)
end
local function execute_search(data)
local filter = data.filter
if filter == "" then
data.items = init_items
return
end
data.items = {}
for _, item in ipairs(init_items) do
local itemdef = minetest.registered_items[item]
local desc = itemdef and itemdef.description:lower() or ""
if item:find(filter, 1, true) or desc:find(filter, 1, true) then
table.insert(data.items, item)
end
end
end
local function reset_data(data)
data.filter = ""
data.pagenum = 1
data.prev_item = nil
data.recipes = nil
data.items = init_items
end
local function on_receive_fields(player, fields)
local name = player:get_player_name()
local data = player_data[name]
if fields.clear then
reset_data(data)
return true
elseif fields.key_enter_field == "filter" or fields.search then
local new = fields.filter:lower()
if new ~= "" and data.filter == new then
return
end
data.filter = new
data.pagenum = 1
execute_search(data)
return true
elseif fields.prev or fields.next then
if data.pagemax == 1 then
return
end
data.pagenum = data.pagenum + (fields.next and 1 or -1)
if data.pagenum > data.pagemax then
data.pagenum = 1
elseif data.pagenum == 0 then
data.pagenum = data.pagemax
end
return true
elseif fields.recipe_next or fields.recipe_prev then
data.rnum = data.rnum + (fields.recipe_next and 1 or -1)
if data.rnum > #data.recipes then
data.rnum = 1
elseif data.rnum == 0 then
data.rnum = #data.recipes
end
return true
else
local item
for field in pairs(fields) do
if field:find(":") then
item = field
break
end
end
if not item then
return
end
if item:sub(-4) == "_inv" then
item = item:sub(1, -5)
end
if item == data.prev_item then
data.show_usages = not data.show_usages
else
data.show_usages = nil
end
if data.show_usages then
data.recipes = usages_cache[item] or {}
else
data.recipes = recipes_cache[item] or {}
end
data.prev_item = item
data.rnum = 1
return true
end
end
minetest.register_on_joinplayer(function(player)
local name = player:get_player_name()
player_data[name] = {
filter = "",
pagenum = 1,
items = init_items
}
end)
minetest.register_on_leaveplayer(function(player)
local name = player:get_player_name()
player_data[name] = nil
end)
sfinv.register_page("craftguide:craftguide", {
title = "Craft Guide",
get = function(self, player, context)
local name = player:get_player_name()
return sfinv.make_formspec(player, context, get_formspec(name))
end,
on_player_receive_fields = function(self, player, context, fields)
if on_receive_fields(player, fields) then
sfinv.set_player_inventory_formspec(player)
end
end
})