summaryrefslogtreecommitdiff
path: root/kvmd/plugins/hid
diff options
context:
space:
mode:
Diffstat (limited to 'kvmd/plugins/hid')
-rw-r--r--kvmd/plugins/hid/otg/keyboard.py79
-rw-r--r--kvmd/plugins/hid/otg/mouse.py36
2 files changed, 64 insertions, 51 deletions
diff --git a/kvmd/plugins/hid/otg/keyboard.py b/kvmd/plugins/hid/otg/keyboard.py
index cd8a1f5f..fab7b992 100644
--- a/kvmd/plugins/hid/otg/keyboard.py
+++ b/kvmd/plugins/hid/otg/keyboard.py
@@ -45,10 +45,22 @@ class _ResetEvent(BaseEvent):
@dataclasses.dataclass(frozen=True)
+class _ModifierEvent(BaseEvent):
+ modifier: keymap.OtgKey
+ state: bool
+
+ def __post_init__(self) -> None:
+ assert self.modifier.is_modifier
+
+
[email protected](frozen=True)
class _KeyEvent(BaseEvent):
key: keymap.OtgKey
state: bool
+ def __post_init__(self) -> None:
+ assert not self.key.is_modifier
+
# =====
class KeyboardProcess(BaseDeviceProcess):
@@ -74,8 +86,11 @@ class KeyboardProcess(BaseDeviceProcess):
self._queue_event(_ResetEvent())
def send_key_event(self, key: str, state: bool) -> None:
- assert key in keymap.KEYMAP
- self._queue_event(_KeyEvent(key=keymap.KEYMAP[key].otg, state=state))
+ otg_key = keymap.KEYMAP[key].otg
+ if otg_key.is_modifier:
+ self._queue_event(_ModifierEvent(otg_key, state))
+ else:
+ self._queue_event(_KeyEvent(otg_key, state))
# =====
@@ -84,6 +99,8 @@ class KeyboardProcess(BaseDeviceProcess):
self.__process_clear_event()
elif isinstance(event, _ResetEvent):
self.__process_clear_event(reopen=True)
+ elif isinstance(event, _ModifierEvent):
+ self.__process_modifier_event(event)
elif isinstance(event, _KeyEvent):
self.__process_key_event(event)
@@ -94,41 +111,41 @@ class KeyboardProcess(BaseDeviceProcess):
self._close_device()
self.__send_current_state()
+ def __process_modifier_event(self, event: _ModifierEvent) -> None:
+ if event.modifier in self.__pressed_modifiers:
+ # Ранее нажатый модификатор отжимаем
+ self.__pressed_modifiers.remove(event.modifier)
+ if not self.__send_current_state():
+ return
+ if event.state:
+ # Нажимаем если нужно
+ self.__pressed_modifiers.add(event.modifier)
+ self.__send_current_state()
+
def __process_key_event(self, event: _KeyEvent) -> None:
- if event.key.is_modifier:
- if event.key in self.__pressed_modifiers:
- # Ранее нажатый модификатор отжимаем
- self.__pressed_modifiers.remove(event.key)
- if not self.__send_current_state():
- return
- if event.state:
- # Нажимаем если нужно
- self.__pressed_modifiers.add(event.key)
- self.__send_current_state()
-
- else: # regular key
- if event.key in self.__pressed_keys:
- # Ранее нажатую клавишу отжимаем
- self.__pressed_keys[self.__pressed_keys.index(event.key)] = None
- if not self.__send_current_state():
- return
- elif event.state and None not in self.__pressed_keys:
- # Если нужно нажать что-то новое, но свободных слотов нет - отжимаем всё
- self.__clear_keys()
- if not self.__send_current_state():
- return
- if event.state:
- # Нажимаем если нужно
- self.__pressed_keys[self.__pressed_keys.index(None)] = event.key
- self.__send_current_state()
+ if event.key in self.__pressed_keys:
+ # Ранее нажатую клавишу отжимаем
+ self.__pressed_keys[self.__pressed_keys.index(event.key)] = None
+ if not self.__send_current_state():
+ return
+ elif event.state and None not in self.__pressed_keys:
+ # Если нужно нажать что-то новое, но свободных слотов нет - отжимаем всё
+ self.__clear_keys()
+ if not self.__send_current_state():
+ return
+ if event.state:
+ # Нажимаем если нужно
+ self.__pressed_keys[self.__pressed_keys.index(None)] = event.key
+ self.__send_current_state()
+
+ # =====
def __send_current_state(self) -> bool:
ok = False
if self._ensure_device():
modifiers = 0
- for key in self.__pressed_modifiers:
- assert key.is_modifier
- modifiers |= key.code
+ for modifier in self.__pressed_modifiers:
+ modifiers |= modifier.code
assert len(self.__pressed_keys) == 6
keys = [
diff --git a/kvmd/plugins/hid/otg/mouse.py b/kvmd/plugins/hid/otg/mouse.py
index 7c201ab6..4c9facaf 100644
--- a/kvmd/plugins/hid/otg/mouse.py
+++ b/kvmd/plugins/hid/otg/mouse.py
@@ -32,14 +32,6 @@ from .device import BaseDeviceProcess
# =====
-_BUTTONS = {
- "left": 0x1,
- "right": 0x2,
- "middle": 0x4,
-}
-
-
-# =====
class _ClearEvent(BaseEvent):
pass
@@ -59,20 +51,12 @@ class _MoveEvent(BaseEvent):
to_x: int
to_y: int
- def __post_init__(self) -> None:
- assert -32768 <= self.to_x <= 32767
- assert -32768 <= self.to_y <= 32767
-
@dataclasses.dataclass(frozen=True)
class _WheelEvent(BaseEvent):
delta_x: int
delta_y: int
- def __post_init__(self) -> None:
- assert -127 <= self.delta_x <= 127
- assert -127 <= self.delta_y <= 127
-
# =====
class MouseProcess(BaseDeviceProcess):
@@ -99,13 +83,23 @@ class MouseProcess(BaseDeviceProcess):
self._queue_event(_ResetEvent())
def send_button_event(self, button: str, state: bool) -> None:
- assert button in _BUTTONS
- self._queue_event(_ButtonEvent(code=_BUTTONS[button], state=state))
+ code: int = {
+ "left": 0x1,
+ "right": 0x2,
+ "middle": 0x4,
+ }[button]
+ self._queue_event(_ButtonEvent(code, state))
def send_move_event(self, to_x: int, to_y: int) -> None:
+ assert -32768 <= to_x <= 32767
+ assert -32768 <= to_y <= 32767
+ to_x = (to_x + 32768) // 2
+ to_y = (to_y + 32768) // 2
self._queue_event(_MoveEvent(to_x, to_y))
def send_wheel_event(self, delta_x: int, delta_y: int) -> None:
+ assert -127 <= delta_x <= 127
+ assert -127 <= delta_y <= 127
self._queue_event(_WheelEvent(delta_x, delta_y))
# =====
@@ -147,6 +141,8 @@ class MouseProcess(BaseDeviceProcess):
def __process_wheel_event(self, event: _WheelEvent) -> None:
self.__send_current_state(event.delta_x, event.delta_y)
+ # =====
+
def __send_current_state(self, delta_x: int, delta_y: int) -> bool:
ok = False
if self._ensure_device():
@@ -167,6 +163,6 @@ class MouseProcess(BaseDeviceProcess):
self.__y = 0
def __make_report(self, buttons: int, to_x: int, to_y: int, delta_x: int, delta_y: int) -> bytes:
- to_x = (to_x + 32768) // 2
- to_y = (to_y + 32768) // 2
+ # XXX: Delta Y before X: it's ok.
+ # See /kvmd/apps/otg/hid/keyboard.py for details
return struct.pack("<BHHbb", buttons, to_x, to_y, delta_y, delta_x)