2023-04-12 16:40:13 +00:00
|
|
|
-- Sidebar Graphics Element
|
|
|
|
|
2023-06-03 21:40:57 +00:00
|
|
|
local tcd = require("scada-common.tcd")
|
2023-09-29 23:34:10 +00:00
|
|
|
local util = require("scada-common.util")
|
2023-04-12 16:40:13 +00:00
|
|
|
|
2023-05-10 15:46:06 +00:00
|
|
|
local core = require("graphics.core")
|
2023-04-12 16:40:13 +00:00
|
|
|
local element = require("graphics.element")
|
|
|
|
|
2023-09-17 01:06:16 +00:00
|
|
|
local MOUSE_CLICK = core.events.MOUSE_CLICK
|
2023-05-10 15:46:06 +00:00
|
|
|
|
2023-04-12 16:40:13 +00:00
|
|
|
---@class sidebar_args
|
|
|
|
---@field parent graphics_element
|
|
|
|
---@field id? string element id
|
|
|
|
---@field x? integer 1 if omitted
|
2023-07-10 03:42:44 +00:00
|
|
|
---@field y? integer auto incremented if omitted
|
2023-04-12 16:40:13 +00:00
|
|
|
---@field height? integer parent height if omitted
|
|
|
|
---@field fg_bg? cpair foreground/background colors
|
2023-05-25 21:40:16 +00:00
|
|
|
---@field hidden? boolean true to hide on initial draw
|
2023-04-12 16:40:13 +00:00
|
|
|
|
|
|
|
-- new sidebar tab selector
|
|
|
|
---@param args sidebar_args
|
|
|
|
---@return graphics_element element, element_id id
|
|
|
|
local function sidebar(args)
|
|
|
|
args.width = 3
|
|
|
|
|
|
|
|
-- create new graphics element base object
|
|
|
|
local e = element.new(args)
|
|
|
|
|
|
|
|
-- default to 1st tab
|
|
|
|
e.value = 1
|
|
|
|
|
2023-09-29 23:34:10 +00:00
|
|
|
local was_pressed = false
|
2024-05-10 03:05:55 +00:00
|
|
|
local tabs = {}
|
2023-09-29 23:34:10 +00:00
|
|
|
|
2023-04-12 16:40:13 +00:00
|
|
|
-- show the button state
|
2023-09-29 23:34:10 +00:00
|
|
|
---@param pressed? boolean if the currently selected tab should appear as actively pressed
|
2023-05-10 15:46:06 +00:00
|
|
|
---@param pressed_idx? integer optional index to show as held (that is not yet selected)
|
|
|
|
local function draw(pressed, pressed_idx)
|
2023-09-29 23:34:10 +00:00
|
|
|
pressed = util.trinary(pressed == nil, was_pressed, pressed)
|
|
|
|
was_pressed = pressed
|
2023-05-10 15:46:06 +00:00
|
|
|
pressed_idx = pressed_idx or e.value
|
|
|
|
|
2024-05-10 03:05:55 +00:00
|
|
|
-- clear
|
|
|
|
e.w_set_fgd(e.fg_bg.fgd)
|
|
|
|
e.w_set_bkg(e.fg_bg.bkg)
|
|
|
|
for y = 1, e.frame.h do
|
|
|
|
e.w_set_cur(1, y)
|
|
|
|
e.w_write(" ")
|
|
|
|
end
|
2023-04-12 16:40:13 +00:00
|
|
|
|
2024-05-10 03:05:55 +00:00
|
|
|
-- draw tabs
|
|
|
|
for i = 1, #tabs do
|
|
|
|
local tab = tabs[i] ---@type sidebar_tab
|
|
|
|
local y = tab.y_start
|
2023-04-12 16:40:13 +00:00
|
|
|
|
2023-08-31 01:11:57 +00:00
|
|
|
e.w_set_cur(1, y)
|
2023-04-12 16:40:13 +00:00
|
|
|
|
2023-05-10 15:46:06 +00:00
|
|
|
if pressed and i == pressed_idx then
|
2023-08-31 01:11:57 +00:00
|
|
|
e.w_set_fgd(e.fg_bg.fgd)
|
|
|
|
e.w_set_bkg(e.fg_bg.bkg)
|
2023-04-12 16:40:13 +00:00
|
|
|
else
|
2023-08-31 01:11:57 +00:00
|
|
|
e.w_set_fgd(tab.color.fgd)
|
|
|
|
e.w_set_bkg(tab.color.bkg)
|
2023-04-12 16:40:13 +00:00
|
|
|
end
|
|
|
|
|
2024-05-10 03:05:55 +00:00
|
|
|
if tab.tall then
|
|
|
|
e.w_write(" ")
|
|
|
|
e.w_set_cur(1, y + 1)
|
|
|
|
end
|
|
|
|
|
|
|
|
e.w_write(tab.label)
|
|
|
|
|
|
|
|
if tab.tall then
|
|
|
|
e.w_set_cur(1, y + 2)
|
|
|
|
e.w_write(" ")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- determine which tab was pressed
|
|
|
|
---@param y integer y coordinate
|
|
|
|
local function find_tab(y)
|
|
|
|
for i = 1, #tabs do
|
|
|
|
local tab = tabs[i] ---@type sidebar_tab
|
|
|
|
|
|
|
|
if y >= tab.y_start and y <= tab.y_end then
|
|
|
|
return i
|
|
|
|
end
|
2023-04-12 16:40:13 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- handle mouse interaction
|
|
|
|
---@param event mouse_interaction mouse event
|
|
|
|
function e.handle_mouse(event)
|
|
|
|
-- determine what was pressed
|
|
|
|
if e.enabled then
|
2024-05-10 03:05:55 +00:00
|
|
|
local cur_idx = find_tab(event.current.y)
|
|
|
|
local ini_idx = find_tab(event.initial.y)
|
2023-05-10 15:46:06 +00:00
|
|
|
|
2024-05-10 03:05:55 +00:00
|
|
|
if tabs[cur_idx] ~= nil then
|
2023-09-17 01:06:16 +00:00
|
|
|
if event.type == MOUSE_CLICK.TAP then
|
2023-05-10 15:46:06 +00:00
|
|
|
e.value = cur_idx
|
|
|
|
draw(true)
|
|
|
|
-- show as unpressed in 0.25 seconds
|
|
|
|
tcd.dispatch(0.25, function () draw(false) end)
|
2024-05-10 03:05:55 +00:00
|
|
|
tabs[cur_idx].callback()
|
2023-09-17 01:06:16 +00:00
|
|
|
elseif event.type == MOUSE_CLICK.DOWN then
|
2023-05-10 15:46:06 +00:00
|
|
|
draw(true, cur_idx)
|
2023-09-17 01:06:16 +00:00
|
|
|
elseif event.type == MOUSE_CLICK.UP then
|
2023-05-23 23:51:48 +00:00
|
|
|
if cur_idx == ini_idx and e.in_frame_bounds(event.current.x, event.current.y) then
|
2023-05-10 15:46:06 +00:00
|
|
|
e.value = cur_idx
|
|
|
|
draw(false)
|
2024-05-10 03:05:55 +00:00
|
|
|
tabs[cur_idx].callback()
|
2023-05-10 15:46:06 +00:00
|
|
|
else draw(false) end
|
|
|
|
end
|
2023-09-17 01:06:16 +00:00
|
|
|
elseif event.type == MOUSE_CLICK.UP then
|
2023-05-18 14:58:42 +00:00
|
|
|
draw(false)
|
2023-04-12 16:40:13 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- set the value
|
|
|
|
---@param val integer new value
|
|
|
|
function e.set_value(val)
|
|
|
|
e.value = val
|
|
|
|
draw(false)
|
|
|
|
end
|
|
|
|
|
2024-05-10 03:05:55 +00:00
|
|
|
-- update the sidebar nav options
|
|
|
|
---@param items table sidebar entries
|
|
|
|
function e.on_update(items)
|
|
|
|
local next_y = 1
|
|
|
|
|
|
|
|
tabs = {}
|
|
|
|
|
|
|
|
for i = 1, #items do
|
|
|
|
local item = items[i]
|
|
|
|
local height = util.trinary(item.tall, 3, 1)
|
|
|
|
|
|
|
|
---@class sidebar_tab
|
|
|
|
local entry = {
|
|
|
|
y_start = next_y,
|
|
|
|
y_end = next_y + height - 1,
|
|
|
|
tall = item.tall,
|
|
|
|
label = item.label,
|
|
|
|
color = item.color,
|
|
|
|
callback = item.callback
|
|
|
|
}
|
|
|
|
|
|
|
|
next_y = next_y + height
|
|
|
|
|
|
|
|
tabs[i] = entry
|
|
|
|
end
|
|
|
|
|
|
|
|
draw()
|
|
|
|
end
|
|
|
|
|
2023-09-29 23:34:10 +00:00
|
|
|
-- element redraw
|
|
|
|
e.redraw = draw
|
|
|
|
|
|
|
|
e.redraw()
|
2023-04-12 16:40:13 +00:00
|
|
|
|
2023-05-30 23:51:10 +00:00
|
|
|
return e.complete()
|
2023-04-12 16:40:13 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return sidebar
|