--[[ TechAge ======= Copyright (C) 2019-2025 Joachim Stolberg AGPL v3 See LICENSE.txt for more information A formspec control to generate formspec strings for machine settings and monitoring ]]-- local S = techage.S techage.menu = {} local function index(list, x) for idx, v in ipairs(list or {}) do if tostring(v) == x then return idx end end return nil end local function allow_put(inv, listname, index, stack, player) local list = inv:get_list(listname) stack:set_count(1) inv:set_stack(listname, index, stack) return 0 end local function allow_take(inv, listname, index, stack, player) local list = inv:get_list(listname) stack:set_count(0) inv:set_stack(listname, index, stack) return 0 end -- generate the formspec string to be placed into a container frame local function generate_formspec_substring(pos, meta, form_def, player_name, width, no_note) local tbl = {} local player_inv_needed = false width = width or "10" local xpos1 = tostring(tonumber(width) / 2 - 0.25) local xpos2 = tostring(tonumber(width) / 2) local xpos3 = tostring(tonumber(width) / 2 + 0.3) local xpos4 = tostring(tonumber(width) / 2 + 0.5) local xsize = tostring(tonumber(width) / 2) if meta and form_def then local nvm = techage.get_nvm(pos) for i,elem in ipairs(form_def) do local offs = (i - 1) * 0.9 - 0.2 tbl[#tbl+1] = "label[0," .. offs .. ";" .. minetest.formspec_escape(elem.label) .. ":]" tbl[#tbl+1] = "tooltip[0," .. offs .. ";4,1;" .. elem.tooltip .. "]" if elem.type == "label" then -- none elseif elem.type == "number" then local val = elem.default if meta:contains(elem.name) then val = meta:get_int(elem.name) end if nvm.running or techage.is_running(nvm) then tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. val .. "]" else tbl[#tbl+1] = "field[" .. xpos2 .. "," .. (offs+0.2) .. ";" .. xpos3 .. ",1;" .. elem.name .. ";;" .. val .. "]" end elseif elem.type == "numbers" then local val = elem.default if meta:contains(elem.name) then val = meta:get_string(elem.name) end if nvm.running or techage.is_running(nvm) then tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. val .. "]" else tbl[#tbl+1] = "field[" .. xpos2 .. "," .. (offs+0.2) .. ";" .. xpos3 .. ",1;" .. elem.name .. ";;" .. val .. "]" end elseif elem.type == "float" then local val = elem.default if meta:contains(elem.name) then val = tonumber(meta:get_string(elem.name)) or 0 end if nvm.running or techage.is_running(nvm) then tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. val .. "]" else tbl[#tbl+1] = "field[" .. xpos2 .. "," .. (offs+0.2) .. ";" .. xpos3 .. ",1;" .. elem.name .. ";;" .. val .. "]" end elseif elem.type == "ascii" then local val = elem.default if meta:contains(elem.name) then val = meta:get_string(elem.name) end if nvm.running or techage.is_running(nvm) then tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. minetest.formspec_escape(val) .. "]" else tbl[#tbl+1] = "field[" .. xpos2 .. "," .. (offs+0.2) .. ";" .. xpos3 .. ",1;" .. elem.name .. ";;" .. minetest.formspec_escape(val) .. "]" end elseif elem.type == "const" then tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. elem.value .. "]" elseif elem.type == "output" then local val = nvm[elem.name] or meta:get_string(elem.name) or "" if tonumber(val) then val = techage.round(val) end tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. val .. "]" elseif elem.type == "dropdown" then if nvm.running or techage.is_running(nvm) then local val = elem.default or "" if meta:contains(elem.name) then val = meta:get_string(elem.name) or "" if elem.values then local idx = index(elem.values, val) or 1 local l = elem.choices:split(",") val = l[idx] end end tbl[#tbl+1] = "label[" .. xpos1 .. "," .. offs .. ";" .. val .. "]" elseif elem.on_dropdown then -- block provides a specific list of choice elements local val = elem.default if meta:contains(elem.name) then val = meta:get_string(elem.name) or "" end local choices = elem.on_dropdown(pos) local l = choices:split(",") local idx = index(l, val) or 1 tbl[#tbl+1] = "dropdown[" .. xpos1 .. "," .. (offs) .. ";" .. xpos4 .. ",1.4;" .. elem.name .. ";" .. choices .. ";" .. idx .. "]" else local val = elem.default if meta:contains(elem.name) then val = meta:get_string(elem.name) or "" end local idx if elem.values then idx = index(elem.values, val) or 1 else local l = elem.choices:split(",") idx = index(l, val) or 1 end tbl[#tbl+1] = "dropdown[" .. xpos1 .. "," .. (offs) .. ";" .. xpos4 .. ",1.4;" .. elem.name .. ";" .. elem.choices .. ";" .. idx .. "]" end elseif elem.type == "items" then -- inventory if elem.size then tbl[#tbl+1] = "list[detached:" .. minetest.formspec_escape(player_name) .. "_techage_wrench_menu;cfg;" .. xpos1 .. "," .. offs .. ";" .. elem.size .. ",1;]" else tbl[#tbl+1] = "list[detached:" .. minetest.formspec_escape(player_name) .. "_techage_wrench_menu;cfg;" .. xpos1 .. "," .. offs .. ";" .. elem.width .. "," .. elem.height .. ";]" end player_inv_needed = true end end if not no_note and (nvm.running or techage.is_running(nvm)) then local offs = #form_def * 0.9 - 0.3 tbl[#tbl+1] = "label[0," .. offs .. ";" .. S("Note: You can't change any values while the block is running!") .. "]" end end return player_inv_needed, table.concat(tbl, "") end local function value_check(elem, value, player_name) if elem.check then return elem.check(value, player_name) end return value ~= nil end local function evaluate_data(pos, meta, form_def, fields, player_name) local res = true if meta and form_def then local nvm = techage.get_nvm(pos) if nvm.running or techage.is_running(nvm) then return res end for idx,elem in ipairs(form_def) do if elem.type == "number" then if fields[elem.name] then if fields[elem.name] == "" then meta:set_string(elem.name, "") elseif fields[elem.name]:find("^[%d ]+$") then local val = tonumber(fields[elem.name]) if value_check(elem, val, player_name) then meta:set_int(elem.name, val) --print("set_int", elem.name, val) else res = false end else res = false end end elseif elem.type == "numbers" then if fields[elem.name] then if fields[elem.name] == "" then meta:set_string(elem.name, "") elseif fields[elem.name]:find("^[%d ]+$") and value_check(elem, fields[elem.name], player_name) then meta:set_string(elem.name, fields[elem.name]) else res = false end end elseif elem.type == "float" then if fields[elem.name] == ""then meta:set_string(elem.name, "") elseif fields[elem.name] then local val = tonumber(fields[elem.name]) if val and value_check(elem, val, player_name) then meta:set_string(elem.name, val) else res = false end end elseif elem.type == "ascii" then if fields[elem.name] == ""then meta:set_string(elem.name, "") elseif fields[elem.name] then if value_check(elem, fields[elem.name], player_name) then meta:set_string(elem.name, fields[elem.name]) else res = false end end elseif elem.type == "dropdown" then if fields[elem.name] ~= nil then if elem.values then local l = elem.choices:split(",") local idx = index(l, fields[elem.name]) or 1 local text = elem.values[idx] meta:set_string(elem.name, text) else meta:set_string(elem.name, fields[elem.name]) end end elseif elem.type == "items" and player_name then local inv_name = minetest.formspec_escape(player_name) .. "_techage_wrench_menu" local dinv = minetest.get_inventory({type = "detached", name = inv_name}) local ninv = minetest.get_inventory({type = "node", pos = pos}) if dinv and ninv then for i = 1, ninv:get_size("cfg") do ninv:set_stack("cfg", i, dinv:get_stack("cfg", i)) end end end end end return res end function techage.menu.generate_formspec(pos, ndef, form_def, player_name) local meta = minetest.get_meta(pos) local number = techage.get_node_number(pos) or "-" local mem = techage.get_mem(pos) mem.star = ((mem.star or 0) + 1) % 2 local star = mem.star == 1 and "*" or "" if player_name then local inv_name = minetest.formspec_escape(player_name) .. "_techage_wrench_menu" minetest.create_detached_inventory(inv_name, { allow_put = allow_put, allow_take = allow_take}) local dinv = minetest.get_inventory({type = "detached", name = inv_name}) local ninv = minetest.get_inventory({type = "node", pos = pos}) if dinv and ninv then dinv:set_size('cfg', ninv:get_size("cfg")) for i = 1, ninv:get_size("cfg") do dinv:set_stack("cfg", i, ninv:get_stack("cfg", i)) end end end if meta and number and ndef and form_def then local title = ndef.description .. " (" .. number .. ")" local player_inv_needed, text = generate_formspec_substring(pos, meta, form_def, player_name) local buttons if player_inv_needed then buttons = "button[0.5,6.2;3,1;refresh;" .. S("Refresh") .. "]" .. "button_exit[3.5,6.2;3,1;cancel;" .. S("Cancel") .. "]" .. "button[6.5,6.2;3,1;save;" .. S("Save") .. "]" .. "list[current_player;main;1,7.2;8,2;]" else buttons = "button[0.5,8.4;3,1;refresh;" .. S("Refresh") .. "]" .. "button_exit[3.5,8.4;3,1;cancel;" .. S("Cancel") .. "]" .. "button[6.5,8.4;3,1;save;" .. S("Save") .. "]" end if #form_def > 8 then local size = (#form_def * 10) - 60 return "size[10,9]" .. default.gui_bg .. default.gui_bg_img .. default.gui_slots .. "box[0,-0.1;9.8,0.5;#c6e8ff]" .. "label[0.2,-0.1;" .. minetest.colorize( "#000000", title) .. "]" .. "label[9.5,-0.1;" .. minetest.colorize( "#000000", star) .. "]" .. "scrollbaroptions[max=" .. size .. "]" .. "scrollbar[9.4,0.6;0.4,7.7;vertical;wrenchmenu;]" .. "scroll_container[0,1;12,9;wrenchmenu;vertical;]" .. text .. "scroll_container_end[]" .. buttons else return "size[10,9]" .. default.gui_bg .. default.gui_bg_img .. default.gui_slots .. "box[0,-0.1;9.8,0.5;#c6e8ff]" .. "label[0.2,-0.1;" .. minetest.colorize( "#000000", title) .. "]" .. "label[9.5,-0.1;" .. minetest.colorize( "#000000", star) .. "]" .. "container[0,1]" .. text .. "container_end[]" .. buttons end end return "" end function techage.menu.generate_formspec_container(pos, ndef, form_def, ypos, width) local meta = minetest.get_meta(pos) local number = techage.get_node_number(pos) or "-" local mem = techage.get_mem(pos) mem.star = ((mem.star or 0) + 1) % 2 local star = mem.star == 1 and " *" or " " local bttn_width = (width - 0.3) / 3 if meta and number and ndef and form_def then local _, text = generate_formspec_substring(pos, meta, form_def, "", width, true) local yoffs = math.min(#form_def, 8) * 0.9 + 0.7 local buttons = "button[0.1," .. yoffs .. ";" .. bttn_width .. ",1;refresh;" .. S("Refresh") .. star .. "]" .. "button_exit[" .. (bttn_width + 0.2) .. "," .. yoffs .. ";" .. bttn_width .. ",1;cancel;" .. S("Cancel") .. "]" .. "button[" .. (2 * bttn_width + 0.3) .. "," .. yoffs .. ";" .. bttn_width .. ",1;save;" .. S("Save") .. "]" if #form_def > 8 then local size = (#form_def * 10) - 60 return "container[0," .. ypos .. "]" .. "box[0,0;" .. width .. "," .. (yoffs + 0.8) .. ";#395c74]".. "scrollbaroptions[max=" .. size .. "]" .. "scrollbar[9.4,0.6;0.4,7.7;vertical;wrenchmenu;]" .. "scroll_container[0,1;12,9;wrenchmenu;vertical;]" .. text .. "scroll_container_end[]" .. buttons .. "container_end[]" else return "container[0," .. ypos .. "]" .. "container[0,1]" .. "box[-0.1,-0.3;" .. (width + 0.2) .. "," .. (yoffs + 0.3) .. ";#395c74]".. text .. "container_end[]" .. buttons .. "container_end[]" end end return "" end function techage.menu.eval_input(pos, form_def, fields, player_name) if fields.save or fields.key_enter_field then local meta = minetest.get_meta(pos) evaluate_data(pos, meta, form_def, fields, player_name) end return fields.refresh or fields.save or fields.key_enter_field end function techage.dropdown_index(sChoices, selected_value) local l = sChoices:split(",") return index(l, selected_value) or 1 end