--[[ RavenAcsendant 25MAY2021 Anommaly Mod Configuration Menu (Anomaly MCM) +Modified version of the Anomaly Options menu created by Tronex. It retains all the same features. +Dedicated space for mod options (As of 1_4 this includes easy save game specific options storage) +Dynamicly loads options from mod scripts at main menu load simplifying the process and alowing it to be done before a save is loaded without needing to edit a game script. There by reducing conflicts. +Change History: 1_1 added options table passed by referance 1_2 Fixed a crash that is actualy still part of ui_options 1_3 Added overide controls for option tree/branch text see the tutorial section "Parameters of option Tree" at the end of this sctipt 1_4 Integrated dph-hcl's script for save game specific MCM options. See "Tutorial: Using : dph-hcl's script for save game specific MCM options" 1_5 Pass value to functors 1.6.0 Adopted https://semver.org/spec/v1.0.0.html versioning spec for reasons. Far left list sorted alphabeticaly, except MCM will always be at the top to serve as a landing page. Added logging utility see instruction in mcm_log.script Now pass true to on_option_change callback to make it posible to identify when it is coming from mcm MCM tracks a session ID that only increments when game exe is launched. ui_mcm.get_session_id() Added assert to identify missing val fields as that tends to result in hard to diagnose errors. Added assert to block calling get() while mcm is building the options tree. Added keybind widget includes a simple conflict list and all keybinds list under mcm's tree. Added utility functions to support key binds. Tracking of held status of shift controls and alt keys with single functon to read all three and a no modifier state Functions to help identify single, double and long presses of a key. Provided templated mcm option tables for selecting modifier keys and single/double and long presses that include translation strings. Updated documentation at end of this sctipt for all keybind related features. 1.6.1 Fixed xrdebug 1040 crash. 1.6.2 Added a message for when io.open fails to create a file. Made ui_mcm.script not fail if mcm_log.script is missing. Made debug logging default to false as orginaly intended. 1.6.3 Fixed interferance between doubletap and single press when bonund to same key. 1.6.4 Missed one place in above fix. 1.6.5 Added support for unbound MCM keybinds pressing ESC while binding a key will unbind the key, clicking will still cancel the bind process -1 is the value for unbound unbound keys will not be reported as conflicting added mouse4-8 as valid keybinds. updated mcm_log to 1.0.2 fixing a crash on quit. chnaged the doubletap window to be 200ms. down from 500. this will make mcm keybinds feel more responsive. made it adjustable by increments of 5 special handeling for key wrappers in the keybinds list fixed a bug in dph-hcl's script for save game specific MCM options (HarukaSai) 1.6.6 actually updated mcm_log to 1.0.2 fixing a crash on quit. 1.7.0 Update to Anomaly 1.5.3 earlier versions of anomaly will not be supported due to the stalker games EULA changes. Included support for Catspaw's ui hooks Adds new custom functors "ui_hook_functor" and "on_selection_functor", which respectively pass along UI element handlers and a trap for unsaved value changes. These functors allow for dynamic customizations to MCM's UI elements at the element container level in response to user interactions. See the tutorial section on "UI Functors" for more information. Fixed a typo "dispaly_key" changed to "display_key", legacy name aliased for compatablity. MCM get(id) will no longer read values from axr_options for settings that are not part of the curent options table lacking a defined value type the data returned was always a string, orphaned settings values are also likely to be garbage nil will be returned instead, you need to handel this. Mostly this was an issue when addon A was reading addon B's settings and addon B had been uninstalled these values will still exist in axr_options and can be read directly if needed. Due to degree of code chnages steps have been taken to invalidate any monkey patches of the MCM UI. Tronex 2019/10/12 Anomlous Options Menu Features: - %100 customizable, support for all kind of xml elements - Highlight pending changes - Option description support - Option presets support - Reset button to clear pending changes - Script callback on option changes - Functors capability to excute on apply or init - Precoditions capability to hide/show/execute functors To get an option value inside other scripts, use: ui_mcm.get(parameter) Check the options table here to see the values used See the tutorial at the bottom for adding or modifying options table --]] VERSION = "1.7.0" MCM_DEBUG = axr_main.config:r_value("mcm", "mcm/mcm_log/debug_logging2", 1, false) local enable_debug_prints = false local enable_ui_functors = true -- feature killswitch ------------------------------------------------------------ -- Strings and LTX management ------------------------------------------------------------ local ini_pres = ini_file("presets\\includes.ltx") local ini_loc = ini_file_ex("localization.ltx") local ini_mcm_key = ini_file_ex("mcm_key_localization.ltx") local _opt_ = "/" -- For axr_options.ltx, don't touch! local opt_section = "mcm" -- For axr_options.ltx, don't touch! local opt_str = "ui_mcm_" -- Option name: "ui_mcm_(option ID)" -- Option description: "ui_mcm_(option ID)_desc" local opt_str_menu = "ui_mcm_menu_" -- Option menu: "ui_mcm_menu_(option menu ID)" local opt_str_prst = "ui_mcm_prst_" -- Option preset: "ui_mcm_prst_(option preset ID)" local opt_str_lst = "ui_mcm_lst_" -- List/Radio keys: "ui_mcm_lst_(key)" function cc(path,opt) return (path .. _opt_ .. opt) end ------------------------------------------------------------ -- Utilities ------------------------------------------------------------ local session_id = nil local m_floor, m_ceil, s_find, s_gsub = math.floor, math.ceil, string.find, string.gsub local clr, clamp, round_idp, str_explode = utils_xml.get_color, clamp, round_idp, str_explode local precision = 6 -- allowed number of zeros local width_factor = utils_xml.is_widescreen() and 0.8 or 1 local clr_o = GetARGB(255, 250, 150, 75) local clr_g1 = GetARGB(255, 170, 170, 170) local clr_g2 = GetARGB(255, 200, 200, 200) local clr_w = GetARGB(255, 255, 255, 255) local clr_r = GetARGB(255, 225, 0, 0) local clr_tree = { [1] = GetARGB(255, 180, 180, 180), [2] = GetARGB(255, 180, 180, 180), [3] = GetARGB(255, 180, 180, 180), } local dbg = nil function print_dbg(...) if (not dbg) and MCM_DEBUG and mcm_log then dbg = mcm_log.new("DBG") dbg.enabled = MCM_DEBUG dbg.continuous = true end if dbg then dbg:log(...) end if enable_debug_prints then printf(...) end end function ui_functors_enabled() -- for scripts to explicitly check for support return enable_ui_functors end ------------------------------------------------------------ -- Options ------------------------------------------------------------ options = {} local opt_temp = {} -- stores pending changes local opt_backup = {} -- do a backup of original changes for comparison with pendings local opt_index = {} -- option index by path, so we can locate an option fast without iterating throw the whole options table local opt_val = {} -- option value type by path, to execute proper functions on different type without iterating throw the whole options table local allowed_type = { -- ignore tables from these types in temp tables ["check"] = true, ["list"] = true, ["input"] = true, ["radio_h"] = true, ["radio_v"] = true, ["track"] = true, ["key_bind"] = true, } local key_by_path = {} local paths_by_key = {} local gathering = false AddScriptCallback("mcm_option_change") AddScriptCallback("mcm_option_reset") AddScriptCallback("mcm_option_restore_default") AddScriptCallback("mcm_option_discard") function init_opt_base() print_dbg("MCM options reset.") get_session_id() options = { { id= "mcm" , gr={ {id = "mcm_about", sh=true , gr={ { id= "slide_mcm" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_menu_mcm_about" ,size= {512,50} }, { id= "desc_mcm" ,type= "desc" ,text= "ui_mcm_desc_mcm" ,clr= {255,125,175,200} }, { id= "desc_mcm2" ,type= "desc" ,text= "ui_mcm_desc_mcm2" ,clr= {255,125,175,200} }, --{ id="reset" ,type= "check" ,val= 1 ,def= false , functor = {reload } }, },}, {id = "mcm_kb", gr={ {id = "mcm_kb_main", sh=true ,gr={ { id= "slide_kb" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_menu_mcm_kb" ,size= {512,50} }, { id= "desc_kb" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb" ,clr= {255,125,175,200} }, { id= "desc_kb2" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb2" ,clr= {255,125,175,200} }, {id = "dtaptime2", type = "track", val = 2, def = 200, min = 100, max = 750, step = 5}, {id = "presstime", type = "track", val = 2, def = 1.5, min = 1.2, max = 2, step = 0.01, prec = 2}, },}, {id = "mcm_all_kb", sh=true , gr={ { id= "slide_kb" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_menu_mcm_all_kb" ,size= {512,50} }, { id= "desc_kb" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb_all" ,clr= {255,125,175,200} }, { id= "desc_kb2" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb_all2" ,clr= {255,125,175,200} }, },}, {id = "mcm_kb_conflicts", sh=true , gr={ { id= "slide_kb" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_menu_mcm_kb_conflicts" ,size= {512,50} }, { id= "desc_kb" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb_conflicts" ,clr= {255,125,175,200} }, { id= "desc_kb2" ,type= "desc" ,text= "ui_mcm_mcm_desc_kb_all2" ,clr= {255,125,175,200} }, },}, },}, {id = "mcm_log", sh=true , gr={ { id= "slide_log" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_menu_mcm_log" ,size= {512,50} }, { id= "desc_log" ,type= "desc" ,text= "ui_mcm_mcm_desc_log" ,clr= {255,125,175,200} }, {id = "enable", type = "check", val = 1, def = true}, {id = "numlogs", type = "track", val = 2, def = 2, min = 1, max = 10, step = 1}, {id = "continuous", type = "check", val = 1, def = false}, {id = "savefreq", type = "input", val = 2, def = 1000, min = 100, max = 1000000000000000}, {id = "timestamp", type = "input", val = 2, def = 1000, min = 1, max = 1000000000000000}, {id = "debug_logging", type = "line"}, {id = "debug_logging2", type = "check", val = 1, def = false}, },}, }, } } local kb_meta = {all = {}, conflict = {}} gathering = true gather_options() gathering = false table.sort(options, function(a,b) if a.id == "mcm" then return true end if b.id == "mcm" then return false end return game.translate_string(a.text or opt_str_menu .. a.id) < game.translate_string(b.text or opt_str_menu .. b.id) end) init_opt_coder(kb_meta) table.sort(kb_meta.conflict, function(a,b) return display_key(get(a.kb_path)) < display_key(get(b.kb_path)) end) for i = 1, #kb_meta.all do options[1].gr[2].gr[2].gr[3+i] = kb_meta.all[i] options[1].gr[2].gr[3].gr[3+i] = kb_meta.conflict[i] end init_opt_coder() MCM_DEBUG = get("mcm/mcm_log/debug_logging2") end function init_opt_coder(kb_meta) local keybind_count = 0 ------------------------------------------------------------------------ -- Coding options local function code_option(gr, id, num) local path for i=1,#gr do if allowed_type[gr[i].type] then path = cc(id , gr[i].id) opt_index[path] = cc(num , i) opt_val[path] = assert(gr[i].val, "val is manditory! option path:"..path) --printf("-[%s] | index: %s - type: %s", path, opt_index[path], opt_val[path]) if gr[i].type == "key_bind" and kb_meta then keybind_count = keybind_count + 1 update_conflict(path, get(path)) local temp = dup_table(gr[i]) temp.id = temp.id .. "_"..keybind_count temp.kb_path = path temp.curr = {get,path} temp.hint = temp.hint or id.."_"..gr[i].id temp.functor = {function(p,v) set(p,v) update_conflict(p,v) end, path } if path:find("mcm/key_wrapper/") then temp.precondition = {function() return false end} --temp.precondition = {get, path:sub(1, -).."enable"} end kb_meta.all[#kb_meta.all+1] = dup_table(temp) temp.precondition = {get_conflict, path} kb_meta.conflict[#kb_meta.conflict+1] = dup_table(temp) end end end end local id_1, id_2, id_3 -- Level 1 for i=1,#options do id_1 = options[i].id if options[i].sh then code_option(options[i].gr, id_1, i) else -- Level 2 for ii=1,#options[i].gr do id_2 = options[i].gr[ii].id if options[i].gr[ii].sh then code_option( (options[i].gr[ii].gr), (id_1 .._opt_.. id_2), (i .._opt_.. ii) ) else -- Level 3 for iii=1,#options[i].gr[ii].gr do id_3 = options[i].gr[ii].gr[iii].id if options[i].gr[ii].gr[iii].sh then code_option( (options[i].gr[ii].gr[iii].gr), (id_1 .._opt_.. id_2 .._opt_.. id_3), (i .._opt_.. ii .._opt_.. iii) ) else ---- end end end end end end end function gather_options() -- this is modified from how axr_main loads all the other scripts thanks to Igi for the idea. local ignore = { ["ui_mcm.script"] = true, } local t = {} local size_t = 0 local f = getFS() local flist = f:file_list_open_ex("$game_scripts$",bit_or(FS.FS_ListFiles,FS.FS_RootOnly),"*mcm.script") local f_cnt = flist:Size() for it=0, f_cnt-1 do local file = flist:GetAt(it) local file_name = file:NameShort() --printf("%s size=%s",file_name,file:Size()) if (file:Size() > 0 and ignore[file_name] ~= true) then file_name = file_name:sub(0,file_name:len()-7) if (_G[file_name] and _G[file_name].on_mcm_load) then size_t = size_t + 1 t[size_t] = file_name -- load all scripts first end end end for i=1,#t do --printf("%s.on_game_start()",t[i]) local temp = nil temp, name = _G[ t[i] ].on_mcm_load(options) -- passing options by referance added in AMCM 1.1 if you use this, nil check, emphasise updating MCM in your mod desc or something if temp and name then local collect = false for j=1, #options do if options[j].id == name then collect = j end end if not collect then collection = { id = name , gr = {}} table.insert(collection.gr, temp) table.insert(options, collection) else table.insert(options[collect].gr, temp) end elseif temp then table.insert(options, temp) end end end ------------------------------------------------------------ -- Functors ------------------------------------------------------------ -- Special local function empty_functor() print_dbg("Empty functor called!") end function reload() set("mcm/reset", false) init_opt_base() end -- Preconditions function level_present() return level.present() end function debug_only() return DEV_DEBUG end function for_renderer(...) local rend = {...} local curr_rend = get_console_cmd(0, "renderer") local result = false for i=1,#rend do result = result or curr_rend == rend[i] end return result end --Key bind stuff local function set_conflict(ck) local conflict = nil for k, v in pairs(paths_by_key[ck]) do if k ~= "conflict" then conflict = conflict or conflict == false or false --if it is true it stays true, if false it becomes true if nil it becomes false. 0 entries = nil, 1 entry = false, more than 1 entry = true end end paths_by_key[ck].conflict = conflict print_dbg("set conflict ck:%s con:%s",ck,conflict) end function update_conflict(path, key) key = tonumber(key) local old_key = key_by_path[path] print_dbg("path:%s, key:%s, old:%s", path, key, old_key) if key == old_key then return end key_by_path[path] = key if old_key then paths_by_key[old_key][path] = nil if paths_by_key[old_key].conflict then -- if this key used to have a conflict see if it can be clered set_conflict(old_key) end end if not paths_by_key[key] then paths_by_key[key] = {} end paths_by_key[key][path] = true if not paths_by_key[key].conflict then --if this key didn't have a conflict see if it does now set_conflict(key) end end function get_conflict(path, key) local k = key or key_by_path[path] print_dbg("path:%s, key:%s, k:%s", path, type(key), type(k)) assert(k, "No key found for path") k = tonumber(k) if k == -1 then return false end if dik_to_bind(k) ~= dik_to_bind(1000) then --conflicts with engine keybind print_dbg("bind:%s", dik_to_bind(k)) return true end if key_by_path[path] == k then print_dbg("==") return paths_by_key[k] and paths_by_key[k].conflict end print_dbg("=/= kbp:%s %s, k:%s %s",type(key_by_path[path]), key_by_path[path], k, type(k)) return paths_by_key[k] and (paths_by_key[k].conflict or paths_by_key[k].conflict == false) --limited support for pending changes conflicts. end function display_key(key) local loc = ini_loc:r_value("string_table","language") loc = ini_mcm_key:section_exist(loc) and loc or "eng" return ini_mcm_key:r_value(loc, key,0,"") end dispaly_key = display_key --old typo aliased to prevent breaking mods. kb_mod_radio = "radio_h" kb_mod_list = "list" local tap = {} function double_tap(id, key, multi_tap) local tg = time_continual() tap[key] = tap[key] or {} if not tap[key][id] then --first press, set timer return false tap[key][id] = tg return false end local dtaptime = get("mcm/mcm_kb/mcm_kb_main/dtaptime2") if (tg - tap[key][id] <= dtaptime) then --if inside the dtap window tap[key][id] = multi_tap and tg -- if multi_tap set timer for next tap, else clear it. return true end tap[key][id] = tg -- if we get here we are past the window for a double tap so this is a first press return false end local hold = {} function key_hold(id,key, cycle) local dtaptime = get("mcm/mcm_kb/mcm_kb_main/dtaptime2") local hold_time = get("mcm/mcm_kb/mcm_kb_main/presstime") * dtaptime cycle = cycle or 10000 local tg = time_continual() hold[key] = hold[key] or {} if (not hold[key][id]) or (tg - hold[key][id] > hold_time*1.5) then hold[key][id] = tg elseif (tg - hold[key][id] > hold_time ) then hold[key][id] = tg + cycle * 1000 return true end return false end local function execute_if_not_held(id,key, functor,...) if not (hold[key] and hold[key]["mcm_single_"..id]) then exec(functor,...) end return true end function simple_press(id, key, functor,...) dtap = double_tap("mcm_single_"..id, key, true) local tg = time_continual() hold[key] = hold[key] or {} hold[key]["mcm_single_"..id] = tg local dtaptime = get("mcm/mcm_kb/mcm_kb_main/dtaptime2") if dtap then RemoveTimeEvent("mcm_single_press",id.."_mcm_single_"..key) else CreateTimeEvent("mcm_single_press",id.."_mcm_single_"..key,(dtaptime*1.1)/1000,execute_if_not_held,id, key, functor,...) end end MOD_NONE = true MOD_CTRL = false MOD_SHIFT = false MOD_ALT = false function get_mod_key(val) if val == 1 then return MOD_SHIFT elseif val == 2 then return MOD_CTRL elseif val == 3 then return MOD_ALT end return MOD_NONE end function on_key_release(key) hold[key] = nil if key == DIK_keys.DIK_RCONTROL or key == DIK_keys.DIK_LCONTROL then MOD_CTRL = false elseif key == DIK_keys.DIK_RSHIFT or key == DIK_keys.DIK_LSHIFT then MOD_SHIFT = false elseif key == DIK_keys.DIK_RMENU or key == DIK_keys.DIK_LMENU then MOD_ALT = false elseif key == DIK_keys.DIK_ESCAPE then MOD_CTRL = false MOD_SHIFT = false MOD_ALT = false end MOD_NONE = not(MOD_CTRL or MOD_SHIFT or MOD_ALT) end function on_key_press(key) if key == DIK_keys.DIK_RCONTROL or key == DIK_keys.DIK_LCONTROL then MOD_CTRL = true elseif key == DIK_keys.DIK_RSHIFT or key == DIK_keys.DIK_LSHIFT then MOD_SHIFT = true elseif key == DIK_keys.DIK_RMENU or key == DIK_keys.DIK_LMENU then MOD_ALT = true end MOD_NONE = not(MOD_CTRL or MOD_SHIFT or MOD_ALT) end -- Utilities function is_int(num) return (m_floor(num) == num) end function exec(func,...) if (not func) then return false end return func(...) end function str_opt_explode(id, by_num) local nums = by_num and opt_index[id] or id local t = nums and str_explode(nums, _opt_) or {} if by_num then for i=1,#t do t[i] = tonumber(t[i]) end end return t end function get_opt_table(id) local t = str_opt_explode(id, true) if #t == 0 then return {} end if #t == 1 then return options[t[1]] elseif #t == 2 then return options[t[1]].gr[t[2]] elseif #t == 3 then return options[t[1]].gr[t[2]].gr[t[3]] elseif #t == 4 then return options[t[1]].gr[t[2]].gr[t[3]].gr[t[4]] end end function check_opt_table(id) local t = str_opt_explode(id, true) return #t > 0 end function CTimeTo_mSec(ct) local Y, M, D, h, m, s, ms = 0, 0, 0, 0, 0, 0, 0 Y, M, D, h, m, s, ms = ct:get(Y, M, D, h, m, s, ms) return ((D*24*60*60 + h*60*60 + m*60 + s)*1000 + ms) end local function create_session_id() if not session_id then session_id = axr_main.config:r_value(opt_section, "session_id", 2, 0) if not level.present() then -- if there is a level present then this is an inprogress session, use ID in file. if not compare the stored session start to the curent time and time continual to identify new session local session_start = axr_main.config:r_value(opt_section, "session_start", 2, 0) local now = os.time() * 1000 if (now - session_start) > (time_continual() + 10000) then --ten seconds of slop session_id = session_id + 1 axr_main.config:w_value(opt_section, "session_id", session_id) axr_main.config:w_value(opt_section, "session_start", now) axr_main.config:save() printf("MCM Session ID:%s", session_id) end end end end function get_session_id() create_session_id() return session_id end function is_gathering() return gathering end -------------------- function get(id) assert(not gathering, "ui_mcm.get() cannot be called during script load or on_mcm_load()!") if (#options == 0) then init_opt_base() end if not opt_val[id] then printe("MCM given bad path:%s",id) return end local value = axr_main.config:r_value(opt_section, id, opt_val[id]) if (value ~= nil) then --print_dbg("/Got axr_option [%s] = %s", id, value) return value end -- Write in axr_main if it doesn't exist local v = get_opt_table(id) if v.cmd then if v.val == 0 then value = get_console_cmd(0, v.cmd) elseif v.val == 1 then value = get_console_cmd(1, v.cmd) elseif v.val == 2 then value = get_console_cmd(0, v.cmd) --get_console_cmd(2, v.cmd) value = tonumber(value) if v.min and v.max then value = clamp(value, v.min, v.max) end value = round_idp(value, v.prec or precision) end elseif (type(v.def) == "table") then value = exec(unpack(v.def)) axr_main.config:w_value(opt_section, id, value) axr_main.config:save() else value = v.def axr_main.config:w_value(opt_section, id, value) axr_main.config:save() end --print_dbg("/Got option [%s] = %s", id, value) if (value == nil) then printe("!Found nil option value [%s]", id) end return value end function set(id, value) axr_main.config:w_value(opt_section, id, value) axr_main.config:save() end -------------------- --=========================================================== --//////////////////////// OPTIONS ////////////////////////// --=========================================================== UIMCM = {} --old monkey patch black hole class "UI_MCM" (CUIScriptWnd) function UI_MCM:__init() super() self.last_tree = {} self.last_path = nil self.last_curr_tree = nil self._Cap = {} self._Check = {} self._List = {} self._Input = {} self._Track = {} self._Radio = {} self._Hint = {} -- Prepare the options table if (#options == 0) then init_opt_base() end printf("MCM init") self:InitControls() self:InitCallBacks() self:Reset() end function UI_MCM:__finalize() end function UI_MCM:InitControls() self:SetWndRect (Frect():set(0,0,1024,768)) self:Enable (true) self.xml = CScriptXmlInit() local xml = self.xml xml:ParseFile ("ui_mcm.xml") self.background = xml:InitStatic("background", self) self.dialog = xml:InitStatic("main", self) xml:InitStatic("main:frame", self.dialog) -- Buttons self.btn_accept = xml:Init3tButton("main:btn_accept", self.dialog) self:Register(self.btn_accept, "btn_accept") self.btn_reset = xml:Init3tButton("main:btn_reset", self.dialog) self:Register(self.btn_reset, "btn_reset") self.btn_default = xml:Init3tButton("main:btn_default", self.dialog) self:Register(self.btn_default, "btn_default") self.btn_cancel = xml:Init3tButton("main:btn_cancel", self.dialog) self:Register(self.btn_cancel, "btn_cancel") -- Pending text --xml:InitFrame("main:notify_frame", self.dialog) self.pending = xml:InitTextWnd("main:notify", self.dialog) -- Options lists self.tree = {} self.bl = {} -- Options showcase self.scroll_opt = xml:InitScrollView("main:scroll", self.dialog) -- Presets self.preset_cap = xml:InitStatic("main:cap_preset", self.dialog) self.preset = xml:InitComboBox("main:preset",self.dialog) self:Register(self.preset, "preset") self.preset:Show(false) self.preset_cap:Show(false) -- Message box self.message_box = CUIMessageBoxEx() self:Register (self.message_box, "mb") -- Hint Window self.hint_wnd = utils_ui.UIHint(self) --MCM Key Bind self.k_binder = nil self.k_timer = 0 self.key_input = nil end function UI_MCM:InitCallBacks() self:AddCallback("btn_accept", ui_events.BUTTON_CLICKED, self.OnButton_Accept, self) self:AddCallback("btn_reset", ui_events.BUTTON_CLICKED, self.OnButton_Reset, self) self:AddCallback("btn_default", ui_events.BUTTON_CLICKED, self.OnButton_Default, self) self:AddCallback("btn_cancel", ui_events.BUTTON_CLICKED, self.OnButton_Cancel, self) self:AddCallback("preset", ui_events.LIST_ITEM_SELECT, self.Callback_Preset, self) self:AddCallback("mb", ui_events.MESSAGE_BOX_YES_CLICKED, self.On_Discard, self) --self:AddCallback("mb", ui_events.MESSAGE_BOX_NO_CLICKED, self.On_Discard,self) end function UI_MCM:Update() CUIScriptWnd.Update(self) -- Show hint on hover for id,ctrl in pairs(self._Cap) do if ctrl:IsCursorOverWindow() then local str = opt_str .. (self._Hint[id] or id) .. "_desc" local str_t = game.translate_string(str) if (str ~= str_t) then self.hint_wnd:Update(str_t) end return end end self.hint_wnd:Update() -- Hack to simulate tracing method for TrackBar value changes. TODO: add callback support for CUITrackBar in engine, this is just silly for id,e in pairs(self._Track) do if e.ctrl:IsCursorOverWindow() then local v = self:GetOption(id) local value = round_idp(e.ctrl:GetFValue(), v.prec or precision) if (value ~= e.value) then e.value = value self:Callback_Track(e.txt, e.path, e.opt, v, value) return end end end end function UI_MCM:Reset() -- Clear all trees for i=1,3 do if self.tree[i] then if type(self.tree[i]) == table then for j=1,#self.tree[i] do self.tree[i][j]:Clear() end else self.tree[i]:Clear() end end end self.k_binder = nil self.k_timer = 0 self.key_input = nil self:Register_Tree(1, "", options, 1) end function UI_MCM:Reset_opt(curr_tree, path, flags) flags = flags or {} local xml = self.xml self.scroll_opt:Clear() -- If options tree has a precondition that must be met, don't show it if it returns false if curr_tree.precondition and (not exec(unpack(curr_tree.precondition))) then if curr_tree.output then local _txt = xml:InitTextWnd("elements:block", nil) _txt:SetText( game.translate_string(curr_tree.output) ) self.scroll_opt:AddWindow(_txt, true) _txt:SetAutoDelete(false) end else self.k_binder = nil self.k_timer = 0 self.key_input = nil -- Presets self:Register_Preset(curr_tree) if curr_tree.apply_to_all and curr_tree.id_gr then flags.apply_to_all = true flags.group = curr_tree.id_gr else flags.apply_to_all = nil end empty_table(self._Cap) empty_table(self._Check) empty_table(self._List) empty_table(self._Input) empty_table(self._Track) empty_table(self._Radio) empty_table(self._Hint) for i=1,#curr_tree.gr do -- Check preconditions local to_hide = curr_tree.gr[i].precondition and (not exec(unpack(curr_tree.gr[i].precondition))) for j=1,10 do -- support for 10 preconditions if (not curr_tree.gr[i]["precondition_" .. j]) then break elseif (not exec(unpack(curr_tree.gr[i]["precondition_" .. j]))) then to_hide = true break end end if (not to_hide) then local opt = curr_tree.gr[i].id local v = curr_tree.gr[i] local _st = xml:InitStatic("main:st", nil) local _h = 0 ----------- Support if (v.type == "line") then _h = self:Register_Line(xml, _st, v) elseif (v.type == "image") then _h = self:Register_Image(xml, _st, v) elseif (v.type == "slide") then _h = self:Register_Slide(xml, _st, v) elseif (v.type == "title") then _h = self:Register_Title(xml, _st, v) elseif (v.type == "desc") then _h = self:Register_Desc(xml, _st, v) ----------- Option elseif (v.type == "check") then _h = self:Register_Check(xml, _st, path, opt, v, flags) elseif (v.type == "button") then _h = self:Register_Button(xml, _st, path, opt, v, flags) elseif (v.type == "list") then _h = self:Register_List(xml, _st, path, opt, v, flags) elseif (v.type == "input") then _h = self:Register_Input(xml, _st, path, opt, v, flags) elseif (v.type == "track") then _h = self:Register_Track(xml, _st, path, opt, v, flags) elseif (v.type == "radio_h") then _h = self:Register_Radio(xml, _st, path, opt, v, true, flags) elseif (v.type == "radio_v") then _h = self:Register_Radio(xml, _st, path, opt, v, false, flags) elseif (v.type == "key_bind") then _h = self:Register_Key_Bind(xml, _st, path, opt, v, flags) end _st:SetWndSize(vector2():set(_st:GetWidth(), _h + 10)) self.scroll_opt:AddWindow(_st, true) _st:SetAutoDelete(true) end end if self.Save_AXR then self.Save_AXR = false axr_main.config:save() end end end function UI_MCM:Reset_last_opt() Register_UI("UI_MCM") if self.last_curr_tree and self.last_path then if self.last_path == "mcm/mcm_kb/mcm_conflicts" then self.last_curr_tree = options[1].gr[2].gr[3] end self:Reset_opt(self.last_curr_tree, self.last_path) self:UpdatePending() end end ------------------------------------------------------------ -- Elements ------------------------------------------------------------ function UI_MCM:Init_Wrapper_Box(xml, anchor, w, h, posx, posy) if not (xml and anchor) then return end wrapbox = xml:InitStatic("elements:image", anchor) if not wrapbox then return end w = w or anchor:GetWidth() h = h or anchor:GetHeight() posx = posx and (type(posx) == "number") and posx or 0 posy = posy and (type(posy) == "number") and posy or 0 wrapbox:SetWndSize(vector2():set(w,h)) pos = wrapbox:GetWndPos() wrapbox:SetWndPos(vector2():set(pos.x + posx, pos.y + posy)) return wrapbox end function UI_MCM:Register_Cap(xml, handler, id, hint) id = s_gsub(id, _opt_, "_") self._Cap[id] = xml:InitStatic("elements:cap",handler) self._Cap[id]:TextControl():SetText( game.translate_string(opt_str .. (hint or id)) ) self._Hint[id] = hint return self._Cap[id]:GetHeight() end function UI_MCM:Register_Line(xml, handler, v) -- v parameter added to support ui_hook_functor local line = xml:InitStatic("elements:line",handler) if enable_ui_functors and v and v.ui_hook_functor then -- Test for existence first since older versions of MCM won't be passing v to this function local wrapbox = self:Init_Wrapper_Box(xml,handler,line:GetWidth(),line:GetHeight() + 10,-10) local handlers = { line = line, -- handler for the line element } local flags = { etype = "line", } ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return (line:GetHeight() + 10) end function UI_MCM:Register_Image(xml, handler, v) local pic = xml:InitStatic("elements:image",handler) if v.link then if (v.pos) then local pos = pic:GetWndPos() pic:SetWndPos(vector2():set( pos.x + v.pos[1] , pos.y + v.pos[2] )) end if (v.size) then pic:SetWndSize(vector2():set( v.size[1] , v.size[2] )) end pic:InitTexture(v.link) pic:SetStretchTexture(v.stretch and true or false) end if enable_ui_functors and v.ui_hook_functor then local wrapbox = self:Init_Wrapper_Box(xml,handler,pic:GetWidth(),pic:GetHeight(),-10) local handlers = { pic = pic, -- Handler for the image element } local flags = { etype = "image", } ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return pic:GetHeight() end function UI_MCM:Register_Slide(xml, handler, v) local frame = xml:InitStatic("elements:slide", handler) local _pos = frame:GetWndPos() frame:SetWndPos(vector2():set( _pos.x , _pos.y + (v.spacing or 20) )) local pic = xml:InitStatic("elements:slide:pic", frame) if v.link then pic:InitTexture(v.link) pic:SetStretchTexture(true) if (v.pos) then local pos = pic:GetWndPos() pic:SetWndPos(vector2():set( pos.x + v.pos[1] , pos.y + v.pos[2] )) end if (v.size) then pic:SetWndSize(vector2():set( v.size[1] * width_factor , v.size[2] )) end pic:InitTexture(v.link) end local txt = xml:InitTextWnd("elements:slide:txt", frame) if v.text then txt:SetText( game.translate_string(v.text) ) end if not v.borderless then xml:InitStatic("elements:slide:line_1", frame) xml:InitStatic("elements:slide:line_2", frame) end if enable_ui_functors and v.ui_hook_functor then local wrapbox = self:Init_Wrapper_Box(xml,handler,pic:GetWidth(),pic:GetHeight() + 20,-10) local handlers = { pic = pic, -- handler for the image element txt = txt, -- handler for the text element } local flags = { etype = "slide", } ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return (pic:GetHeight() + 20) end function UI_MCM:Register_Title(xml, handler, v) local title = xml:InitTextWnd("elements:title_" .. (v.align or "l"), handler) title:SetText( game.translate_string(v.text) ) title:AdjustHeightToText() title:SetWndSize(vector2():set(title:GetWidth(), title:GetHeight() + 20)) if v.clr and v.clr[4] then title:SetTextColor( GetARGB(v.clr[1], v.clr[2], v.clr[3], v.clr[4]) ) end if enable_ui_functors and v.ui_hook_functor then local wrapbox = self:Init_Wrapper_Box(xml,handler,title:GetWidth(),title:GetHeight(),-10) local handlers = { title = title, -- Handler for the title element } local flags = { etype = "title", } ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return title:GetHeight() end function UI_MCM:Register_Desc(xml, handler, v) local desc = xml:InitTextWnd("elements:desc", handler) desc:SetText( game.translate_string(v.text) ) desc:AdjustHeightToText() desc:SetWndSize(vector2():set(desc:GetWidth(), desc:GetHeight() + 20)) if v.clr and v.clr[4] then desc:SetTextColor( GetARGB(v.clr[1], v.clr[2], v.clr[3], v.clr[4]) ) end if enable_ui_functors and v.ui_hook_functor then local wrapbox = self:Init_Wrapper_Box(xml,handler,desc:GetWidth(),desc:GetHeight(),-10) local handlers = { desc = desc, -- Handler for the text description element } local flags = { etype = "desc", } ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return desc:GetHeight() end function UI_MCM:Register_Check(xml, handler, path, opt, v, flags) local id = cc(path , opt) -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end -- Create control local ctrl = xml:InitCheck("elements:check",handler) if (ctrl:GetHeight() > h) then h = ctrl:GetHeight() end -- Get values local value = self:GetValue(path, opt, v, flags) ctrl:SetCheck(value) -- Register local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Check(ctrl, path, opt, v) end self:AddCallback(id_ctrl, ui_events.BUTTON_CLICKED, _wrapper, self) if enable_ui_functors and v.ui_hook_functor then local id_cap = s_gsub(id, _opt_, "_") local cap = self._Cap[id_cap] local wrapbox = self:Init_Wrapper_Box(xml,handler,handler:GetWidth(),h,-10) local handlers = { ctrl = ctrl, -- handler for the checkbox control element cap = cap, -- handler for text caption element } flags.etype = "check" flags.path = path -- MCM menu path flags.opt = opt -- MCM option ID ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return h end function UI_MCM:Callback_Check(ctrl, path, opt, v) local value = ctrl:GetCheck() self:CacheValue(path, opt, value, v) end function UI_MCM:Register_Key_Bind(xml, handler, path, opt, v, flags) local id = cc(path , opt) self.k_binder = nil self.k_timer = 0 self.key_input = nil --[[ Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end --]] xml:InitFrame("elements:frame_key_button", handler) local txt = xml:InitStatic("elements:txt_key_button", handler) -- Create control local ctrl = xml:Init3tButton("elements:btn_key_button", handler) local h = ctrl:GetHeight() -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Get values local value = self:GetValue(path, opt, v, flags) txt:TextControl():SetText(display_key(value)) if get_conflict(v.kb_path or id, value) then txt:TextControl():SetTextColor(clr_r) else txt:TextControl():SetTextColor(clr_g1) end -- Register local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Key_Bind(ctrl, path, opt, v, txt) end self:AddCallback(id_ctrl, ui_events.BUTTON_CLICKED, _wrapper, self) return h end function UI_MCM:Callback_Key_Bind(ctrl, path, opt, v, txt) self.key_input = txt:TextControl() self.key_input:SetText( "") txt:TextControl():SetTextColor(clr_w) if self.k_binder then self.k_binder() -- clear other bind inputs end self.k_timer = time_continual() self.k_binder = function(key) self:Key_Binder(ctrl, path, opt, v, key, txt) end end function UI_MCM:Key_Binder(ctrl, path, opt, v, key, txt) self.key_input = nil self.k_binder = nil local value if key then self:CacheValue(path, opt, key, v) value = key else value = self:GetValue(path, opt, v, flags) end txt:TextControl():SetText(display_key(value)) if get_conflict(v.kb_path or cc(path , opt), value) then txt:TextControl():SetTextColor(clr_r) else txt:TextControl():SetTextColor(clr_g1) end end function UI_MCM:Register_Button(xml, handler, path, opt, v, flags) local id = cc(path , opt) --[[ Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end --]] xml:InitFrame("elements:frame_button", handler) -- Create control local ctrl = xml:Init3tButton("elements:btn_button", handler) local h = ctrl:GetHeight() -- Caption local id_cap = s_gsub(id, _opt_, "_") self._Cap[id_cap] = xml:InitStatic("elements:cap_button",handler) self._Cap[id_cap]:TextControl():SetText( game.translate_string(opt_str .. (v.hint or id_cap)) ) if (self._Cap[id_cap]:GetHeight() > h) then h = self._Cap[id_cap]:GetHeight() end -- Register local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Button(ctrl, path, opt, v) end self:AddCallback(id_ctrl, ui_events.BUTTON_CLICKED, _wrapper, self) return h end function UI_MCM:Callback_Button(ctrl, path, opt, v) if v.functor_ui then local id = cc(path , opt) print_dbg("- Executing functor_ui of [%s]",id) exec(unpack(v.functor_ui),self) end if v.functor then local id = cc(path , opt) print_dbg("- Executing functor of [%s]",id) exec(unpack(v.functor)) end end function UI_MCM:Register_List(xml, handler, path, opt, v, flags) local id = cc(path , opt) -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end -- Create control local ctrl = xml:InitComboBox("elements:list",handler) if (ctrl:GetHeight() > h) then --h = ctrl:GetHeight() end -- Get values local idx local value = self:GetValue(path, opt, v, flags) local content = self:GetContent(path, opt, v) -- Setup for i=1,#content do local str_2 = content[i][2] or tostring(content[i][1]) local str = v.no_str and str_2 or game.translate_string(opt_str_lst .. str_2) ctrl:AddItem( game.translate_string(str), i) if content[i][1] == value then idx = i end end idx = idx or 1 --printf(path.." ".. idx) local str_2 = content[idx][2] or tostring(content[idx][1]) local str = v.no_str and str_2 or game.translate_string(opt_str_lst .. str_2) ctrl:enable_id( idx ) ctrl:SetText( game.translate_string(str) ) -- Register local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_List(ctrl, path, opt, v) end self:AddCallback(id_ctrl, ui_events.LIST_ITEM_SELECT, _wrapper, self) if enable_ui_functors and v.ui_hook_functor then local id_cap = s_gsub(id, _opt_, "_") local cap = self._Cap[id_cap] local wrapbox = self:Init_Wrapper_Box(xml,handler,handler:GetWidth(),ctrl:GetHeight(),-10) local handlers = { ctrl = ctrl, -- handler for the list control element cap = cap, -- handler for text caption element } flags.etype = "list" flags.path = path -- MCM menu path flags.opt = opt -- MCM option ID ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return h end function UI_MCM:Callback_List(ctrl, path, opt, v) local i = ctrl:CurrentID() local content = self:GetContent(path, opt, v) self:CacheValue(path, opt, content[i][1], v) end function UI_MCM:Register_Input(xml, handler, path, opt, v, flags) local id = cc(path , opt) -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end -- Create control local ctrl = xml:InitEditBox("elements:input",handler) if (ctrl:GetHeight() > h) then h = ctrl:GetHeight() end -- Get values local value = self:GetValue(path, opt, v, flags) ctrl:SetText(value) -- Register local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Input(ctrl, path, opt, v) end self:AddCallback(id_ctrl, ui_events.EDIT_TEXT_COMMIT, _wrapper, self) if enable_ui_functors and v.ui_hook_functor then local id_cap = s_gsub(id, _opt_, "_") local cap = self._Cap[id_cap] local wrapbox = self:Init_Wrapper_Box(xml,handler,handler:GetWidth(),h,-10) local handlers = { ctrl = ctrl, -- handler for the input control element cap = cap, -- handler for text caption element } flags.etype = "input" flags.path = path -- MCM menu path flags.opt = opt -- MCM option ID ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return h end function UI_MCM:Callback_Input(ctrl, path, opt, v) local value = ctrl:GetText() if not (value and value ~= "") then ctrl:SetText( self:GetCurrentValue(path, opt, v) or self:GetDefaultValue(path, opt, v) ) return end if (v.val == 2) then value = tonumber(value) if (not value) then ctrl:SetText( self:GetCurrentValue(path, opt, v) or self:GetDefaultValue(path, opt, v) ) return end value = clamp(value, v.min, v.max) end self:CacheValue(path, opt, value, v) ctrl:SetText(value) end function UI_MCM:Register_Track(xml, handler, path, opt, v, flags) local id = cc(path , opt) -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end -- Create control self._Track[id] = {} self._Track[id].ctrl = xml:InitTrackBar("elements:track",handler) self._Track[id].path = path self._Track[id].opt = opt if (self._Track[id].ctrl:GetHeight() > h) then h = self._Track[id].ctrl:GetHeight() end self._Track[id].txt = xml:InitTextWnd("elements:track_value",handler) -- Get values local value = self:GetValue(path, opt, v, flags) value = clamp(value, v.min, v.max) value = round_idp(value, v.prec or precision) local int = false --is_int(value) and is_int(v.step) and is_int(v.min) and is_int(v.max) self._Track[id].value = value -- temp self._Track[id].ctrl:SetInvert(v.invert and true or false) self._Track[id].ctrl:SetStep(v.step) if int then self._Track[id].ctrl:SetOptIBounds(v.min, v.max) self._Track[id].ctrl:SetIValue(value) else self._Track[id].ctrl:SetOptFBounds(v.min, v.max) self._Track[id].ctrl:SetFValue(value) end if (not v.no_str) then self._Track[id].txt:SetText(value) end if enable_ui_functors and v.ui_hook_functor then local id_cap = s_gsub(id, _opt_, "_") local cap = self._Cap[id_cap] local wrapbox = self:Init_Wrapper_Box(xml,handler,10,10,-10) local ctrl = self._Track[id].ctrl local txt = self._Track[id].txt local handlers = { ctrl = ctrl, -- handler for input control txt = txt, -- handler for current value display text cap = cap, -- handler for text caption element } flags.etype = "track" flags.path = path -- MCM menu path flags.opt = opt -- MCM option ID ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,handlers,v,flags) end return h end function UI_MCM:Callback_Track(ctrl, path, opt, v, value) if (not v.no_str) then ctrl:SetText(value) end self:CacheValue(path, opt, value, v) end function UI_MCM:Register_Radio(xml, handler, path, opt, v, typ, flags) local id = cc(path , opt) -- Caption local h = self:Register_Cap(xml, handler, id, v.hint) -- Apply to all button if flags.apply_to_all and flags.group then self:Register_BtnAll(xml, handler, path, opt, v, flags) end -- Determine type local str = typ and "horz" or "vert" local content = self:GetContent(path, opt, v) local num = #content if num > 8 and (not v.force_horz) then typ = false str = "vert" end -- Create control local frame = xml:InitStatic("elements:radio_" .. str, handler) local ctrl = {} local txttbl = {} local txt local offset = typ and m_floor(frame:GetWidth()/num) or 30 local h_factor = typ and 1 or 0 local v_factor = typ and 0 or 1 local h1, h2 = 0, 0 --printf("offset: %s - h_factor: %s - v_factor: %s - num: %s", offset, h_factor, v_factor, num) for i=1,num do -- Buttons ctrl[i] = xml:InitCheck("elements:radio_" .. str .. ":btn", frame) local pos = ctrl[i]:GetWndPos() h1 = (h1 * v_factor) + ctrl[i]:GetHeight() ctrl[i]:SetWndPos(vector2():set( pos.x + ((i-1) * offset * h_factor) , pos.y + ((i-1) * offset * v_factor) )) -- Text txt = xml:InitTextWnd("elements:radio_" .. str .. ":txt", frame) local pos2 = txt:GetWndPos() h2 = h_factor * txt:GetHeight() txt:SetWndPos(vector2():set( pos2.x + ((i-1) * offset * h_factor) , pos2.y - (v_factor * 30) + ((i-1) * offset * v_factor) )) local str_2 = content[i][2] or tostring(content[i][1]) local str = v.no_str and game.translate_string(str_2) or game.translate_string(opt_str_lst .. str_2) txt:SetText( str ) txttbl[i] = txt if (h1 + h2 > h) then h = h1 + h2 end end -- Get values local value = self:GetValue(path, opt, v, flags) local id_ctrl = self:Stacker(path, opt, v) for i=1,num do if (content[i][1] == value) then ctrl[i]:SetCheck(true) else ctrl[i]:SetCheck(false) end -- Register self:Register(ctrl[i], id_ctrl .. i) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Radio(ctrl, path, opt, v, i) end self:AddCallback(id_ctrl .. i, ui_events.BUTTON_CLICKED, _wrapper, self) end if enable_ui_functors and v.ui_hook_functor then local id_cap = s_gsub(id, _opt_, "_") local cap = self._Cap[id_cap] local wrapbox = self:Init_Wrapper_Box(xml,handler,handler:GetWidth(),h,-10) local ctrltbl = ctrl local handlers = { ctrltbl = ctrltbl, -- TABLE of radio options from 1 to (flags.num_opts) txttbl = txttbl, -- TABLE of handlers for the radio options cap = cap, -- handler for text caption element } flags.etype = "radio" flags.num_opts = num -- number of options flags.hvtype = str -- "horz" or "vert" flags.path = path -- MCM menu path flags.opt = opt -- MCM option ID ui_mcm.exec(unpack(v.ui_hook_functor),wrapbox,ctrltbl,v,flags) end return h end function UI_MCM:Callback_Radio(ctrl, path, opt, v, n) local value = ctrl[n]:GetCheck() --printf("n = %s", n) if value then for i=1,#ctrl do if i ~= n then ctrl[i]:SetCheck(false) end end local content = self:GetContent(path, opt, v) self:CacheValue(path, opt, content[n][1], v) else ctrl[n]:SetCheck(true) end end function UI_MCM:Register_BtnAll(xml, handler, path, opt, v, flags) local ctrl = xml:Init3tButton("elements:btn_all",handler) xml:InitStatic("elements:cap_all",handler) local id_ctrl = self:Stacker(path, opt, v) self:Register(ctrl, id_ctrl) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_BtnAll(ctrl, path, opt, v, flags) end self:AddCallback(id_ctrl, ui_events.BUTTON_CLICKED, _wrapper, self) end function UI_MCM:Callback_BtnAll(ctrl, path, opt, v, flags) local id = cc(path , opt) local group = flags.group local value = self:GetValue(path, opt, v, flags) -- Set same value for identical options of same group local function set_in_group(p, group, path, opt, value) print_dbg("~set_in_group | current path: %s - target opt: %s", path, opt) for i=1,#p do local path_ext = path and (path ~= "") and cc(path , p[i].id) or p[i].id if p[i].sh then print_dbg("~set_in_group | current path: %s - target opt: %s", path_ext, opt) if (p[i].id_gr == group) then local gr = p[i].gr for j=1,#gr do if gr[j].id == opt then local id_ext = cc(path_ext , opt) if check_opt_table(id_ext) then print_dbg("-set_in_group | Found match: %s", id_ext) self:CacheValue(path_ext, opt, value, gr[j]) end end end end else set_in_group(p[i].gr, group, path_ext, opt, value) end end end set_in_group(options, group, "", opt, value) end function UI_MCM:Register_Preset(ct) if ct.presets then self.preset:ClearList() -- for i=1,#ct.presets do self.preset:AddItem( game.translate_string(opt_str_prst .. ct.presets[i]), i) end if (ct.curr_preset) then self.preset:SetText( game.translate_string(opt_str_prst .. ct.curr_preset) ) end self.preset:Show(true) self.preset_cap:Show(true) else self.preset:ClearList() self.preset:Show(false) self.preset_cap:Show(false) end end function UI_MCM:Callback_Preset() if not (self.last_curr_tree and self.last_path) then return end local txt = self.preset:GetText() if not (txt and txt ~= "") then return end -- Retrieve the preset section local pres local presets = self.last_curr_tree.presets for i=1,#presets do if game.translate_string(opt_str_prst .. presets[i]) == txt then pres = presets[i] break end end if pres and ini_pres:section_exist(pres) then self.last_curr_tree.curr_preset = pres --self:Reset_opt(self.last_curr_tree, self.last_path, { preset = pres }) local n = ini_pres:line_count(pres) local result, id, value for i=0,n-1 do result, id, value = ini_pres:r_line_ex(pres,i,"","") -- Validate option local v = get_opt_table(id) if v and v.type then -- No need to modify options that can't be seen local to_hide = v.precondition and (not exec(unpack(v.precondition))) if (not to_hide) then -- Get proper value if v.val == 0 then elseif v.val == 1 then value = (value == "true") and true or false elseif v.val == 2 then value = tonumber(value) end -- Extract path and opt local t = str_opt_explode(id) local opt = t[#t] local path = t[1] for i=2,#t-1 do path = cc(path , t[i]) end -- Cache changes self:CacheValue(path, opt, value, v) end end end -- Update XML elements self:Reset_opt(self.last_curr_tree, self.last_path) -- Update state self:UpdatePending() end end function UI_MCM:Register_Tree(tr, path, group, idx) print_dbg("-Register_Tree | tr: %s - path: %s", tr, path) local xml = self.xml self.key_input = nil self.k_binder = nil if (not self.tree[tr]) then self.tree[tr] = {} end if (not self.tree[tr][path]) then self.tree[tr][path] = xml:InitScrollView("main:tree_" .. tr, self.dialog) --[[ local pos = self.tree[tr][path]:GetWndPos() if tr == 3 then idx = 1 end self.tree[tr][path]:SetWndPos(vector2():set( pos.x , pos.y + (25*(idx-1)) )) --]] if (not self.bl[tr]) then self.bl[tr] = {} end self.bl[tr][path] = {} -- Fill tree for i=1,#group do local _st = xml:InitStatic("main:st_tree", nil) self.bl[tr][path][i] = xml:InitCheck("elements:btn_list", _st) self.bl[tr][path][i]:SetCheck(false) local txt = xml:InitTextWnd("elements:txt_list", _st) txt:SetText( game.translate_string(group[i].text or opt_str_menu .. group[i].id) ) txt:SetTextColor( clr_tree[tr] ) self.tree[tr][path]:AddWindow(_st, true) _st:SetAutoDelete(false) end -- Set Callback for tree buttons for i=1,#self.bl[tr][path] do local path_i = (path ~= "") and cc(path , group[i].id) or group[i].id self:Register(self.bl[tr][path][i], ("tree_"..path_i)) local _wrapper = function(handler) -- we need wrapper in order to pass ctrl to method self:Callback_Tree(tr, path_i, group[i], self.bl[tr][path], i) end self:AddCallback(("tree_"..path_i), ui_events.BUTTON_CLICKED, _wrapper, self) end end self.tree[tr][path]:Show(true) self.bl[tr][path][1]:SetCheck(true) local path_1 = (path ~= "") and cc(path , group[1].id) or group[1].id self:Callback_Tree(tr, path_1, group[1], self.bl[tr][path], 1) end function UI_MCM:Callback_Tree(tr, path, group, ctrl, i) print_dbg("-Callback_Tree | tr: %s - path: %s - index: %s", tr, path, i) self.key_input = nil self.k_binder = nil -- Radio buttons behavior if (ctrl[i]:GetCheck() == false) then ctrl[i]:SetCheck(true) return end for k=1,#ctrl do if k ~= i then ctrl[k]:SetCheck(false) end end -- Hide all sub trees for k=tr+1,#self.tree do for _,v in pairs(self.tree[k]) do v:Show(false) end end -- If its an option list, show it if group.sh then self:Reset_opt(group, path) -- Caching current options self.last_path = path if (not self.last_curr_tree) then self.last_curr_tree = {} end empty_table(self.last_curr_tree) copy_table(self.last_curr_tree, group) else self:Register_Tree(tr+1, path, group.gr, i) end end ------------------------------------------------------------ -- Utilities ------------------------------------------------------------ function UI_MCM:GetValue(path, opt, v, flags) -- NOTE: make sure to check for nil values only, since false exists as legit value for commands and check boxes local value if flags and flags.def then value = self:GetDefaultValue(path, opt, v) elseif flags and flags.preset then local pres = flags.preset local id = cc(path , opt) if v.val == 0 then value = ini_pres:r_string_ex(pres, id) elseif v.val == 1 then value = ini_pres:r_bool_ex(pres, id) elseif v.val == 2 then value = ini_pres:r_float_ex(pres, id) end end if (value ~= nil) or (flags and flags.def) then if (value ~= nil) and (v.type == "track") then value = clamp(value, v.min, v.max) end self:CacheValue(path, opt, value, v) end if (value == nil) then value = self:GetCurrentValue(path, opt, v) end if (value == nil) then value = self:GetDefaultValue(path, opt, v) end if (value ~= nil) and (v.type == "track") then value = clamp(value, v.min, v.max) end return value end function UI_MCM:GetDefaultValue(path, opt, v) local id = cc(path , opt) local value if (type(v.def) == "table") then value = exec(unpack(v.def)) else value = v.def end -- We cache default values for the first time, so current values rely on them up later -- because some default values are randomized and player might not touch them, thus causing randomized effects where they are used in-game if (axr_main.config:r_value(opt_section, id, v.val) == nil) and (value ~= nil) then axr_main.config:w_value(opt_section, id, value) self.Save_AXR = true end return value end function UI_MCM:GetCurrentValue(path, opt, v) local id = cc(path , opt) if (opt_temp[id] ~= nil) then local _id = s_gsub(id, _opt_, "_") if self._Cap[_id] and self._Cap[_id]:IsShown() then self._Cap[_id]:TextControl():SetTextColor( clr_o ) end return opt_temp[id] end local value if v.curr then value = exec(unpack(v.curr)) elseif v.cmd then if v.val == 0 then value = get_console_cmd(0, v.cmd) elseif v.val == 1 then value = get_console_cmd(1, v.cmd) elseif v.val == 2 then value = get_console_cmd(0, v.cmd) --get_console_cmd(2, v.cmd) -- some commands are integers, using get_float will return 0. This is a walkaround value = tonumber(value) if v.min and v.max then value = clamp(value, v.min, v.max) end value = round_idp(value, v.prec or precision) end else value = axr_main.config:r_value(opt_section, id, v.val) end return value end function UI_MCM:GetContent(path, opt, v) if v.cmd and (not v.content) then local value if v.val == 0 then value = get_console_cmd(0, v.cmd) elseif v.val == 1 then value = get_console_cmd(1, v.cmd) elseif v.val == 2 then value = get_console_cmd(0, v.cmd) --get_console_cmd(2, v.cmd) value = tonumber(value) if v.min and v.max then value = clamp(value, v.min, v.max) end value = round_idp(value, v.prec or precision) end return {{value,tostring(value)}} elseif (type(v.content[1]) == "function") then return exec(unpack(v.content)) else return v.content end end function UI_MCM:GetOption(id) local t = str_explode(id,_opt_ ) local v = options for i=1,#t do for j=1,#v do if v[j].id == t[i] then if i == #t then v = v[j] else v = v[j].gr end break end end end return v end function UI_MCM:CacheValue(path, opt, value, v) local id = cc(path , opt) -- Do a backup of current values first if (opt_backup[id] == nil) then opt_backup[id] = self:GetValue(path, opt, v) print_dbg("# Backup [%s] = %s", id, opt_backup[id]) end -- Cache changed values if (value ~= nil) and (value ~= opt_backup[id]) then opt_temp[id] = value print_dbg("/ Cached [%s] = %s", id, value) else opt_temp[id] = nil -- no need to cache current values print_dbg("~ Cleared cache [%s]", id) end -- Change text color local _id = s_gsub(id, _opt_, "_") if self._Cap[_id] and self._Cap[_id]:IsShown() then if (opt_temp[id] ~= nil) and (opt_temp[id] ~= opt_backup[id]) then self._Cap[_id]:TextControl():SetTextColor( clr_o ) else self._Cap[_id]:TextControl():SetTextColor( clr_g1 ) end end -- Update state self:UpdatePending() if enable_ui_functors and v and v.on_selection_functor then ui_mcm.exec(unpack(v.on_selection_functor),path,opt,value,v) end -- If the on_selection_functor attribute contains a functor, pass a copy of the same args to it end function UI_MCM:Stacker(path, opt, v) -- This assure that each time a control is created, an unique callback id is given to it -- Why? because in normal case, jumping between options removes the previous ones constantly, getting back to them will create new controls and assign them to the old ids -- This is bad because callbacks are still attached to the old controls, any fresh controls that get assigned to those ids will be inactive as a result -- My solution is this function to generate unique id each time a control is created if (not v.stack) then v.stack = 0 end v.stack = v.stack + 1 return cc( cc(path , opt) , v.stack) end function UI_MCM:UpdatePending() local size = size_table(opt_temp) if size > 0 then self.pending:SetText( strformat( game.translate_string("ui_mm_warning_pending"), size) ) else self.pending:SetText("") end end ------------------------------------------------------------ -- Callbacks ------------------------------------------------------------ function UI_MCM:OnButton_Accept() --if self.Need_VidRestart then -- self.message_box:InitMessageBox("message_box_yes_no") -- self.message_box:SetText(string.format("%s %d% s", game.translate_string("ui_mm_confirm_changes"), 15, game.translate_string("mp_si_sec"))) -- self.message_box:ShowDialog(true) --else self:On_Accept() --end end function UI_MCM:OnButton_Reset() if self.last_path and self.last_curr_tree and is_not_empty(opt_temp) then local to_reset for id, val in pairs(opt_temp) do if s_find(id,self.last_path) then to_reset = true opt_temp[id] = nil local _id = s_gsub(id, _opt_, "_") if self._Cap[_id] and self._Cap[_id]:IsShown() then self._Cap[_id]:TextControl():SetTextColor( clr_g1 ) end end end if (to_reset) then self:UpdatePending() self:Reset_opt(self.last_curr_tree, self.last_path) print_dbg("% Sent callback (mcm_option_reset)") SendScriptCallback(mcm_option_reset) end end end function UI_MCM:OnButton_Default() if self.last_path and self.last_curr_tree then self:Reset_opt(self.last_curr_tree, self.last_path, { def = true }) print_dbg("% Sent callback (mcm_option_restore_default)") SendScriptCallback(mcm_option_restore_default) end end function UI_MCM:OnButton_Cancel() if is_not_empty(opt_temp) then self.message_box:InitMessageBox("message_box_yes_no") self.message_box:SetText(game.translate_string("ui_mm_discard_changes")) self.message_box:ShowDialog(true) else self:On_Cancel() end end function UI_MCM:On_Accept() self.Changes ={} for id, val in pairs(opt_temp) do local v = self:GetOption(id) -- Cache the changes if (not v.curr) then print_dbg("- Saved [%s] := %s", id, val) axr_main.config:w_value(opt_section, id, val) if key_by_path[id] then update_conflict(id, val) end self.Save_AXR = true end self.Change_Done = true self.Changes[id] = true -- Execute functors if found if v.functor then if v.postcondition then if exec(unpack(v.postcondition))then print_dbg("- Executing postcondition functor of [%s]",id) v.functor[#v.functor+1] = val exec(unpack(v.functor)) v.functor[#v.functor] = nil end else print_dbg("- Executing functor of [%s]",id) v.functor[#v.functor+1] = val exec(unpack(v.functor)) v.functor[#v.functor] = nil end end if v.type == "key_bind" then self.key_changed = true end -- See if it needs restart if v.restart then self.Need_Restart = true end if v.vid then self.Need_VidRestart = true end -- Send callback and apply changes if v.cmd then local cmd_value = val if type(cmd_value) == "boolean" then if v.bool_to_num then cmd_value = cmd_value and "1" or "0" else cmd_value = cmd_value and "on" or "off" end end print_dbg("- Saved CMD [%s] := %s", id, cmd_value) exec_console_cmd(v.cmd .. " " .. cmd_value) self.Save_CFG = true end end -- Save axr_options if self.Save_AXR then axr_main.config:save() self.Save_AXR = false end -- appdata if self.Save_CFG then print_dbg("- Saved CFG") exec_console_cmd("cfg_save") --exec_console_cmd("cfg_save tmp") end if level.present() and self.Change_Done then print_dbg("% Sent callback (on_option_change)") SendScriptCallback("on_option_change", true) end if AddScriptCallback and self.Change_Done then print_dbg("% Sent callback (mcm_option_change)") SendScriptCallback("mcm_option_change", self.Changes ) end -- Clear cache empty_table(opt_temp) empty_table(opt_backup) if self.key_changed then --resort the conflict key list. self.key_changed = false table.sort(options[1].gr[2].gr[3].gr, function(a,b) local t = {slide_kb = 1, desc_kb = 2, desc_kb2 = 3} if t[a.id] and t[b.id] then return t[a.id] < t[b.id] end if t[a.id] and not t[b.id] then return true end if (not t[a.id]) and t[b.id] then return false end return display_key(get(a.kb_path)) < display_key(get(b.kb_path)) end) init_opt_coder() end -- Exit self:On_Cancel() end function UI_MCM:On_Cancel() self.owner:ShowDialog(true) self:HideDialog() self.owner:Show(true) -- Restart vid if self.Need_VidRestart then exec_console_cmd("vid_restart") end if self.Need_Restart then self.owner:SetMsg( game.translate_string("ui_mm_change_done_restart") , 7 ) self.message_box:InitMessageBox("message_box_restart_game") self.message_box:ShowDialog(true) elseif self.Change_Done then self.owner:SetMsg( game.translate_string("ui_mm_change_done") , 5 ) end self.Change_Done = false self.Need_VidRestart = false self.Need_Restart = false self.Save_CFG = false Unregister_UI("UI_MCM") end function UI_MCM:On_Discard() empty_table(opt_temp) if (self.last_path and self.last_curr_tree) then self:UpdatePending() self:Reset_opt(self.last_curr_tree, self.last_path) print_dbg("% Sent callback (mcm_option_discard)") SendScriptCallback(mcm_option_discard) end self:On_Cancel() end function UI_MCM:OnKeyboard(dik, keyboard_action) local res = CUIScriptWnd.OnKeyboard(self,dik,keyboard_action) if (res == false) then local bind = dik_to_bind(dik) if keyboard_action == ui_events.WINDOW_KEY_PRESSED then if dik == DIK_keys.DIK_ESCAPE then if self.k_binder then self.k_binder(-1) else self:OnButton_Cancel() end elseif self.k_binder and display_key(dik) then self.k_binder(dik) end end end if self.k_binder and ((time_continual() - self.k_timer) > 500) then --clear k_binder on any other input after half a second. self.k_binder() end return res end function trader_cond(x) if x == 'get' then -- printf('@@@ returning %s', alife_storage_manager.get_state().trader_buy_condition_override or "0 (DEFAULT)") return alife_storage_manager.get_state().trader_buy_condition_override or 0 else -- printf('@@@ setting %s', opt_temp["gameplay/economy_diff/condition_buy_override"] or '0 (DEFAULT)') alife_storage_manager.get_state().trader_buy_condition_override = opt_temp["gameplay/economy_diff/condition_buy_override"] or 0 end end function store_in_save() printf("MCM Error: dph_mcm_save_storage.script not found") end if dph_mcm_save_storage and dph_mcm_save_storage.register_module then store_in_save = dph_mcm_save_storage.register_module end function on_game_start() RegisterScriptCallback("on_key_release",on_key_release) RegisterScriptCallback("on_key_press",on_key_press) end ------------------------------------------------------------ -- Tutorial:Table of Contents: ------------------------------------------------------------ --[[ Use the [-] in the margin in notpad++ to colapse sections for easier navigation 1. How to read these options in your script (RavenAscendant) 2. How to add new options (Tronex orginal turotial from ui_options) 3. How to make your script talk to MCM 4. Examples ]]-- ------------------------------------------------------------ -- Tutorial: How to read these options in your script: ------------------------------------------------------------ --[[ Feilds in [backets] are described in "How to add new options" First a bit about setting up your options. Nine times out of ten you don't need any functors. MCM will read the curent value of the setting from axr_options with out a [curr] functor and will write values to axr_options if no [functor] is provided. For simple global settings this will be more than adequate. The easiest way to read your setting is call ui_mcm.get(path) where path is the id fields of the nested tables down to the option in the table you returned in on_mcm_load() . Mostlikely this will take the form of "modname/settingname" but you can break your settings into multiple panels if you want resulting in a loinger path. see the options section of axr_configs for how anomaly options menu translates into paths, same system is used here. ui_mcm.get(path) is cached and fails back to the value you set in [def=] Just like ui_options when MCM applies a settings change it sends an on_option_change callback you can use this to do a one time read of your options into variables in your script. you can either get the values with ui_mcm.get(path) or read them directly from axr_configs like so: axr_main.config:r_value("mcm", path, type,default) --see _g for how r_value functions. Examples of when you might want to use functors: Saving mod settings to the save game file instead of globaly to axr_configs You are building your settings dynaicaly can't rely on the path being consistant. You otherwise like to over complicate things. ]]-- ------------------------------------------------------------ -- Tutorial: How to add new options: ------------------------------------------------------------ --[[ The only thing changed from this as compared to the version in ui_options is changing all ocurances of ui_mm_ to ui_mcm_ ------------------------------------------------------------------------------------------------ Option name: script will read option name (id) and show it automatically, naming should be like this: "ui_mcm_[tree_1]_[tree_2]_[tree_...]_[option]" [tree_n] and [option] are detemined from option path inside the table Example: options["video"]["general"]["renderer"] name will come from "ui_mcm_video_general_renderer" string ------------------------------------------------------------------------------------------------ Option description: option description can show up in the hint window if its defined by its name followed by "_desc" Example: option with a name of "ui_mcm_video_general_renderer" will show its hint if "ui_mcm_video_general_renderer_desc" exists ------------------------------------------------------------------------------------------------ Parameters of option Tree: ------------------------------------------------------------------------------------------------ - [id] - Define: (string) To give a tree its own identity - [sh] - Define: (boolean) used to detemine that the sub-tree tables are actual list of options to set and show - [text] - Define: (string) To over ride the display text for the tree in the tree select - [precondition] - Define: ( table {function, parameters} ) don't show tree options if its precondition return false - [output] - Define: (string) Text to show when precondition fails - [gr] - Define: ( table { ... } ) Table of a sub-tree or options list - [apply_to_all] - Define: (boolean) when you have options trees with similar options and group, you can use this to add "Apply to All" button to each option clicking it will apply option changes to this option in all other trees from same group you must give these a tree a group id - [id_gr] - Define: (string) allows you to give options tree a group id, to connect them when you want to use "Apply to all" button for options ------------------------------------------------------------------------------------------------ Parameters of options: ------------------------------------------------------------------------------------------------ ---------------------- Critical parameters: -------------------- These parameters must be declared for elements [id] - Define: (string) Option identity/name. Option get stored in axr_main or called in other sripts by its path (IDs of sub trees and option): Example: ( tree_1_id/tree_2_id/.../option_id ) The top id in the table you return to MCM (tree_1_id in the above example) should be as unique as posible to prevent it from conflicting with another mod. [type] - Define: (string) - Possible values: - Option elements: "check" : Option, check box, either ON or OFF "list" : Option, list of strings, useful for options with too many selections "input" : Option, input box, you can type a value of your choice "radio_h" : Option, radio box, select one out of many choices. Can fit up to 8 selections (Horizental layout) "radio_v" : Option, radio box, select one out of many choices. Can fit up any number of selections (Vertical layout) "track" : Option, track bar, easy way to control numric options with min/max values (can be used only if [val] = 2) "key_bind" : Option, button that registers a keypress after being clicked. (See suplimental instructions below) - Support elements: "line" : Support element, a simple line to separate things around "image" : Support element, 563x50 px image box, full-area coverage "slide" : Support element, image box on left, text on right "title" : Support element, title (left/right/center alignment) "desc" : Support element, description (left alignment) ---------------------- Dependable parameters: ---------------------- These parameters must be declared when other specific parameters are declared already. They work along with them [val] - Define: (number) - Used by: option elements: ALL Option's value type: 0. string | 1. boolean | 2. float It tells the script what kind of value the option is storing / dealing with [cmd]: - Define: (string) - Used by: option elements: ALL (needed if you want to control a console command) Tie an option to a console command, so when the option value get changed, it get applied directly to the command The option will show command's current value NOTE: cmd options don't get cached in axr_options, instead they get stored in appdata/user.ltx [def] parameter is not needed here since we engine applies default values to commands if they don't exist in user.ltx automatically [def] - Define: (boolean) / (number) / (string) / ( table {function, parameters} ) - Used by: option elements: ALL (not needed if [cmd] is used) Default value of an option when no cached values are found in axr_options, the default value will be used [min] - Define: (number) - Used by: option elements: "input" / "track": (only if [val] = 2) Minimum viable value for an option, to make sure a value stays in range [max] - Define: (number) - Used by: option elements: "input" / "track": (only if [val] = 2) Maximum viable value for an option, to make sure a value stays in range [step] - Define: (number) - Used by: option elements: "track": (only if [val] = 2) How much a value can be increased/decreased in one step [content] - Define: ( table {double pairs} ) / ( table {function, parameters} ) - Used by: option elements: "list" / "radio_h" / "radio_v": Delcares option's selection list Pairs: { value of the selection, string to show on UI } Example: content= { {0,"off"} , {1,"half"} , {2,"full"}} So the list or radio option will show 3 selections (translated strings): (ui_mcm_lst_off) and (ui_mcm_lst_half) and (ui_mcm_lst_full) When you select one and it get applied, the assosiated value will get applied So picking the first one will pass ( 0 ) Because all lists and radio button elements share the same prefix, "ui_mcm_lst_" it is important that you not use common words like the ones in the example above. Make your element names unique. [link] - Define: (string) - Used by: support elements: "image" / "slide" Link to texture you want to show [text] - Define: (string) - Used by: support elements: "slide" / "title" / "desc" String to show near the image, it will be translated ---------------------- Optional parameters: ---------------------- These parameters are completely optionals, and can be used for custom stuff [force_horz] - Define: (boolean) - Used by: option elements: "radio_h" Force the radio buttons into horizental layout, despite their number [no_str] - Define: (boolean) - Used by: option elements: "list" / "radio_h" / "radio_v" / "track" Usually, the 2nd key of pairs in content table are strings to show on the UI, by translating "opt_str_lst_(string)" when we set [no_str] to true, it will show the string fromm table as it is without translations or "opt_str_lst_" For TrackBars: no_str won't show value next to the slider [prec] - Define: (number) - Used by: option elements: "track" allowed number of zeros in a number [precondition] - Define: ( table {function, parameters} ) - Used by: option elements: ALL Show the option on UI if the precondition function returns true [functor] - Define: ( table {function, parameters} ) - Used by: option elements: ALL Execute a function when option's changes get applied The value of the option is added to the end of the parameters list. [postcondition] - Define: ( table {function, parameters} ) - Used by: option elements: ALL, with the defined functor Option won't execute its functor when changes are applied, unless if the postcondition function returns true [ui_hook_functor] - Define: ( table {function, parameters} ) - Used by: option elements: ALL except keybind, with the defined functor - Used by: support elements: ALL Passes handling elements and metadata back to the defined functor for UI customization For ADVANCED scripting use only - see below [on_selection_functor] - Define: ( table {function, parameters} ) - Used by option elements: ALL, with the defined functor Execute a defined functor upon any live selection of a new option value For ADVANCED scripting use only - see below [curr] - Define: ( table {function, parameters} ) - Used by: option elements: ALL get current value of an option by executing the declared function, instead of reading it from axr_options.ltx [hint] (as of MCM 1.6.0 this will actualy show _desc strings) - Define: (string) - Used by: option elements: ALL Override default name / desc rule to replace the translation of an option with a custom one, should be set without "ui_mcm_" and "_desc" Example: { hint = "alife_warfare_capture"} will force the script to use "ui_mcm_alife_warfare_capture" and "ui_mcm_alife_warfare_capture_desc" for name and desc of the option [clr] - Define: ( table {a,r,b,g} ) - Used by: support elements: "title" / "desc" determines the color of the text [stretch] - Define: (boolean) - Used by: support elements: "slide" force the texture to stretch or not [pos] - Define: ( table {x,y} ) - Used by: support elements: "slide" custom pos for the texture [size] - Define: ( table {w,z} ) - Used by: support elements: "slide" custom size for the texture [align] - Define: (string) "l" "r" "c" - Used by: support elements: "title" determines the alignment of the title [spacing] - Define: (number) - Used by: support elements: "slide" height offset to add extra space [borderless] - Define: (boolean) - Used by: support elements: "slide" disables the border lines above and below the slide --]] ------------------------------------------------------------ -- Tutorial: How to make your script talk to MCM: ------------------------------------------------------------ --[[ MCM looks for scripts with names ending in mcm: *mcm.script you can use an _ to sperate it from the rest of the name of your script but it isn't necessary. In those scripts MCM will execute the function on_mcm_load() In order for options to be added to MCM, on_mcm_load() must return a valid options tree as described in the tutorial here, used in the ui_options script and shown in the examples below An aditioanl retun value of a string naming a collection is optional. The string will be used to create a catagory to which the the options menues of mods returning the same collection name will be added to. This is to allow for modular mods to have the settings for each module be grooped under a common heading. Note the collection name becomes the root name in your settings path and translation strings. As a root name care should be taken to ensure it will not conflict with another mod. ]]-- --------------------------------------------------------------------------------------- -- Tutorial: Using dph-hcl's script for save game specific MCM options --------------------------------------------------------------------------------------- --[[ dph-hcl's orginal script from https://www.moddb.com/mods/stalker-anomaly/addons/151-mcm-13-mcm-savefile-storage is included un altered and can be used as described and documented in thier mod and script Aditionaly for convinence the function has been aliased here as ui_mcm.store_in_save(path) this function can be called safely as MCM will simply print an error if dph-hcl's script is missing To make an option be stored in a save game instead of globaly call ui_mcm.store_in_save(path) path can be a full option path like is used by ui_mcm.get(path) or a partial path If a partial path is used all options that caontain that path will be stored in the savegame partial paths must start with a valid root and cannot end with a / In the second example below the second checkbox in the second options menu would be stored buy ui_mcm.store_in_save("example_example/example_two/2check2") In the same example storing all options (both checks) in the first option menu would be: ui_mcm.store_in_save("example_example/example_one") Lastly storing all of the options for your mod would look like: ui_mcm.store_in_save("example_example") ui_mcm.store_in_save(path) can be called at any time. The easyiest is probably in on_mcm_load() however it could be done as late as on_game_start() if one wanted to have an MCM option for global vs save specific options storing (calling ui_mcm.get(path) in on_mcm_load() is a bad idea don't do that ) ]]-- --------------------------------------------------------------------------------------- -- Tutorial: Additional information on key_bind --------------------------------------------------------------------------------------- --[[ Key binds are gathered into two meta lists for the users convienance. This means it is very important that your translation strings clearly identify what the key does and ideally it should be clear what addon the keybiind is from. The value stored by the key bind is the DIK_keys value of the key. Same number that will be given to the key related callbacks. val must be set to 2 and is still manditory. curr and functor are not curently supported. Post an issue on github describing the usecase you had for them, if it's cool enough they might get fixed. Old (pre 1.6.0) versions of MCM will not display key_bind and calling ui_mcm.get for it will return nil, take that into acount if you want reverse compatablity. ]]-- --------------------------------------------------------------------------------------- -- Tutorial: Additional Key Bind utilities --------------------------------------------------------------------------------------- --[[ MCM tracks the held status of the control and shift keys as well as a flag that is true when neither is pressed ui_mcm.MOD_NONE ui_mcm.MOD_SHIFT and ui_mcm.MOD_CTRL ui_mcm.get_mod_key(val) will return the above flags based on val: 0:MOD_NONE 1:MOD_SHIFT and 2:MOD_CTRL If these somehow get latched they reset when Escape is pressed. Please report cases of latching. MCM provides functions for detecting key double taps and keys that are held down, and single key presses that do not go on to be double or long presses. ui_mcm.double_tap(id, key, [multi_tap]) should be called from on_key_press callback after you have filtered for your key id: should be a unique identifier for your event, scriptname and a number work well:"ui_mcm01" key: is of course they key passed into the on_key_press callback. multi_tap: if true timer is updated instead of cleared allowing for the detection of triple/quad/ect taps returns: true for a given id and key if less than X ms has elapsed since the last time it was called with that id and key (X is a user configurable value between 100ms and 1000 ms returns false otherwise. If multi_tap is false timer is reset when true is returned preventing the function from returning true twice in a row If multi_tap is true the function will return true any time the gap between a call and the one before is within the window. ui_mcm.key_hold(id, key, [repeat]) should be called from on_key_hold callback after you have filtered for your key id: should be a unique identifer for your event, scriptname and a number work well:"ui_mcm01" key: is the key passed into the on_key_hold callback. repeat: Optional. time in seconds. If the key continues to be held down will return true again after this many seconds on a cycle. when called from the on_key_hold callback it will return true after the key has been held down for Y ms (determined by applying a user defined multiplier to X above) and then again every repeat seconds if repeat is provided. sequence resets when key is released. ui_mcm.simple_press(id, key, functor) should be called from on_key_press callback after you have filtered for your key id: should be a unique identifier for your event, scrip name and a number work well:"ui_mcm01" key: is the key passed into the on_key_press callback. function: table {function, parameters}, to be executed when it is determined that the press is not long or double (or multi press in general) Unlike the other two this does not return any thing but instead you give it a function to execute. Using this function you gain exclusivity, your event won't fire when the key is double(multi) taped or held (long press), at the cost of a small bit of input delay. This delay is dependent on the double tap window the used defines in the MCM Key Bind settings. The following option entries have translation stings provided by MCM and are setup to be ignored by pre 1.6.0 versions of MCM Note the keybind conflict identification in MCM does NOT look for these and reports conflict on the keybind value alone. With shift and control, radio buton style {id = "modifier", type = ui_mcm.kb_mod_radio, val = 2, def = 0, hint = "mcm_kb_modifier" , content= { {0,"mcm_kb_mod_none"} , {1,"mcm_kb_mod_shift"} , {2,"mcm_kb_mod_ctrl"},{3,"mcm_kb_mod_alt"}}}, With shift and control, list style {id = "modifier", type = ui_mcm.kb_mod_list, val = 2, def = 0, hint = "mcm_kb_modifier" , content= { {0,"mcm_kb_mod_none"} , {1,"mcm_kb_mod_shift"} , {2,"mcm_kb_mod_ctrl"},{3,"mcm_kb_mod_alt"}}}, Single double or long press, , radio buton style {id = "mode", type = ui_mcm.kb_mod_radio, val = 2, def = 0, hint = "mcm_kb_mode" , content= { {0,"mcm_kb_mode_press"} , {1,"mcm_kb_mode_dtap"} , {2,"mcm_kb_mode_hold"}}}, Single double or long press, , radio buton style {id = "mode", type = ui_mcm.kb_mod_list, val = 2, def = 0, hint = "mcm_kb_mode" , content= { {0,"mcm_kb_mode_press"} , {1,"mcm_kb_mode_dtap"} , {2,"mcm_kb_mode_hold"}}}, An example script making use of all of these can be found at: https://github.com/RAX-Anomaly/MiniMapToggle/blob/main/gamedata/scripts/mini_map_toggle_mcm.script ]]-- ------------------------------------------------------------ -- Tutorial: Examples: ------------------------------------------------------------ -- these examples can all be copied to a blank script example_mcm.script and ran. -- A simple menu with a title slide and check boxes. --[[ function on_mcm_load() op = { id= "example_example" ,sh=true ,gr={ { id= "slide_example_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_example_example" ,size= {512,50} ,spacing= 20 }, {id = "check1", type = "check", val = 1, def = false}, {id = "check2", type = "check", val = 1, def = false}, } } return op end ]]-- -- A a tree with a root containing three menues with a title slide and check boxes. --[[ function on_mcm_load() op = { id= "example_example" , ,gr={ { id= "example_one" ,sh=true ,gr={ { id= "slide_example_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_example_example" ,size= {512,50} ,spacing= 20 }, {id = "1check1", type = "check", val = 1, def = false}, {id = "1check2", type = "check", val = 1, def = false}, } }, { id= "example_two" ,sh=true ,gr={ { id= "slide_example_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_example_example" ,size= {512,50} ,spacing= 20 }, {id = "2check1", type = "check", val = 1, def = false}, {id = "2check2", type = "check", val = 1, def = false}, } }, { id= "example_three" ,sh=true ,gr={ { id= "slide_example_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_example_example" ,size= {512,50} ,spacing= 20 }, {id = "3check1", type = "check", val = 1, def = false}, {id = "3check2", type = "check", val = 1, def = false}, } }, } } return op end ]]-- -- Two scripts with a simple menu with a title slide and check boxes, that will be added to a collection named "collection_example" --[[ -- example1_mcm.script function on_mcm_load() op = { id= "first_example" ,sh=true ,gr={ { id= "slide_first_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_first_example" ,size= {512,50} ,spacing= 20 }, {id = "check1", type = "check", val = 1, def = false}, {id = "check2", type = "check", val = 1, def = false}, } } return op, "collection_example" end -- example2_mcm.script. function on_mcm_load() op = { id= "second_example" ,sh=true ,gr={ { id= "slide_second_example" ,type= "slide" ,link= "AMCM_Banner.dds" ,text= "ui_mcm_title_second_example" ,size= {512,50} ,spacing= 20 }, {id = "check1", type = "check", val = 1, def = false}, {id = "check2", type = "check", val = 1, def = false}, } } return op, "collection_example" end ]]-- --[[===================================================================== Tutorial: UI Functors *** ADVANCED SCRIPTING USE ONLY *** -- ====================================================================== The UI functors "ui_hook_functor" and "on_selection_functor", respectively, pass along UI element handlers and a trap for unsaved value changes. They are recommended only for advanced scripters who fully understand how to work with UI elements and want to customize their menu beyond what the template can achieve. These functors allow for dynamic customizations to MCM's UI elements at the element container level in response to user interactions. They can also compeltely hose your addon's entire MCM menu with a single error. This is not a power to be used lightly. If you use these, you are assumed to know what you're doing. Please don't bug us with how-to questions. - [ui_hook_functor] - Define: ( table {function, parameters} ) - Used by: option elements: ALL (except keybind), with the defined functor - Used by: support elements: ALL, with the defined functor - Parameters passed: anchor, handlers, attrs, flags Execute a function upon the initial registration of a UI element that occurs during on_mcm_load. anchor - empty static container to use as an anchor for other elements handlers - table containing any necessary UI control handlers, varies by element type attrs - table of MCM attributes for the menu option flags - flags.etype is always a string with the element type, other metadata varies by type As with other functor attributes, the value of the "parameters" option in the table is added to the end of the parameters list. - [on_selection_functor] - Define: ( table {function, parameters} ) - Used by option elements: ALL, with the defined functor - Parameters passed: path, opt, value, attrs Execute a function on any unsaved/uncommitted change to an option value. Allows realtime response to user selections. path - MCM path to changed option opt - ID of the changed option value - value of the uncommitted change attrs - table of MCM attributes for the menu option As with other functor attributes, the value of the "parameters" option in the table is added to the end of the parameters list. New supporting callbacks: mcm_option_reset - sent from OnButton_Reset mcm_option_restore_default - sent from OnButton_Default mcm_option_discard - sent from On_Discard These callbacks all fire on their respective events resulting in cancellation of pending MCM changes. This lets you clear your own table of changes or do any other necessary cleanup at the end of the MCM session. Each menu element type has its own set of handlers and flags that are passed in these two tables. They are documented below. In addition to those listed, all elements pass the flag "etype" containing the name of the element (in brackets below). SUPPORT ELEMENTS [line] Handlers - line: Static container for the line element and its texture [image] Handlers - pic: Static container for the image element and its texture [slide] Handlers - pic: Static container for the slide image element and its texture - txt: TextWnd container for the slide label text [title] Handlers - title: Static container for the title element text [desc] Handlers - desc: Static container for the description element text OPTION ELEMENTS All Option elements pass the following Flags: - path: MCM menu path to the option - opt: MCM ID for the option [check] Handlers - cap: Static container for the localized text caption - ctrl: Static container for the checkbox input control [list] Handlers - cap: Static container for the localized text caption - ctrl: Static container for the dropdown list input control [input] Handlers - cap: Static container for the localized text caption - ctrl: Static container for the input box control [track] Handlers - cap: Static container for the localized text caption - ctrl: Static container for the track input control [radio] Handlers - cap: Static container for the localized text caption - txt: Table of TextWnd containers for the radio options - ctrltbl: Table of radio button input controls Flags - num_opts: the number of radio button options in the above tables - hvtype: string "horz" or "vert" denoting layout style -- ====================================================================== Tutorial: UI Functors: Example: -- ====================================================================== You should be very familiar with how to work with UI containers in Anomaly before going any further. This is a very simple example of how to use the UI functors. Assume this script is saved as element_test_mcm.script. 1. MCM calls do_ui_hook_functor for each of the menu elements that have it defined: a slide and a checkbox 2. do_ui_hook_functor creates a new empty static anchored on the image element to contain the icon texture, stores the handlers for it and the slide text, and changes the caption text for the checkbox 3. once during init, and each time the user clicks the checkbox, the icon and slide text will change in response -- ====================================================================]] --[[ function on_mcm_load() op = {id= "ui_functor_test", sh=true, gr={ {id = "img_container", type= "slide", ui_hook_functor= {element_test_mcm.do_ui_hook_functor}}, {id = "checkbox", type= "check", val = 1, def = true, hint = "", ui_hook_functor= {element_test_mcm.do_ui_hook_functor}, on_selection_functor= {element_test_mcm.do_on_selection_functor}}, } } return op end xml = CScriptXmlInit() xml:ParseFile("ui_mcm.xml") local wnd,txt function do_on_selection_functor(path, opt, value, attrs) -- This function is called every time the user makes a selection or change -- We also call it manually below, once during init, to set the default texture if not (wnd and txt) then return end local primary = value and true or false if primary then wnd:InitTexture("ui_inGame2_PDA_icon_Primary_mission") txt:SetText("Primary Mission icon") else wnd:InitTexture("ui_inGame2_PDA_icon_Secondary_mission") txt:SetText("Secondary Mission icon") end end function do_ui_hook_functor(anchor, handlers, attrs, flags) -- This function should do any first-time setup for each menu option if not (anchor and handlers and flags) then return end local etype = flags and flags.etype if etype == "slide" then wnd = xml:InitStatic("elements:image",anchor) wnd:SetWndSize(vector2():set(24,28)) local pos = wnd:GetWndPos() wnd:SetWndPos(vector2():set(pos.x, pos.y + 28)) txt = handlers.txt local curr_state = ui_mcm.get("ui_functor_test/checkbox") do_on_selection_functor(nil,nil,curr_state) elseif etype == "check" then local newtext = "Click the checkbox to toggle the icon type" local cap = handlers.cap cap:TextControl():SetText(newtext) end end --]]