Merge pull request #2430 from myk002/myk_overlay_gui_prep

[overlay] prep lua code for usage by gui/overlay
develop
Myk 2022-11-29 15:15:48 -08:00 committed by GitHub
commit b7bb0c49e6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 53 additions and 33 deletions

@ -116,6 +116,10 @@ The ``overlay.OverlayWidget`` superclass defines the following class attributes:
viewscreen's ``logic()`` function). This call to ``overlay_onupdate`` is in viewscreen's ``logic()`` function). This call to ``overlay_onupdate`` is in
addition to any calls initiated from associated interposed viewscreens and addition to any calls initiated from associated interposed viewscreens and
will come after calls from associated viewscreens. will come after calls from associated viewscreens.
- ``overlay_only`` (default: ``false``)
If set to ``true``, no widget frame will be drawn in `gui/overlay` for drag
and drop repositioning. Overlay widgets that don't have a "widget" to
reposition should set this to ``true``.
- ``overlay_onupdate_max_freq_seconds`` (default: ``5``) - ``overlay_onupdate_max_freq_seconds`` (default: ``5``)
This throttles how often a widget's ``overlay_onupdate`` function can be This throttles how often a widget's ``overlay_onupdate`` function can be
called (from any source). Set this to the largest amount of time (in called (from any source). Set this to the largest amount of time (in

@ -6,8 +6,9 @@ overlay
:tags: dfhack interface :tags: dfhack interface
The overlay framework manages the on-screen widgets that other tools (including The overlay framework manages the on-screen widgets that other tools (including
3rd party plugins and scripts) can register for display. If you are a developer 3rd party plugins and scripts) can register for display. For a graphical
who wants to write an overlay widget, please see the `overlay-dev-guide`. configuration interface, please see `gui/overlay`. If you are a developer who
wants to write an overlay widget, please see the `overlay-dev-guide`.
Usage Usage
----- -----

@ -111,6 +111,7 @@ MiseryWidget.ATTRS{
function MiseryWidget:init() function MiseryWidget:init()
self.colors = getStressCategoryColors() self.colors = getStressCategoryColors()
self.stress_category_counts = {} self.stress_category_counts = {}
self.frame.w = 2*#self.colors + 1
end end
function MiseryWidget:overlay_onupdate() function MiseryWidget:overlay_onupdate()

@ -14,20 +14,40 @@ local DEFAULT_X_POS, DEFAULT_Y_POS = -2, -2
-- state and config -- -- state and config --
-- ---------------- -- -- ---------------- --
local active_triggered_widget = nil local trigger_lock_holder_description = nil
local active_triggered_screen = nil -- if non-nil, hotspots will not get updates local trigger_lock_holder_screen = nil -- if non-nil, no triggering allowed
local widget_db = {} -- map of widget name to ephermeral state local widget_db = {} -- map of widget name to ephermeral state
local widget_index = {} -- ordered list of widget names local widget_index = {} -- ordered list of widget names
local overlay_config = {} -- map of widget name to persisted state local overlay_config = {} -- map of widget name to persisted state
local active_hotspot_widgets = {} -- map of widget names to the db entry local active_hotspot_widgets = {} -- map of widget names to the db entry
local active_viewscreen_widgets = {} -- map of vs_name to map of w.names -> db local active_viewscreen_widgets = {} -- map of vs_name to map of w.names -> db
local function reset() function get_state()
if active_triggered_screen then return {index=widget_index, config=overlay_config, db=widget_db}
active_triggered_screen:dismiss() end
function register_trigger_lock_screen(scr, desc)
if trigger_lock_holder_screen then
if not trigger_lock_holder_screen:isActive() then
trigger_lock_holder_screen:dismiss()
end
trigger_lock_holder_description = nil
end
trigger_lock_holder_screen = scr
if trigger_lock_holder_screen then
trigger_lock_holder_description = desc
return true
end end
active_triggered_widget = nil end
active_triggered_screen = nil
local function triggered_screen_has_lock()
if not trigger_lock_holder_screen then return false end
if trigger_lock_holder_screen:isActive() then return true end
return register_trigger_lock_screen(nil, nil)
end
local function reset()
register_trigger_lock_screen(nil, nil)
widget_db = {} widget_db = {}
widget_index = {} widget_index = {}
@ -46,19 +66,11 @@ local function save_config()
end end
end end
local function triggered_screen_has_lock()
if not active_triggered_screen then return false end
if active_triggered_screen:isActive() then return true end
active_triggered_widget = nil
active_triggered_screen = nil
return false
end
-- ----------- -- -- ----------- --
-- utility fns -- -- utility fns --
-- ----------- -- -- ----------- --
local function normalize_list(element_or_list) function normalize_list(element_or_list)
if type(element_or_list) == 'table' then return element_or_list end if type(element_or_list) == 'table' then return element_or_list end
return {element_or_list} return {element_or_list}
end end
@ -70,7 +82,7 @@ local function normalize_viewscreen_name(vs_name)
end end
-- reduce "long form" viewscreen names to "short form" -- reduce "long form" viewscreen names to "short form"
local function simplify_viewscreen_name(vs_name) function simplify_viewscreen_name(vs_name)
_,_,short_name = vs_name:find('^viewscreen_(.*)st$') _,_,short_name = vs_name:find('^viewscreen_(.*)st$')
if short_name then return short_name end if short_name then return short_name end
return vs_name return vs_name
@ -162,7 +174,7 @@ local function do_enable(args, quiet, skip_save)
end end
end end
local function do_disable(args) local function do_disable(args, quiet)
local disable_fn = function(name, db_entry) local disable_fn = function(name, db_entry)
overlay_config[name].enabled = false overlay_config[name].enabled = false
if db_entry.widget.hotspot then if db_entry.widget.hotspot then
@ -175,7 +187,9 @@ local function do_disable(args)
active_viewscreen_widgets[vs_name] = nil active_viewscreen_widgets[vs_name] = nil
end end
end end
print(('disabled widget %s'):format(name)) if not quiet then
print(('disabled widget %s'):format(name))
end
end end
if args[1] == 'all' then if args[1] == 'all' then
for name,db_entry in pairs(widget_db) do for name,db_entry in pairs(widget_db) do
@ -301,7 +315,7 @@ local function dump_widget_config(name, widget)
end end
end end
local function do_position(args) local function do_position(args, quiet)
local name_or_number, x, y = table.unpack(args) local name_or_number, x, y = table.unpack(args)
local name = get_name(name_or_number) local name = get_name(name_or_number)
if not widget_db[name] then if not widget_db[name] then
@ -328,11 +342,13 @@ local function do_position(args)
widget.frame = make_frame(pos, widget.frame) widget.frame = make_frame(pos, widget.frame)
widget:updateLayout(get_screen_rect()) widget:updateLayout(get_screen_rect())
save_config() save_config()
print(('repositioned widget %s to x=%d, y=%d'):format(name, pos.x, pos.y)) if not quiet then
print(('repositioned widget %s to x=%d, y=%d'):format(name, pos.x, pos.y))
end
end end
-- note that the widget does not have to be enabled to be triggered -- note that the widget does not have to be enabled to be triggered
local function do_trigger(args) local function do_trigger(args, quiet)
if triggered_screen_has_lock() then if triggered_screen_has_lock() then
dfhack.printerr(('cannot trigger widget; widget "%s" is already active') dfhack.printerr(('cannot trigger widget; widget "%s" is already active')
:format(active_triggered_widget)) :format(active_triggered_widget))
@ -341,11 +357,10 @@ local function do_trigger(args)
do_by_names_or_numbers(args[1], function(name, db_entry) do_by_names_or_numbers(args[1], function(name, db_entry)
local widget = db_entry.widget local widget = db_entry.widget
if widget.overlay_trigger then if widget.overlay_trigger then
active_triggered_screen = widget:overlay_trigger() register_trigger_lock_screen(widget:overlay_trigger(), name)
if active_triggered_screen then if not quiet then
active_triggered_widget = name print(('triggered widget %s'):format(name))
end end
print(('triggered widget %s'):format(name))
end end
end) end)
end end
@ -360,10 +375,10 @@ local command_fns = {
local HELP_ARGS = utils.invert{'help', '--help', '-h'} local HELP_ARGS = utils.invert{'help', '--help', '-h'}
function overlay_command(args) function overlay_command(args, quiet)
local command = table.remove(args, 1) or 'help' local command = table.remove(args, 1) or 'help'
if HELP_ARGS[command] or not command_fns[command] then return false end if HELP_ARGS[command] or not command_fns[command] then return false end
command_fns[command](args) command_fns[command](args, quiet)
return true return true
end end
@ -398,9 +413,7 @@ local function do_update(name, db_entry, now_ms, vs)
local w = db_entry.widget local w = db_entry.widget
db_entry.next_update_ms = get_next_onupdate_timestamp(now_ms, w) db_entry.next_update_ms = get_next_onupdate_timestamp(now_ms, w)
if detect_frame_change(w, function() return w:overlay_onupdate(vs) end) then if detect_frame_change(w, function() return w:overlay_onupdate(vs) end) then
active_triggered_screen = w:overlay_trigger() if register_trigger_lock_screen(w:overlay_trigger(), name) then
if active_triggered_screen then
active_triggered_widget = name
return true return true
end end
end end
@ -463,6 +476,7 @@ OverlayWidget = defclass(OverlayWidget, widgets.Widget)
OverlayWidget.ATTRS{ OverlayWidget.ATTRS{
name=DEFAULT_NIL, -- this is set by the framework to the widget name name=DEFAULT_NIL, -- this is set by the framework to the widget name
default_pos={x=DEFAULT_X_POS, y=DEFAULT_Y_POS}, -- 1-based widget screen pos default_pos={x=DEFAULT_X_POS, y=DEFAULT_Y_POS}, -- 1-based widget screen pos
overlay_only=false, -- true if there is no widget to reposition
hotspot=false, -- whether to call overlay_onupdate on all screens hotspot=false, -- whether to call overlay_onupdate on all screens
viewscreens={}, -- override with associated viewscreen or list of viewscrens viewscreens={}, -- override with associated viewscreen or list of viewscrens
overlay_onupdate_max_freq_seconds=5, -- throttle calls to overlay_onupdate overlay_onupdate_max_freq_seconds=5, -- throttle calls to overlay_onupdate