mirror of
				https://github.com/luanti-org/luanti.git
				synced 2025-10-26 13:25:27 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			464 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			464 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| --Luanti
 | |
| --Copyright (C) 2022 rubenwardy
 | |
| --
 | |
| --This program is free software; you can redistribute it and/or modify
 | |
| --it under the terms of the GNU Lesser General Public License as published by
 | |
| --the Free Software Foundation; either version 2.1 of the License, or
 | |
| --(at your option) any later version.
 | |
| --
 | |
| --This program is distributed in the hope that it will be useful,
 | |
| --but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| --MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| --GNU Lesser General Public License for more details.
 | |
| --
 | |
| --You should have received a copy of the GNU Lesser General Public License along
 | |
| --with this program; if not, write to the Free Software Foundation, Inc.,
 | |
| --51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 | |
| 
 | |
| local make = {}
 | |
| 
 | |
| 
 | |
| -- This file defines various component constructors, of the form:
 | |
| --
 | |
| --     make.component(setting)
 | |
| --
 | |
| -- `setting` is a table representing the settingtype.
 | |
| --
 | |
| -- A component is a table with the following:
 | |
| --
 | |
| -- * `full_width`: (Optional) true if the component shouldn't reserve space for info / reset.
 | |
| -- * `info_text`: (Optional) string, informational text shown in an info icon.
 | |
| -- * `setting`: (Optional) the setting.
 | |
| -- * `max_w`: (Optional) maximum width, `avail_w` will never exceed this.
 | |
| -- * `resettable`: (Optional) if this is true, a reset button is shown.
 | |
| -- * `get_formspec = function(self, avail_w)`:
 | |
| --     * `avail_w` is the available width for the component.
 | |
| --     * Returns `fs, used_height`.
 | |
| --     * `fs` is a string for the formspec.
 | |
| --       Components should be relative to `0,0`, and not exceed `avail_w` or the returned `used_height`.
 | |
| --     * `used_height` is the space used by components in `fs`.
 | |
| -- * `on_submit = function(self, fields, parent)`:
 | |
| --     * `fields`: submitted formspec fields
 | |
| --     * `parent`: the fstk element for the settings UI, use to show dialogs
 | |
| --     * Return true if the event was handled, to prevent future components receiving it.
 | |
| 
 | |
| 
 | |
| local function get_label(setting)
 | |
| 	local show_technical_names = core.settings:get_bool("show_technical_names")
 | |
| 	if not show_technical_names and setting.readable_name then
 | |
| 		return fgettext(setting.readable_name)
 | |
| 	end
 | |
| 	return setting.name
 | |
| end
 | |
| 
 | |
| 
 | |
| local function is_valid_number(value)
 | |
| 	return type(value) == "number" and not (value ~= value or value >= math.huge or value <= -math.huge)
 | |
| end
 | |
| 
 | |
| 
 | |
| function make.heading(text)
 | |
| 	return {
 | |
| 		full_width = true,
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			return ("label[0,0.6;%s]box[0,0.9;%f,0.05;#ccc6]"):format(core.formspec_escape(text), avail_w), 1.2
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| function make.note(text)
 | |
| 	return {
 | |
| 		full_width = true,
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			-- Assuming label height 0.4:
 | |
| 			-- Position at y=0 to eat 0.2 of the padding above, leave 0.05.
 | |
| 			-- The returned used_height doesn't include padding.
 | |
| 			return ("label[0,0;%s]"):format(core.colorize("#bbb", core.formspec_escape(text))), 0.2
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| --- Used for string and numeric style fields
 | |
| ---
 | |
| --- @param converter Function to coerce values from strings.
 | |
| --- @param validator Validator function, optional. Returns true when valid.
 | |
| --- @param stringifier Function to convert values to strings, optional.
 | |
| local function make_field(converter, validator, stringifier)
 | |
| 	return function(setting)
 | |
| 		return {
 | |
| 			info_text = setting.comment,
 | |
| 			setting = setting,
 | |
| 
 | |
| 			get_formspec = function(self, avail_w)
 | |
| 				local value = core.settings:get(setting.name) or setting.default
 | |
| 				self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 				local fs = ("field[0,0.3;%f,0.8;%s;%s;%s]"):format(
 | |
| 					avail_w - 1.5, setting.name, get_label(setting), core.formspec_escape(value))
 | |
| 				fs = fs .. ("field_enter_after_edit[%s;true]"):format(setting.name)
 | |
| 				fs = fs .. ("field_close_on_enter[%s;false]"):format(setting.name) -- for pause menu env
 | |
| 				fs = fs .. ("button[%f,0.3;1.5,0.8;%s;%s]"):format(avail_w - 1.5, "set_" .. setting.name, fgettext("Set"))
 | |
| 
 | |
| 				return fs, 1.1
 | |
| 			end,
 | |
| 
 | |
| 			on_submit = function(self, fields)
 | |
| 				if fields["set_" .. setting.name] or fields.key_enter_field == setting.name then
 | |
| 					local value = converter(fields[setting.name])
 | |
| 					if value == nil or (validator and not validator(value)) then
 | |
| 						return true
 | |
| 					end
 | |
| 
 | |
| 					if setting.min then
 | |
| 						value = math.max(value, setting.min)
 | |
| 					end
 | |
| 					if setting.max then
 | |
| 						value = math.min(value, setting.max)
 | |
| 					end
 | |
| 					core.settings:set(setting.name, (stringifier or tostring)(value))
 | |
| 					return true
 | |
| 				end
 | |
| 			end,
 | |
| 		}
 | |
| 	end
 | |
| end
 | |
| 
 | |
| 
 | |
| make.float = make_field(tonumber, is_valid_number, function(x)
 | |
| 	local str = tostring(x)
 | |
| 	if str:match("^[+-]?%d+$") then
 | |
| 		str = str .. ".0"
 | |
| 	end
 | |
| 	return str
 | |
| end)
 | |
| make.int = make_field(function(x)
 | |
| 	local value = tonumber(x)
 | |
| 	return value and math.floor(value)
 | |
| end, is_valid_number)
 | |
| make.string = make_field(tostring, nil)
 | |
| 
 | |
| 
 | |
| function make.bool(setting)
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			local value = core.settings:get_bool(setting.name, core.is_yes(setting.default))
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			local fs = ("checkbox[0,0.25;%s;%s;%s]"):format(
 | |
| 				setting.name, get_label(setting), tostring(value))
 | |
| 			return fs, 0.5
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields)
 | |
| 			if fields[setting.name] == nil then
 | |
| 				return false
 | |
| 			end
 | |
| 
 | |
| 			core.settings:set_bool(setting.name, core.is_yes(fields[setting.name]))
 | |
| 			return true
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| function make.enum(setting)
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 		max_w = 4.5,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			local value = core.settings:get(setting.name) or setting.default
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			local labels = setting.option_labels or {}
 | |
| 
 | |
| 			local items = {}
 | |
| 			for i, option in ipairs(setting.values) do
 | |
| 				items[i] = core.formspec_escape(labels[option] or option)
 | |
| 			end
 | |
| 
 | |
| 			local selected_idx = table.indexof(setting.values, value)
 | |
| 			local fs = "label[0,0.1;" .. get_label(setting) .. "]"
 | |
| 
 | |
| 			fs = fs .. ("dropdown[0,0.3;%f,0.8;%s;%s;%d;true]"):format(
 | |
| 				avail_w, setting.name, table.concat(items, ","), selected_idx, value)
 | |
| 
 | |
| 			return fs, 1.1
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields)
 | |
| 			local old_value = core.settings:get(setting.name) or setting.default
 | |
| 			local idx = tonumber(fields[setting.name]) or 0
 | |
| 			local value = setting.values[idx]
 | |
| 			if value == nil or value == old_value then
 | |
| 				return false
 | |
| 			end
 | |
| 
 | |
| 			core.settings:set(setting.name, value)
 | |
| 			return true
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| local function make_path(setting)
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			local value = core.settings:get(setting.name) or setting.default
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			local fs = ("field[0,0.3;%f,0.8;%s;%s;%s]"):format(
 | |
| 				avail_w - 3, setting.name, get_label(setting), core.formspec_escape(value))
 | |
| 			fs = fs .. ("field_enter_after_edit[%s;true]"):format(setting.name)
 | |
| 			fs = fs .. ("field_close_on_enter[%s;false]"):format(setting.name) -- for pause menu env
 | |
| 			fs = fs .. ("button[%f,0.3;1.5,0.8;%s;%s]"):format(avail_w - 3, "pick_" .. setting.name, fgettext("Browse"))
 | |
| 			fs = fs .. ("button[%f,0.3;1.5,0.8;%s;%s]"):format(avail_w - 1.5, "set_" .. setting.name, fgettext("Set"))
 | |
| 
 | |
| 			return fs, 1.1
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields)
 | |
| 			local dialog_name = "dlg_path_" .. setting.name
 | |
| 			if fields["pick_" .. setting.name] then
 | |
| 				local is_file = setting.type ~= "path"
 | |
| 				core.show_path_select_dialog(dialog_name,
 | |
| 					is_file and fgettext_ne("Select file") or fgettext_ne("Select directory"), is_file)
 | |
| 				return true
 | |
| 			end
 | |
| 			if fields[dialog_name .. "_accepted"] then
 | |
| 				local value = fields[dialog_name .. "_accepted"]
 | |
| 				if value ~= nil then
 | |
| 					core.settings:set(setting.name, value)
 | |
| 				end
 | |
| 				return true
 | |
| 			end
 | |
| 			if fields["set_" .. setting.name] or fields.key_enter_field == setting.name then
 | |
| 				local value = fields[setting.name]
 | |
| 				if value ~= nil then
 | |
| 					core.settings:set(setting.name, value)
 | |
| 				end
 | |
| 				return true
 | |
| 			end
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| if PLATFORM == "Android" or INIT == "pause_menu" then
 | |
| 	-- The Irrlicht file picker doesn't work on Android.
 | |
| 	-- Access to the Irrlicht file picker isn't implemented in the pause menu.
 | |
| 	-- We want to delete the Irrlicht file picker anyway, so any time spent on
 | |
| 	-- that would be wasted.
 | |
| 	make.path = make.string
 | |
| 	make.filepath = make.string
 | |
| else
 | |
| 	make.path = make_path
 | |
| 	make.filepath = make_path
 | |
| end
 | |
| 
 | |
| 
 | |
| function make.v3f(setting)
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			local value = vector.from_string(core.settings:get(setting.name) or setting.default)
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			-- Allocate space for "Set" button
 | |
| 			avail_w = avail_w - 1
 | |
| 
 | |
| 			local fs = "label[0,0.1;" .. get_label(setting) .. "]"
 | |
| 
 | |
| 			local field_width = (avail_w - 3*0.25) / 3
 | |
| 
 | |
| 			fs = fs .. ("field[%f,0.6;%f,0.8;%s;%s;%s]"):format(
 | |
| 				0, field_width, setting.name .. "_x", "X", value.x)
 | |
| 			fs = fs .. ("field[%f,0.6;%f,0.8;%s;%s;%s]"):format(
 | |
| 				field_width + 0.25, field_width, setting.name .. "_y", "Y", value.y)
 | |
| 			fs = fs .. ("field[%f,0.6;%f,0.8;%s;%s;%s]"):format(
 | |
| 				2 * (field_width + 0.25), field_width, setting.name .. "_z", "Z", value.z)
 | |
| 
 | |
| 			fs = fs .. ("field_enter_after_edit[%s;true]"):format(setting.name .. "_x")
 | |
| 			fs = fs .. ("field_enter_after_edit[%s;true]"):format(setting.name .. "_y")
 | |
| 			fs = fs .. ("field_enter_after_edit[%s;true]"):format(setting.name .. "_z")
 | |
| 			-- for pause menu env
 | |
| 			fs = fs .. ("field_close_on_enter[%s;false]"):format(setting.name .. "_x")
 | |
| 			fs = fs .. ("field_close_on_enter[%s;false]"):format(setting.name .. "_y")
 | |
| 			fs = fs .. ("field_close_on_enter[%s;false]"):format(setting.name .. "_z")
 | |
| 
 | |
| 			fs = fs .. ("button[%f,0.6;1,0.8;%s;%s]"):format(avail_w, "set_" .. setting.name, fgettext("Set"))
 | |
| 
 | |
| 			return fs, 1.4
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields)
 | |
| 			if fields["set_" .. setting.name]  or
 | |
| 					fields.key_enter_field == setting.name .. "_x" or
 | |
| 					fields.key_enter_field == setting.name .. "_y" or
 | |
| 					fields.key_enter_field == setting.name .. "_z" then
 | |
| 				local x = tonumber(fields[setting.name .. "_x"])
 | |
| 				local y = tonumber(fields[setting.name .. "_y"])
 | |
| 				local z = tonumber(fields[setting.name .. "_z"])
 | |
| 				if is_valid_number(x) and is_valid_number(y) and is_valid_number(z) then
 | |
| 					core.settings:set(setting.name, vector.new(x, y, z):to_string())
 | |
| 				else
 | |
| 					core.log("error", "Invalid vector: " .. dump({x, y, z}))
 | |
| 				end
 | |
| 				return true
 | |
| 			end
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| function make.flags(setting)
 | |
| 	local checkboxes = {}
 | |
| 
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			local fs = {
 | |
| 				"label[0,0.1;" .. get_label(setting) .. "]",
 | |
| 			}
 | |
| 
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			checkboxes = {}
 | |
| 			for _, name in ipairs(setting.possible) do
 | |
| 				checkboxes[name] = false
 | |
| 			end
 | |
| 			local function apply_flags(flag_string, what)
 | |
| 				local prefixed_flags = {}
 | |
| 				for _, name in ipairs(flag_string:split(",")) do
 | |
| 					prefixed_flags[name:trim()] = true
 | |
| 				end
 | |
| 				for _, name in ipairs(setting.possible) do
 | |
| 					local enabled = prefixed_flags[name]
 | |
| 					local disabled = prefixed_flags["no" .. name]
 | |
| 					if enabled and disabled then
 | |
| 						core.log("warning", "Flag " .. name .. " in " .. what .. " " ..
 | |
| 								setting.name .. " both enabled and disabled, ignoring")
 | |
| 					elseif enabled then
 | |
| 						checkboxes[name] = true
 | |
| 					elseif disabled then
 | |
| 						checkboxes[name] = false
 | |
| 					end
 | |
| 				end
 | |
| 			end
 | |
| 			-- First apply the default, which is necessary since flags
 | |
| 			-- which are not overridden may be missing from the value.
 | |
| 			apply_flags(setting.default, "default for setting")
 | |
| 			local value = core.settings:get(setting.name)
 | |
| 			if value then
 | |
| 				apply_flags(value, "setting")
 | |
| 			end
 | |
| 
 | |
| 			local columns = math.max(math.floor(avail_w / 2.5), 1)
 | |
| 			local column_width = avail_w / columns
 | |
| 			local x = 0
 | |
| 			local y = 0.55
 | |
| 
 | |
| 			for _, possible in ipairs(setting.possible) do
 | |
| 				if x >= avail_w then
 | |
| 					x = 0
 | |
| 					y = y + 0.5
 | |
| 				end
 | |
| 
 | |
| 				local is_checked = checkboxes[possible]
 | |
| 				fs[#fs + 1] = ("checkbox[%f,%f;%s;%s;%s]"):format(
 | |
| 					x, y, setting.name .. "_" .. possible,
 | |
| 					core.formspec_escape(possible), tostring(is_checked))
 | |
| 				x = x + column_width
 | |
| 			end
 | |
| 
 | |
| 			return table.concat(fs, ""), y + 0.25
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields)
 | |
| 			local changed = false
 | |
| 			for name, _ in pairs(checkboxes) do
 | |
| 				local value = fields[setting.name .. "_" .. name]
 | |
| 				if value ~= nil then
 | |
| 					checkboxes[name] = core.is_yes(value)
 | |
| 					changed = true
 | |
| 				end
 | |
| 			end
 | |
| 
 | |
| 			if changed then
 | |
| 				local values = {}
 | |
| 				for _, name in ipairs(setting.possible) do
 | |
| 					if checkboxes[name] then
 | |
| 						table.insert(values, name)
 | |
| 					else
 | |
| 						table.insert(values, "no" .. name)
 | |
| 					end
 | |
| 				end
 | |
| 
 | |
| 				core.settings:set(setting.name, table.concat(values, ","))
 | |
| 			end
 | |
| 			return changed
 | |
| 		end
 | |
| 	}
 | |
| end
 | |
| 
 | |
| 
 | |
| local function make_noise_params(setting)
 | |
| 	return {
 | |
| 		info_text = setting.comment,
 | |
| 		setting = setting,
 | |
| 
 | |
| 		get_formspec = function(self, avail_w)
 | |
| 			-- The "defaults" noise parameter flag doesn't reset a noise
 | |
| 			-- setting to its default value, so we offer a regular reset button.
 | |
| 			self.resettable = core.settings:has(setting.name)
 | |
| 
 | |
| 			local fs = "label[0,0.4;" .. get_label(setting) .. "]" ..
 | |
| 					("button[%f,0;2.5,0.8;%s;%s]"):format(avail_w - 2.5, "edit_" .. setting.name, fgettext("Edit"))
 | |
| 			return fs, 0.8
 | |
| 		end,
 | |
| 
 | |
| 		on_submit = function(self, fields, tabview)
 | |
| 			if fields["edit_" .. setting.name] then
 | |
| 				local dlg = create_change_mapgen_flags_dlg(setting)
 | |
| 				dlg:set_parent(tabview)
 | |
| 				tabview:hide()
 | |
| 				dlg:show()
 | |
| 
 | |
| 				return true
 | |
| 			end
 | |
| 		end,
 | |
| 	}
 | |
| end
 | |
| 
 | |
| if INIT == "pause_menu" then
 | |
| 	-- Making the noise parameter dialog work in the pause menu settings would
 | |
| 	-- require porting "FSTK" (at least the dialog API) from the mainmenu formspec
 | |
| 	-- API to the in-game formspec API.
 | |
| 	-- There's no reason you'd want to adjust mapgen noise parameter settings
 | |
| 	-- in-game (they only apply to new worlds, hidden as [world_creation]),
 | |
| 	-- so there's no reason to implement this.
 | |
| 	local empty = function()
 | |
| 		return { get_formspec = function() return "", 0 end }
 | |
| 	end
 | |
| 	make.noise_params_2d = empty
 | |
| 	make.noise_params_3d = empty
 | |
| else
 | |
| 	make.noise_params_2d = make_noise_params
 | |
| 	make.noise_params_3d = make_noise_params
 | |
| end
 | |
| 
 | |
| 
 | |
| return make
 |