2022-06-11 20:44:31 +00:00
|
|
|
-- Spinbox Numeric Graphics Element
|
|
|
|
|
|
|
|
local util = require("scada-common.util")
|
|
|
|
|
2022-11-25 03:49:35 +00:00
|
|
|
local element = require("graphics.element")
|
|
|
|
|
2022-06-11 20:44:31 +00:00
|
|
|
---@class spinbox_args
|
|
|
|
---@field default? number default value, defaults to 0.0
|
2022-11-17 16:58:14 +00:00
|
|
|
---@field min? number default 0, currently must be 0 or greater
|
|
|
|
---@field max? number default max number that can be displayed with the digits configuration
|
2022-06-11 20:44:31 +00:00
|
|
|
---@field whole_num_precision integer number of whole number digits
|
|
|
|
---@field fractional_precision integer number of fractional digits
|
|
|
|
---@field arrow_fg_bg cpair arrow foreground/background colors
|
2023-01-23 20:10:41 +00:00
|
|
|
---@field arrow_disable? color color when disabled (default light gray)
|
2022-06-11 20:44:31 +00:00
|
|
|
---@field parent graphics_element
|
2022-07-28 14:09:34 +00:00
|
|
|
---@field id? string element id
|
2022-06-11 20:44:31 +00:00
|
|
|
---@field x? integer 1 if omitted
|
|
|
|
---@field y? integer 1 if omitted
|
2022-06-11 21:06:32 +00:00
|
|
|
---@field fg_bg? cpair foreground/background colors
|
2022-06-11 20:44:31 +00:00
|
|
|
|
2022-07-24 00:08:52 +00:00
|
|
|
-- new spinbox control (minimum value is 0)
|
2022-06-11 20:44:31 +00:00
|
|
|
---@param args spinbox_args
|
2022-07-28 14:09:34 +00:00
|
|
|
---@return graphics_element element, element_id id
|
2022-06-11 20:44:31 +00:00
|
|
|
local function spinbox(args)
|
|
|
|
-- properties
|
|
|
|
local digits = {}
|
|
|
|
local wn_prec = args.whole_num_precision
|
|
|
|
local fr_prec = args.fractional_precision
|
|
|
|
|
2022-06-16 15:19:32 +00:00
|
|
|
assert(util.is_int(wn_prec), "graphics.element.controls.spinbox_numeric: whole number precision must be an integer")
|
|
|
|
assert(util.is_int(fr_prec), "graphics.element.controls.spinbox_numeric: fractional precision must be an integer")
|
2022-06-11 20:44:31 +00:00
|
|
|
|
2023-04-12 20:02:29 +00:00
|
|
|
local fmt, fmt_init ---@type string, string
|
2023-01-23 20:10:41 +00:00
|
|
|
|
|
|
|
if fr_prec > 0 then
|
|
|
|
fmt = "%" .. (wn_prec + fr_prec + 1) .. "." .. fr_prec .. "f"
|
|
|
|
fmt_init = "%0" .. (wn_prec + fr_prec + 1) .. "." .. fr_prec .. "f"
|
|
|
|
else
|
|
|
|
fmt = "%" .. wn_prec .. "d"
|
|
|
|
fmt_init = "%0" .. wn_prec .. "d"
|
|
|
|
end
|
|
|
|
|
2022-07-16 17:25:07 +00:00
|
|
|
local dec_point_x = args.whole_num_precision + 1
|
|
|
|
|
2022-06-11 21:06:32 +00:00
|
|
|
assert(type(args.arrow_fg_bg) == "table", "graphics.element.spinbox_numeric: arrow_fg_bg is a required field")
|
|
|
|
|
2022-06-11 20:44:31 +00:00
|
|
|
-- determine widths
|
|
|
|
args.width = wn_prec + fr_prec + util.trinary(fr_prec > 0, 1, 0)
|
|
|
|
args.height = 3
|
|
|
|
|
|
|
|
-- create new graphics element base object
|
|
|
|
local e = element.new(args)
|
|
|
|
|
2022-09-12 16:59:28 +00:00
|
|
|
-- set initial value
|
2023-01-23 20:10:41 +00:00
|
|
|
e.value = args.default or 0
|
2022-09-12 16:59:28 +00:00
|
|
|
|
2022-06-11 20:44:31 +00:00
|
|
|
-- draw the arrows
|
2023-01-23 20:10:41 +00:00
|
|
|
local function draw_arrows(color)
|
|
|
|
e.window.setBackgroundColor(args.arrow_fg_bg.bkg)
|
|
|
|
e.window.setTextColor(color)
|
|
|
|
e.window.setCursorPos(1, 1)
|
|
|
|
e.window.write(util.strrep("\x1e", wn_prec))
|
|
|
|
e.window.setCursorPos(1, 3)
|
|
|
|
e.window.write(util.strrep("\x1f", wn_prec))
|
|
|
|
if fr_prec > 0 then
|
|
|
|
e.window.setCursorPos(1 + wn_prec, 1)
|
|
|
|
e.window.write(" " .. util.strrep("\x1e", fr_prec))
|
|
|
|
e.window.setCursorPos(1 + wn_prec, 3)
|
|
|
|
e.window.write(" " .. util.strrep("\x1f", fr_prec))
|
|
|
|
end
|
2022-06-11 20:44:31 +00:00
|
|
|
end
|
|
|
|
|
2023-01-23 20:10:41 +00:00
|
|
|
draw_arrows(args.arrow_fg_bg.fgd)
|
|
|
|
|
2022-10-07 15:21:17 +00:00
|
|
|
-- populate digits from current value
|
|
|
|
local function set_digits()
|
|
|
|
local initial_str = util.sprintf(fmt_init, e.value)
|
2022-07-24 00:08:52 +00:00
|
|
|
|
2022-11-17 16:58:14 +00:00
|
|
|
digits = {}
|
2022-10-07 15:21:17 +00:00
|
|
|
---@diagnostic disable-next-line: discard-returns
|
|
|
|
initial_str:gsub("%d", function (char) table.insert(digits, char) end)
|
2022-09-12 16:59:28 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- update the value per digits table
|
|
|
|
local function update_value()
|
|
|
|
e.value = 0
|
|
|
|
for i = 1, #digits do
|
|
|
|
local pow = math.abs(wn_prec - i)
|
|
|
|
if i <= wn_prec then
|
|
|
|
e.value = e.value + (digits[i] * (10 ^ pow))
|
|
|
|
else
|
|
|
|
e.value = e.value + (digits[i] * (10 ^ -pow))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-10-07 15:21:17 +00:00
|
|
|
-- print out the current value
|
|
|
|
local function show_num()
|
|
|
|
-- enforce limits
|
2023-01-15 18:11:46 +00:00
|
|
|
if (type(args.min) == "number") and (e.value < args.min) then
|
2022-11-17 16:58:14 +00:00
|
|
|
e.value = args.min
|
|
|
|
set_digits()
|
2023-01-15 18:11:46 +00:00
|
|
|
elseif e.value < 0 then
|
|
|
|
e.value = 0
|
|
|
|
set_digits()
|
2022-11-17 16:58:14 +00:00
|
|
|
else
|
|
|
|
if string.len(util.sprintf(fmt, e.value)) > args.width then
|
2023-01-15 18:11:46 +00:00
|
|
|
-- max printable exceeded, so max out to all 9s
|
2022-11-17 16:58:14 +00:00
|
|
|
for i = 1, #digits do digits[i] = 9 end
|
|
|
|
update_value()
|
|
|
|
elseif (type(args.max) == "number") and (e.value > args.max) then
|
|
|
|
e.value = args.max
|
|
|
|
set_digits()
|
2023-01-15 18:11:46 +00:00
|
|
|
else
|
|
|
|
set_digits()
|
2022-11-17 16:58:14 +00:00
|
|
|
end
|
2022-09-12 16:59:28 +00:00
|
|
|
end
|
|
|
|
|
2022-10-07 15:21:17 +00:00
|
|
|
-- draw
|
|
|
|
e.window.setBackgroundColor(e.fg_bg.bkg)
|
|
|
|
e.window.setTextColor(e.fg_bg.fgd)
|
|
|
|
e.window.setCursorPos(1, 2)
|
|
|
|
e.window.write(util.sprintf(fmt, e.value))
|
2022-07-16 17:25:07 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- init with the default value
|
|
|
|
show_num()
|
|
|
|
|
2023-04-09 01:33:54 +00:00
|
|
|
-- handle mouse interaction
|
|
|
|
---@param event mouse_interaction mouse event
|
|
|
|
function e.handle_mouse(event)
|
2022-06-11 20:44:31 +00:00
|
|
|
-- only handle if on an increment or decrement arrow
|
2022-10-20 16:22:45 +00:00
|
|
|
if e.enabled and event.x ~= dec_point_x then
|
2022-06-11 20:44:31 +00:00
|
|
|
local idx = util.trinary(event.x > dec_point_x, event.x - 1, event.x)
|
2022-11-17 16:58:14 +00:00
|
|
|
if digits[idx] ~= nil then
|
|
|
|
if event.y == 1 then
|
|
|
|
-- increment
|
|
|
|
digits[idx] = digits[idx] + 1
|
|
|
|
elseif event.y == 3 then
|
|
|
|
-- decrement
|
|
|
|
digits[idx] = digits[idx] - 1
|
|
|
|
end
|
|
|
|
|
|
|
|
update_value()
|
|
|
|
show_num()
|
2022-06-11 20:44:31 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-09-12 16:59:28 +00:00
|
|
|
-- set the value
|
|
|
|
---@param val number number to show
|
|
|
|
function e.set_value(val)
|
|
|
|
e.value = val
|
2022-10-07 15:21:17 +00:00
|
|
|
show_num()
|
2022-09-12 16:59:28 +00:00
|
|
|
end
|
2022-06-11 20:44:31 +00:00
|
|
|
|
2022-11-17 16:58:14 +00:00
|
|
|
-- set minimum input value
|
|
|
|
---@param min integer minimum allowed value
|
|
|
|
function e.set_min(min)
|
2022-11-17 17:00:00 +00:00
|
|
|
if min >= 0 then
|
|
|
|
args.min = min
|
|
|
|
show_num()
|
|
|
|
end
|
2022-11-17 16:58:14 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- set maximum input value
|
|
|
|
---@param max integer maximum allowed value
|
|
|
|
function e.set_max(max)
|
|
|
|
args.max = max
|
2022-11-17 17:00:00 +00:00
|
|
|
show_num()
|
2022-11-17 16:58:14 +00:00
|
|
|
end
|
|
|
|
|
2023-01-23 20:10:41 +00:00
|
|
|
-- enable this input
|
|
|
|
function e.enable()
|
|
|
|
draw_arrows(args.arrow_fg_bg.fgd)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- disable this input
|
|
|
|
function e.disable()
|
|
|
|
draw_arrows(args.arrow_disable or colors.lightGray)
|
|
|
|
end
|
|
|
|
|
2023-01-15 18:11:46 +00:00
|
|
|
-- default to zero, init digits table
|
|
|
|
e.value = 0
|
|
|
|
set_digits()
|
|
|
|
|
2022-07-28 15:17:34 +00:00
|
|
|
return e.get()
|
2022-06-11 20:44:31 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return spinbox
|