2023-05-10 00:29:07 +00:00
|
|
|
--
|
|
|
|
-- Graphics Events and Event Handlers
|
|
|
|
--
|
|
|
|
|
|
|
|
local util = require("scada-common.util")
|
|
|
|
|
|
|
|
local events = {}
|
|
|
|
|
|
|
|
---@enum CLICK_BUTTON
|
|
|
|
events.CLICK_BUTTON = {
|
|
|
|
GENERIC = 0,
|
|
|
|
LEFT_BUTTON = 1,
|
|
|
|
RIGHT_BUTTON = 2,
|
|
|
|
MID_BUTTON = 3
|
|
|
|
}
|
|
|
|
|
2023-09-17 01:06:16 +00:00
|
|
|
---@enum MOUSE_CLICK
|
|
|
|
local MOUSE_CLICK = {
|
2023-05-10 00:29:07 +00:00
|
|
|
TAP = 1, -- screen tap (complete click)
|
|
|
|
DOWN = 2, -- button down
|
|
|
|
UP = 3, -- button up (completed a click)
|
|
|
|
DRAG = 4, -- mouse dragged
|
|
|
|
SCROLL_DOWN = 5, -- scroll down
|
2023-05-18 14:58:42 +00:00
|
|
|
SCROLL_UP = 6 -- scroll up
|
2023-05-10 00:29:07 +00:00
|
|
|
}
|
|
|
|
|
2023-09-17 01:06:16 +00:00
|
|
|
events.MOUSE_CLICK = MOUSE_CLICK
|
|
|
|
|
|
|
|
---@enum KEY_CLICK
|
|
|
|
local KEY_CLICK = {
|
|
|
|
DOWN = 1,
|
|
|
|
HELD = 2,
|
|
|
|
UP = 3,
|
|
|
|
CHAR = 4
|
|
|
|
}
|
|
|
|
|
|
|
|
events.KEY_CLICK = KEY_CLICK
|
|
|
|
|
2023-05-11 23:55:02 +00:00
|
|
|
-- create a new 2D coordinate
|
|
|
|
---@param x integer
|
|
|
|
---@param y integer
|
|
|
|
---@return coordinate_2d
|
|
|
|
local function _coord2d(x, y) return { x = x, y = y } end
|
|
|
|
|
2023-09-05 19:32:45 +00:00
|
|
|
events.new_coord_2d = _coord2d
|
|
|
|
|
2023-05-10 00:29:07 +00:00
|
|
|
---@class mouse_interaction
|
|
|
|
---@field monitor string
|
|
|
|
---@field button CLICK_BUTTON
|
2023-09-17 01:06:16 +00:00
|
|
|
---@field type MOUSE_CLICK
|
2023-05-10 00:29:07 +00:00
|
|
|
---@field initial coordinate_2d
|
|
|
|
---@field current coordinate_2d
|
|
|
|
|
2023-09-17 01:06:16 +00:00
|
|
|
---@class key_interaction
|
|
|
|
---@field type KEY_CLICK
|
|
|
|
---@field key number key code
|
|
|
|
---@field name string key character name
|
|
|
|
---@field shift boolean shift held
|
|
|
|
---@field ctrl boolean ctrl held
|
|
|
|
---@field alt boolean alt held
|
|
|
|
|
2023-05-10 00:29:07 +00:00
|
|
|
local handler = {
|
2023-05-11 23:55:02 +00:00
|
|
|
-- left, right, middle button down tracking
|
2023-09-17 01:06:16 +00:00
|
|
|
button_down = { _coord2d(0, 0), _coord2d(0, 0), _coord2d(0, 0) },
|
|
|
|
-- keyboard modifiers
|
|
|
|
shift = false,
|
|
|
|
alt = false,
|
|
|
|
ctrl = false
|
2023-05-10 00:29:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
-- create a new monitor touch mouse interaction event
|
|
|
|
---@nodiscard
|
|
|
|
---@param monitor string
|
|
|
|
---@param x integer
|
|
|
|
---@param y integer
|
|
|
|
---@return mouse_interaction
|
|
|
|
local function _monitor_touch(monitor, x, y)
|
|
|
|
return {
|
|
|
|
monitor = monitor,
|
|
|
|
button = events.CLICK_BUTTON.GENERIC,
|
2023-09-17 01:06:16 +00:00
|
|
|
type = MOUSE_CLICK.TAP,
|
2023-05-10 00:29:07 +00:00
|
|
|
initial = _coord2d(x, y),
|
|
|
|
current = _coord2d(x, y)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create a new mouse button mouse interaction event
|
|
|
|
---@nodiscard
|
|
|
|
---@param button CLICK_BUTTON mouse button
|
2023-09-17 01:06:16 +00:00
|
|
|
---@param type MOUSE_CLICK click type
|
2023-05-10 00:29:07 +00:00
|
|
|
---@param x1 integer initial x
|
|
|
|
---@param y1 integer initial y
|
|
|
|
---@param x2 integer current x
|
|
|
|
---@param y2 integer current y
|
|
|
|
---@return mouse_interaction
|
|
|
|
local function _mouse_event(button, type, x1, y1, x2, y2)
|
|
|
|
return {
|
|
|
|
monitor = "terminal",
|
|
|
|
button = button,
|
|
|
|
type = type,
|
|
|
|
initial = _coord2d(x1, y1),
|
|
|
|
current = _coord2d(x2, y2)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create a new generic mouse interaction event
|
|
|
|
---@nodiscard
|
2023-09-17 01:06:16 +00:00
|
|
|
---@param type MOUSE_CLICK
|
2023-05-10 00:29:07 +00:00
|
|
|
---@param x integer
|
|
|
|
---@param y integer
|
|
|
|
---@return mouse_interaction
|
|
|
|
function events.mouse_generic(type, x, y)
|
|
|
|
return {
|
|
|
|
monitor = "",
|
|
|
|
button = events.CLICK_BUTTON.GENERIC,
|
|
|
|
type = type,
|
|
|
|
initial = _coord2d(x, y),
|
|
|
|
current = _coord2d(x, y)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create a new transposed mouse interaction event using the event's monitor/button fields
|
|
|
|
---@nodiscard
|
|
|
|
---@param event mouse_interaction
|
|
|
|
---@param elem_pos_x integer element's x position: new x = (event x - element x) + 1
|
|
|
|
---@param elem_pos_y integer element's y position: new y = (event y - element y) + 1
|
|
|
|
---@return mouse_interaction
|
|
|
|
function events.mouse_transposed(event, elem_pos_x, elem_pos_y)
|
|
|
|
return {
|
|
|
|
monitor = event.monitor,
|
|
|
|
button = event.button,
|
|
|
|
type = event.type,
|
|
|
|
initial = _coord2d((event.initial.x - elem_pos_x) + 1, (event.initial.y - elem_pos_y) + 1),
|
|
|
|
current = _coord2d((event.current.x - elem_pos_x) + 1, (event.current.y - elem_pos_y) + 1)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2023-05-10 15:08:24 +00:00
|
|
|
-- check if an event qualifies as a click (tap or up)
|
|
|
|
---@nodiscard
|
2023-09-17 01:06:16 +00:00
|
|
|
---@param t MOUSE_CLICK
|
|
|
|
function events.was_clicked(t) return t == MOUSE_CLICK.TAP or t == MOUSE_CLICK.UP end
|
2023-05-10 15:08:24 +00:00
|
|
|
|
2023-05-10 00:29:07 +00:00
|
|
|
-- create a new mouse event to pass onto graphics renderer<br>
|
|
|
|
-- supports: mouse_click, mouse_up, mouse_drag, mouse_scroll, and monitor_touch
|
|
|
|
---@param event_type os_event OS event to handle
|
|
|
|
---@param opt integer|string button, scroll direction, or monitor for monitor touch
|
|
|
|
---@param x integer x coordinate
|
|
|
|
---@param y integer y coordinate
|
|
|
|
---@return mouse_interaction|nil
|
|
|
|
function events.new_mouse_event(event_type, opt, x, y)
|
|
|
|
if event_type == "mouse_click" then
|
|
|
|
---@cast opt 1|2|3
|
2023-05-11 23:55:02 +00:00
|
|
|
handler.button_down[opt] = _coord2d(x, y)
|
2023-09-17 01:06:16 +00:00
|
|
|
return _mouse_event(opt, MOUSE_CLICK.DOWN, x, y, x, y)
|
2023-05-10 00:29:07 +00:00
|
|
|
elseif event_type == "mouse_up" then
|
|
|
|
---@cast opt 1|2|3
|
|
|
|
local initial = handler.button_down[opt] ---@type coordinate_2d
|
2023-09-17 01:06:16 +00:00
|
|
|
return _mouse_event(opt, MOUSE_CLICK.UP, initial.x, initial.y, x, y)
|
2023-05-10 00:29:07 +00:00
|
|
|
elseif event_type == "monitor_touch" then
|
|
|
|
---@cast opt string
|
|
|
|
return _monitor_touch(opt, x, y)
|
|
|
|
elseif event_type == "mouse_drag" then
|
|
|
|
---@cast opt 1|2|3
|
|
|
|
local initial = handler.button_down[opt] ---@type coordinate_2d
|
2023-09-17 01:06:16 +00:00
|
|
|
return _mouse_event(opt, MOUSE_CLICK.DRAG, initial.x, initial.y, x, y)
|
2023-05-10 00:29:07 +00:00
|
|
|
elseif event_type == "mouse_scroll" then
|
|
|
|
---@cast opt 1|-1
|
2023-09-17 01:06:16 +00:00
|
|
|
local scroll_direction = util.trinary(opt == 1, MOUSE_CLICK.SCROLL_DOWN, MOUSE_CLICK.SCROLL_UP)
|
2023-05-10 00:29:07 +00:00
|
|
|
return _mouse_event(events.CLICK_BUTTON.GENERIC, scroll_direction, x, y, x, y)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-09-17 01:06:16 +00:00
|
|
|
-- create a new keyboard interaction event
|
|
|
|
---@nodiscard
|
|
|
|
---@param click_type KEY_CLICK key click type
|
|
|
|
---@param key integer|string keyboard key code or character for 'char' event
|
|
|
|
---@return key_interaction
|
|
|
|
local function _key_event(click_type, key)
|
|
|
|
local name = key
|
|
|
|
if type(key) == "number" then name = keys.getName(key) end
|
|
|
|
return { type = click_type, key = key, name = name, shift = handler.shift, ctrl = handler.ctrl, alt = handler.alt }
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create a new keyboard event to pass onto graphics renderer<br>
|
2023-05-10 00:29:07 +00:00
|
|
|
-- supports: char, key, and key_up
|
2023-09-17 01:06:16 +00:00
|
|
|
---@param event_type os_event OS event to handle
|
|
|
|
---@param key integer keyboard key code
|
|
|
|
---@param held boolean? if the key is being held (for 'key' event)
|
|
|
|
---@return key_interaction|nil
|
|
|
|
function events.new_key_event(event_type, key, held)
|
2023-05-10 00:29:07 +00:00
|
|
|
if event_type == "char" then
|
2023-09-17 01:06:16 +00:00
|
|
|
return _key_event(KEY_CLICK.CHAR, key)
|
2023-05-10 00:29:07 +00:00
|
|
|
elseif event_type == "key" then
|
2023-09-17 01:06:16 +00:00
|
|
|
if key == keys.leftShift or key == keys.rightShift then
|
|
|
|
handler.shift = true
|
|
|
|
elseif key == keys.leftCtrl or key == keys.rightCtrl then
|
|
|
|
handler.ctrl = true
|
|
|
|
elseif key == keys.leftAlt or key == keys.rightAlt then
|
|
|
|
handler.alt = true
|
|
|
|
else
|
|
|
|
return _key_event(util.trinary(held, KEY_CLICK.HELD, KEY_CLICK.DOWN), key)
|
|
|
|
end
|
2023-05-10 00:29:07 +00:00
|
|
|
elseif event_type == "key_up" then
|
2023-09-17 01:06:16 +00:00
|
|
|
if key == keys.leftShift or key == keys.rightShift then
|
|
|
|
handler.shift = false
|
|
|
|
elseif key == keys.leftCtrl or key == keys.rightCtrl then
|
|
|
|
handler.ctrl = false
|
|
|
|
elseif key == keys.leftAlt or key == keys.rightAlt then
|
|
|
|
handler.alt = false
|
|
|
|
else
|
|
|
|
return _key_event(KEY_CLICK.UP, key)
|
|
|
|
end
|
2023-05-10 00:29:07 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return events
|