mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 23:26:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			314 lines
		
	
	
		
			No EOL
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			314 lines
		
	
	
		
			No EOL
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| local ui = require('openmw.ui')
 | |
| local util = require('openmw.util')
 | |
| local async = require('openmw.async')
 | |
| local core = require('openmw.core')
 | |
| local storage = require('openmw.storage')
 | |
| 
 | |
| local common = require('scripts.omw.settings.common')
 | |
| 
 | |
| local renderers = {}
 | |
| local function registerRenderer(name, renderFunction)
 | |
|     renderers[name] = renderFunction
 | |
| end
 | |
| 
 | |
| local pages = {}
 | |
| local groups = {}
 | |
| local pageOptions = {}
 | |
| 
 | |
| local padding = function(size)
 | |
|     return {
 | |
|         props = {
 | |
|             size = util.vector2(size, size),
 | |
|         }
 | |
|     }
 | |
| end
 | |
| local smallPadding = padding(10)
 | |
| local bigPadding = padding(25)
 | |
| 
 | |
| local pageHeader = {
 | |
|     props = {
 | |
|         textColor = util.color.rgb(1, 1, 1),
 | |
|         textSize = 30,
 | |
|     },
 | |
| }
 | |
| local groupHeader = {
 | |
|     props = {
 | |
|         textColor = util.color.rgb(1, 1, 1),
 | |
|         textSize = 25,
 | |
|     },
 | |
| }
 | |
| local normal = {
 | |
|     props = {
 | |
|         textColor = util.color.rgb(1, 1, 1),
 | |
|         textSize = 20,
 | |
|     },
 | |
| }
 | |
| 
 | |
| local function renderSetting(group, setting, value, global)
 | |
|     local renderFunction = renderers[setting.renderer]
 | |
|     if not renderFunction then
 | |
|         error(('Setting %s of %s has unknown renderer %s'):format(setting.key, group.key, setting.renderer))
 | |
|     end
 | |
|     local set = function(value)
 | |
|         if global then
 | |
|             core.sendGlobalEvent(common.setGlobalEvent, {
 | |
|                 groupKey = group.key,
 | |
|                 settingKey = setting.key,
 | |
|                 value = value,
 | |
|             })
 | |
|         else
 | |
|             storage.playerSection(group.key):set(setting.key, value)
 | |
|         end
 | |
|     end
 | |
|     local l10n = core.l10n(group.l10n)
 | |
|     return {
 | |
|         name = setting.key,
 | |
|         type = ui.TYPE.Flex,
 | |
|         content = ui.content {
 | |
|             {
 | |
|                 type = ui.TYPE.Flex,
 | |
|                 props = {
 | |
|                     horizontal = true,
 | |
|                     align = ui.ALIGNMENT.Start,
 | |
|                     arrange = ui.ALIGNMENT.End,
 | |
|                 },
 | |
|                 content = ui.content {
 | |
|                     {
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = normal,
 | |
|                         props = {
 | |
|                             text = l10n(setting.name),
 | |
|                         },
 | |
|                     },
 | |
|                     smallPadding,
 | |
|                     renderFunction(value, set, setting.argument),
 | |
|                     smallPadding,
 | |
|                     {
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = normal,
 | |
|                         props = {
 | |
|                             text = 'Reset',
 | |
|                         },
 | |
|                         events = {
 | |
|                             mouseClick = async:callback(function()
 | |
|                                 set(setting.default)
 | |
|                             end),
 | |
|                         },
 | |
|                     },
 | |
|                 },
 | |
|             },
 | |
|         },
 | |
|     }
 | |
| end
 | |
| 
 | |
| local groupLayoutName = function(key, global)
 | |
|     return ('%s%s'):format(global and 'global_' or 'player_', key)
 | |
| end
 | |
| 
 | |
| local function renderGroup(group, global)
 | |
|     local l10n = core.l10n(group.l10n)
 | |
|     local layout = {
 | |
|         name = groupLayoutName(group.key, global),
 | |
|         type = ui.TYPE.Flex,
 | |
|         content = ui.content {
 | |
|             {
 | |
|                 type = ui.TYPE.Flex,
 | |
|                 props = {
 | |
|                     horizontal = true,
 | |
|                     align = ui.ALIGNMENT.Start,
 | |
|                     arrange = ui.ALIGNMENT.End,
 | |
|                 },
 | |
|                 content = ui.content {
 | |
|                     {
 | |
|                         name = 'name',
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = groupHeader,
 | |
|                         props = {
 | |
|                             text = l10n(group.name),
 | |
|                         },
 | |
|                     },
 | |
|                     smallPadding,
 | |
|                     {
 | |
|                         name = 'description',
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = normal,
 | |
|                         props = {
 | |
|                             text = l10n(group.description),
 | |
|                         },
 | |
|                     },
 | |
|                 },
 | |
|             },
 | |
|             smallPadding,
 | |
|             {
 | |
|                 name = 'settings',
 | |
|                 type = ui.TYPE.Flex,
 | |
|                 content = ui.content{},
 | |
|             },
 | |
|             bigPadding,
 | |
|         },
 | |
|     }
 | |
|     local settingsContent = layout.content.settings.content
 | |
|     local valueSection = common.getSection(global, group.key)
 | |
|     for _, setting in pairs(group.settings) do
 | |
|         settingsContent:add(renderSetting(group, setting, valueSection:get(setting.key), global))
 | |
|     end
 | |
|     return layout
 | |
| end
 | |
| 
 | |
| local function pageGroupComparator(a, b)
 | |
|     return a.order < b.order or (
 | |
|         a.order == b.order and a.key < b.key
 | |
|     )
 | |
| end
 | |
| 
 | |
| local function generateSearchHints(page)
 | |
|     local hints = {}
 | |
|     local l10n = core.l10n(page.l10n)
 | |
|     table.insert(hints, l10n(page.name))
 | |
|     table.insert(hints, l10n(page.description))
 | |
|     local pageGroups = groups[page.key]
 | |
|     for _, pageGroup in pairs(pageGroups) do
 | |
|         local group = common.getSection(pageGroup.global, common.groupSectionKey):get(pageGroup.key)
 | |
|         local l10n = core.l10n(group.l10n)
 | |
|         table.insert(hints, l10n(group.name))
 | |
|         table.insert(hints, l10n(group.description))
 | |
|         for _, setting in pairs(group.settings) do
 | |
|             table.insert(hints, l10n(setting.name))
 | |
|             table.insert(hints, l10n(setting.description))
 | |
|         end
 | |
|     end
 | |
|     return table.concat(hints, ' ')
 | |
| end
 | |
| 
 | |
| local function renderPage(page)
 | |
|     local l10n = core.l10n(page.l10n)
 | |
|     local layout = {
 | |
|         name = page.key,
 | |
|         type = ui.TYPE.Flex,
 | |
|         content = ui.content {
 | |
|             smallPadding,
 | |
|             {
 | |
|                 type = ui.TYPE.Flex,
 | |
|                 props = {
 | |
|                     horizontal = true,
 | |
|                     align = ui.ALIGNMENT.Start,
 | |
|                     arrange = ui.ALIGNMENT.End,
 | |
|                 },
 | |
|                 content = ui.content {
 | |
|                     {
 | |
|                         name = 'name',
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = pageHeader,
 | |
|                         props = {
 | |
|                             text = l10n(page.name),
 | |
|                         },
 | |
|                     },
 | |
|                     smallPadding,
 | |
|                     {
 | |
|                         name = 'description',
 | |
|                         type = ui.TYPE.Text,
 | |
|                         template = normal,
 | |
|                         props = {
 | |
|                             text = l10n(page.description),
 | |
|                         },
 | |
|                     },
 | |
|                 },
 | |
|             },
 | |
|             bigPadding,
 | |
|             {
 | |
|                 name = 'groups',
 | |
|                 type = ui.TYPE.Flex,
 | |
|                 content = ui.content {},
 | |
|             },
 | |
|         },
 | |
|     }
 | |
|     local groupsContent = layout.content.groups.content
 | |
|     local pageGroups = groups[page.key]
 | |
|     local sortedGroups = {}
 | |
|     for i, v in ipairs(pageGroups) do sortedGroups[i] = v end
 | |
|     table.sort(sortedGroups, pageGroupComparator)
 | |
|     for _, pageGroup in ipairs(sortedGroups) do
 | |
|         local group = common.getSection(pageGroup.global, common.groupSectionKey):get(pageGroup.key)
 | |
|         groupsContent:add(renderGroup(group, pageGroup.global))
 | |
|     end
 | |
|     return {
 | |
|         name = l10n(page.name),
 | |
|         element = ui.create(layout),
 | |
|         searchHints = generateSearchHints(page),
 | |
|     }
 | |
| end
 | |
| 
 | |
| local function onSettingChanged(global)
 | |
|     return async:callback(function(groupKey, settingKey)
 | |
|         local group = common.getSection(global, common.groupSectionKey):get(groupKey)
 | |
|         if not pageOptions[group.page] then return end
 | |
| 
 | |
|         local element = pageOptions[group.page].element
 | |
|         local groupLayout = element.layout.content.groups.content[groupLayoutName(group.key, global)]
 | |
|         local settingsLayout = groupLayout.content.settings
 | |
|         local value = common.getSection(global, group.key):get(settingKey)
 | |
|         settingsLayout.content[settingKey] = renderSetting(group, group.settings[settingKey], value, global)
 | |
|         element:update()
 | |
|     end)
 | |
| end
 | |
| local function onGroupRegistered(global, key)
 | |
|     local group = common.getSection(global, common.groupSectionKey):get(key)
 | |
|     groups[group.page] = groups[group.page] or {}
 | |
|     local pageGroup =  {
 | |
|         key = group.key,
 | |
|         global = global,
 | |
|         order = group.order,
 | |
|     }
 | |
|     table.insert(groups[group.page], pageGroup)
 | |
|     common.getSection(global, group.key):subscribe(onSettingChanged(global))
 | |
| 
 | |
|     if not pages[group.page] then return end
 | |
|     local options = renderPage(pages[group.page])
 | |
|     pageOptions[group.page].element:destroy()
 | |
|     for k, v in pairs(options) do
 | |
|         pageOptions[group.page][k] = v
 | |
|     end
 | |
| end
 | |
| local globalGroups = storage.globalSection(common.groupSectionKey)
 | |
| for groupKey in pairs(globalGroups:asTable()) do
 | |
|     onGroupRegistered(true, groupKey)
 | |
| end
 | |
| globalGroups:subscribe(async:callback(function(_, key)
 | |
|     if key then onGroupRegistered(true, key) end
 | |
| end))
 | |
| storage.playerSection(common.groupSectionKey):subscribe(async:callback(function(_, key)
 | |
|     if key then onGroupRegistered(false, key) end
 | |
| end))
 | |
| 
 | |
| local function registerPage(options)
 | |
|     if type(options) ~= 'table' then
 | |
|         error('Page options must be a table')
 | |
|     end
 | |
|     if type(options.key) ~= 'string' then
 | |
|         error('Page must have a key')
 | |
|     end
 | |
|     if type(options.l10n) ~= 'string' then
 | |
|         error('Page must have a localization context')
 | |
|     end
 | |
|     if type(options.name) ~= 'string' then
 | |
|         error('Page must have a name')
 | |
|     end
 | |
|     if type(options.description) ~= 'string' then
 | |
|         error('Page must have a description')
 | |
|     end
 | |
|     local page = {
 | |
|         key = options.key,
 | |
|         l10n = options.l10n,
 | |
|         name = options.name,
 | |
|         description = options.description,
 | |
|     }
 | |
|     pages[page.key] = page
 | |
|     groups[page.key] = groups[page.key] or {}
 | |
|     pageOptions[page.key] = renderPage(page)
 | |
|     ui.registerSettingsPage(pageOptions[page.key])
 | |
| end
 | |
| 
 | |
| return {
 | |
|     registerPage = registerPage,
 | |
|     registerRenderer = registerRenderer,
 | |
| } |