--[[ TechAge ======= Copyright (C) 2019-2020 Joachim Stolberg GPL v3 See LICENSE.txt for more information ICTA Controller - Formspec A sub-menu control to generate a formspec sting for conditions and actions ]]-- local function index(list, x) for idx, v in ipairs(list) do if v == x then return idx end end return nil end -- generate the choice dependent part of the form local function add_controls_to_table(tbl, kvDefinition, kvSelect) local val = "" local offs = 1.4 if kvDefinition[kvSelect.choice] then local lControls = kvDefinition[kvSelect.choice].formspec for idx,elem in ipairs(lControls) do if elem.type == "label" then tbl[#tbl+1] = "label[0,"..offs..";Description:\n"..elem.label.."]" offs = offs + 0.4 elseif elem.label and elem.label ~= "" then tbl[#tbl+1] = "label[0,"..offs..";"..elem.label..":]" offs = offs + 0.4 end if elem.type == "numbers" or elem.type == "digits" or elem.type == "letters" or elem.type == "ascii" then val = kvSelect[elem.name] or elem.default tbl[#tbl+1] = "field[0.3,"..(offs+0.2)..";8,1;"..elem.name..";;"..val.."]" offs = offs + 0.9 elseif elem.type == "textlist" then local l = elem.choices:split(",") val = index(l, kvSelect[elem.name]) or elem.default tbl[#tbl+1] = "dropdown[0.0,"..(offs)..";8.5,1.4;"..elem.name..";"..elem.choices..";"..val.."]" offs = offs + 0.9 end end end return tbl end local function default_data(kvDefinition, kvSelect) local lControls = kvDefinition[kvSelect.choice].formspec for idx,elem in ipairs(lControls) do kvSelect[elem.name] = elem.default end kvSelect.button = kvDefinition[kvSelect.choice].button(kvSelect) return kvSelect end -- Copy field/formspec data to the table kvSelect -- kvDefinition: submenu formspec definition -- kvSelect: form data -- fields: formspec input local function field_to_kvSelect(kvDefinition, kvSelect, fields) local error = false local lControls = kvDefinition[kvSelect.choice].formspec for idx,elem in ipairs(lControls) do if elem.type == "numbers" then if fields[elem.name] then if fields[elem.name]:find("^[%d ]+$") then kvSelect[elem.name] = fields[elem.name] else kvSelect[elem.name] = elem.default error = true end end elseif elem.type == "digits" then -- including positions if fields[elem.name] then if fields[elem.name]:find("^[+%%-,%d]+$") then kvSelect[elem.name] = fields[elem.name] else kvSelect[elem.name] = elem.default error = true end end elseif elem.type == "letters" then if fields[elem.name] then if fields[elem.name]:find("^[+-]?%a+$") then kvSelect[elem.name] = fields[elem.name] else kvSelect[elem.name] = elem.default error = true end end elseif elem.type == "ascii" then if fields[elem.name] then kvSelect[elem.name] = fields[elem.name] end elseif elem.type == "textlist" then if fields[elem.name] ~= nil then kvSelect[elem.name] = fields[elem.name] end end end -- store user input of button text if fields._button_ then kvSelect._button_ = fields._button_ end -- select button text if error then kvSelect.button = "invalid" elseif kvSelect._button_ and kvSelect._button_ ~= "" then kvSelect.button = kvSelect._button_ else kvSelect.button = kvDefinition[kvSelect.choice].button(kvSelect) end return kvSelect end function techage.submenu_verify(owner, kvDefinition, kvSelect) local error = false local lControls = kvDefinition[kvSelect.choice].formspec for idx,elem in ipairs(lControls) do if elem.type == "numbers" then if not kvSelect[elem.name]:find("^[%d ]+$") then error = true end if not techage.check_numbers(kvSelect[elem.name], owner) then error = true end elseif elem.type == "number" then if not kvSelect[elem.name]:find("^[%d]+$") then error = true end if not techage.check_numbers(kvSelect[elem.name], owner) then error = true end elseif elem.type == "digits" then -- including positions if not kvSelect[elem.name]:find("^[+%%-,%d]+$") then error = true end elseif elem.type == "letters" then if not kvSelect[elem.name]:find("^[+-]?%a+$") then error = true end elseif elem.type == "ascii" then if kvSelect[elem.name] == "" or kvSelect[elem.name] == nil then error = true end elseif elem.type == "textlist" then if kvSelect[elem.name] == "" or kvSelect[elem.name] == nil then error = true end end end return (error == false) end -- generate a formspec string from the given control definition -- row, col: numbers to identify the control -- title: Title text for the control -- lKeys: list of keywords of selected choices according to fields -- lChoice: list of possible choices for the control -- kvDefinition: definitions of the choice dependent controls -- kvSelect: data of the last selected item {choice, number, value, ...} function techage.submenu_generate_formspec(row, col, title, lKeys, lChoice, kvDefinition, kvSelect) if kvSelect == nil or next(kvSelect) == nil then kvSelect = {choice = "default"} end local tbl = {"size[8.2,9]".. default.gui_bg.. default.gui_bg_img.. default.gui_slots.. "field[0,0;0,0;_row_;;"..row.."]".. "field[0,0;0,0;_col_;;"..col.."]"} local sChoice = table.concat(lChoice, ",") local idx = index(lKeys, kvSelect.choice) or 1 tbl[#tbl+1] = "label[0,0;"..title..":]" tbl[#tbl+1] = "dropdown[0,0.5;8.5,1;choice;"..sChoice..";"..idx.."]" tbl = add_controls_to_table(tbl, kvDefinition, kvSelect) tbl[#tbl+1] = "field[0.2,8.7;4,1;_button_;Alternative button text;"..(kvSelect._button_ or "").."]" tbl[#tbl+1] = "button[4,8.4;2,1;_cancel_;cancel]" tbl[#tbl+1] = "button[6,8.4;2,1;_exit_;ok]" return table.concat(tbl) end -- return the selected and configured menu item based on user inputs (fields) function techage.submenu_eval_input(kvDefinition, lKeys, lChoice, kvSelect, fields) -- determine selected choice if fields.choice then -- load with default values local idx = index(lChoice, fields.choice) or 1 kvSelect = {choice = lKeys[idx]} kvSelect = default_data(kvDefinition, kvSelect) kvSelect = field_to_kvSelect(kvDefinition, kvSelect, fields) else -- add real data kvSelect = field_to_kvSelect(kvDefinition, kvSelect, fields) end return kvSelect end