diff options
author | Maxim Devaev <[email protected]> | 2024-12-30 18:55:59 +0200 |
---|---|---|
committer | Maxim Devaev <[email protected]> | 2024-12-30 18:55:59 +0200 |
commit | fed3bf1efdea7528919eb9f318c5ada2dcf35410 (patch) | |
tree | f1bd27d882b85cf845b92b489b1c494c4d1dd715 /kvmd | |
parent | d52bb34bb9f5bc669f6b915ba585461356128e39 (diff) |
pikvm/pikvm#1334: Bad link mode for keyboard events
Diffstat (limited to 'kvmd')
-rw-r--r-- | kvmd/apps/kvmd/api/hid.py | 17 | ||||
-rw-r--r-- | kvmd/clients/kvmd.py | 6 | ||||
-rw-r--r-- | kvmd/keyboard/mappings.py | 2 | ||||
-rw-r--r-- | kvmd/plugins/hid/__init__.py | 9 |
4 files changed, 23 insertions, 11 deletions
diff --git a/kvmd/apps/kvmd/api/hid.py b/kvmd/apps/kvmd/api/hid.py index 5ba22abb..98b96313 100644 --- a/kvmd/apps/kvmd/api/hid.py +++ b/kvmd/apps/kvmd/api/hid.py @@ -149,16 +149,17 @@ class HidApi: async def __ws_bin_key_handler(self, _: WsSession, data: bytes) -> None: try: key = valid_hid_key(data[1:].decode("ascii")) - state = valid_bool(data[0]) + state = bool(data[0] & 0b01) + finish = bool(data[0] & 0b10) except Exception: return - self.__hid.send_key_event(key, state) + self.__hid.send_key_event(key, state, finish) @exposed_ws(2) async def __ws_bin_mouse_button_handler(self, _: WsSession, data: bytes) -> None: try: button = valid_hid_mouse_button(data[1:].decode("ascii")) - state = valid_bool(data[0]) + state = bool(data[0] & 0b01) except Exception: return self.__hid.send_mouse_button_event(button, state) @@ -183,7 +184,7 @@ class HidApi: def __process_ws_bin_delta_request(self, data: bytes, handler: Callable[[Iterable[tuple[int, int]], bool], None]) -> None: try: - squash = valid_bool(data[0]) + squash = bool(data[0] & 0b01) data = data[1:] deltas: list[tuple[int, int]] = [] for index in range(0, len(data), 2): @@ -200,9 +201,10 @@ class HidApi: try: key = valid_hid_key(event["key"]) state = valid_bool(event["state"]) + finish = valid_bool(event.get("finish", False)) except Exception: return - self.__hid.send_key_event(key, state) + self.__hid.send_key_event(key, state, finish) @exposed_ws("mouse_button") async def __ws_mouse_button_handler(self, _: WsSession, event: dict) -> None: @@ -249,9 +251,10 @@ class HidApi: key = valid_hid_key(req.query.get("key")) if "state" in req.query: state = valid_bool(req.query["state"]) - self.__hid.send_key_event(key, state) + finish = valid_bool(req.query.get("finish", False)) + self.__hid.send_key_event(key, state, finish) else: - await self.__hid.send_key_events([(key, True), (key, False)], slow=True) + self.__hid.send_key_event(key, True, True) return make_json_response() @exposed_http("POST", "/hid/events/send_mouse_button") diff --git a/kvmd/clients/kvmd.py b/kvmd/clients/kvmd.py index a0412ce2..d9b38339 100644 --- a/kvmd/clients/kvmd.py +++ b/kvmd/clients/kvmd.py @@ -183,10 +183,12 @@ class KvmdClientWs: self.__communicated = False async def send_key_event(self, key: str, state: bool) -> None: - await self.__writer_queue.put(bytes([1, state]) + key.encode("ascii")) + mask = (0b01 if state else 0) + await self.__writer_queue.put(bytes([1, mask]) + key.encode("ascii")) async def send_mouse_button_event(self, button: str, state: bool) -> None: - await self.__writer_queue.put(bytes([2, state]) + button.encode("ascii")) + mask = (0b01 if state else 0) + await self.__writer_queue.put(bytes([2, mask]) + button.encode("ascii")) async def send_mouse_move_event(self, to_x: int, to_y: int) -> None: await self.__writer_queue.put(struct.pack(">bhh", 3, to_x, to_y)) diff --git a/kvmd/keyboard/mappings.py b/kvmd/keyboard/mappings.py index 8e7d87d1..bfefa712 100644 --- a/kvmd/keyboard/mappings.py +++ b/kvmd/keyboard/mappings.py @@ -170,6 +170,8 @@ class WebModifiers: CTRL_RIGHT = "ControlRight" CTRLS = set([CTRL_RIGHT, CTRL_RIGHT]) + ALL = (SHIFTS | ALTS | CTRLS) + class X11Modifiers: SHIFT_LEFT = 65505 diff --git a/kvmd/plugins/hid/__init__.py b/kvmd/plugins/hid/__init__.py index 447b4d07..5063f53e 100644 --- a/kvmd/plugins/hid/__init__.py +++ b/kvmd/plugins/hid/__init__.py @@ -37,6 +37,7 @@ from ...validators.basic import valid_string_list from ...validators.hid import valid_hid_key from ...validators.hid import valid_hid_mouse_move +from ...keyboard.mappings import WebModifiers from ...mouse import MouseRange from .. import BasePlugin @@ -148,10 +149,14 @@ class BaseHid(BasePlugin): # pylint: disable=too-many-instance-attributes if no_ignore_keys or key not in self.__ignore_keys: if slow: await asyncio.sleep(0.02) - self.send_key_event(key, state) + self.send_key_event(key, state, False) - def send_key_event(self, key: str, state: bool) -> None: + def send_key_event(self, key: str, state: bool, finish: bool) -> None: self._send_key_event(key, state) + if state and finish and (key not in WebModifiers.ALL and key != "PrintScreen"): + # Считаем что PrintScreen это модификатор для Alt+SysRq+... + # По-хорошему надо учитывать факт нажатия на Alt, но можно и забить. + self._send_key_event(key, False) self.__bump_activity() def _send_key_event(self, key: str, state: bool) -> None: |