diff options
author | Devaev Maxim <[email protected]> | 2019-10-01 05:48:20 +0300 |
---|---|---|
committer | Devaev Maxim <[email protected]> | 2019-10-01 05:48:20 +0300 |
commit | 8cc5f631cafe84794c878244649ab11087631efd (patch) | |
tree | cca3e02fe967f5291b2ff791ad871e9e1fcf990e | |
parent | 74931a8096aed89ef7d03e327978d1a2963c92c7 (diff) |
horizontal scrolling for otg
-rw-r--r-- | kvmd/apps/kvmd/server.py | 3 | ||||
-rw-r--r-- | kvmd/apps/otg/hid/mouse.py | 11 | ||||
-rw-r--r-- | kvmd/plugins/hid/__init__.py | 2 | ||||
-rw-r--r-- | kvmd/plugins/hid/otg/__init__.py | 4 | ||||
-rw-r--r-- | kvmd/plugins/hid/otg/mouse.py | 25 | ||||
-rw-r--r-- | kvmd/plugins/hid/serial.py | 7 | ||||
-rw-r--r-- | web/share/js/kvm/mouse.js | 10 |
7 files changed, 41 insertions, 21 deletions
diff --git a/kvmd/apps/kvmd/server.py b/kvmd/apps/kvmd/server.py index 4f197641..95926470 100644 --- a/kvmd/apps/kvmd/server.py +++ b/kvmd/apps/kvmd/server.py @@ -421,10 +421,11 @@ class Server: # pylint: disable=too-many-instance-attributes async def __handle_ws_mouse_wheel_event(self, event: Dict) -> None: try: + delta_x = valid_hid_mouse_wheel(event["delta"]["x"]) delta_y = valid_hid_mouse_wheel(event["delta"]["y"]) except Exception: return - await self.__hid.send_mouse_wheel_event(delta_y) + await self.__hid.send_mouse_wheel_event(delta_x, delta_y) # ===== HID diff --git a/kvmd/apps/otg/hid/mouse.py b/kvmd/apps/otg/hid/mouse.py index 325d27ef..bed1ffb2 100644 --- a/kvmd/apps/otg/hid/mouse.py +++ b/kvmd/apps/otg/hid/mouse.py @@ -28,7 +28,7 @@ MOUSE_HID = Hid( protocol=0, subclass=0, - report_length=6, + report_length=7, report_descriptor=bytes([ # https://github.com/NicoHood/HID/blob/0835e6a/src/SingleReport/SingleAbsoluteMouse.cpp @@ -71,6 +71,15 @@ MOUSE_HID = Hid( 0x95, 0x01, # REPORT_COUNT (1) 0x81, 0x06, # INPUT (Data,Var,Rel) + # Horizontal wheel + 0x05, 0x0C, # USAGE PAGE (Consumer Devices) + 0x0A, 0x38, 0x02, # USAGE (AC Pan) + 0x15, 0x81, # LOGICAL_MINIMUM (-127) + 0x25, 0x7F, # LOGICAL_MAXIMUM (127) + 0x75, 0x08, # REPORT_SIZE (8) + 0x95, 0x01, # REPORT_COUNT (1) + 0x81, 0x06, # INPUT (Data,Var,Rel) + # End 0xC0, # END_COLLECTION ]), diff --git a/kvmd/plugins/hid/__init__.py b/kvmd/plugins/hid/__init__.py index bb4ec677..5bc5ca52 100644 --- a/kvmd/plugins/hid/__init__.py +++ b/kvmd/plugins/hid/__init__.py @@ -57,7 +57,7 @@ class BaseHid(BasePlugin): async def send_mouse_button_event(self, button: str, state: bool) -> None: raise NotImplementedError - async def send_mouse_wheel_event(self, delta_y: int) -> None: + async def send_mouse_wheel_event(self, delta_x: int, delta_y: int) -> None: raise NotImplementedError async def clear_events(self) -> None: diff --git a/kvmd/plugins/hid/otg/__init__.py b/kvmd/plugins/hid/otg/__init__.py index 7129e5d4..1d05199b 100644 --- a/kvmd/plugins/hid/otg/__init__.py +++ b/kvmd/plugins/hid/otg/__init__.py @@ -121,8 +121,8 @@ class Plugin(BaseHid): async def send_mouse_move_event(self, to_x: int, to_y: int) -> None: self.__mouse_proc.send_move_event(to_x, to_y) - async def send_mouse_wheel_event(self, delta_y: int) -> None: - self.__mouse_proc.send_wheel_event(delta_y) + async def send_mouse_wheel_event(self, delta_x: int, delta_y: int) -> None: + self.__mouse_proc.send_wheel_event(delta_x, delta_y) async def clear_events(self) -> None: self.__keyboard_proc.send_clear_event() diff --git a/kvmd/plugins/hid/otg/mouse.py b/kvmd/plugins/hid/otg/mouse.py index 4132a1cc..1c0ad1df 100644 --- a/kvmd/plugins/hid/otg/mouse.py +++ b/kvmd/plugins/hid/otg/mouse.py @@ -66,9 +66,11 @@ class _MoveEvent(BaseEvent): @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 @@ -86,7 +88,7 @@ class MouseProcess(DeviceProcess): get_logger().info("Clearing HID-mouse events ...") if self._ensure_device(): try: - self._write_report(self.__make_report(0, self.__x, self.__y, 0)) # Release all buttons + self._write_report(self.__make_report(0, self.__x, self.__y, 0, 0)) # Release all buttons finally: self._close_device() @@ -103,8 +105,8 @@ class MouseProcess(DeviceProcess): def send_move_event(self, to_x: int, to_y: int) -> None: self._queue_event(_MoveEvent(to_x, to_y)) - def send_wheel_event(self, delta_y: int) -> None: - self._queue_event(_WheelEvent(delta_y)) + def send_wheel_event(self, delta_x: int, delta_y: int) -> None: + self._queue_event(_WheelEvent(delta_x, delta_y)) # ===== @@ -124,34 +126,35 @@ class MouseProcess(DeviceProcess): self.__clear_state() if reopen: self._close_device() - self.__send_current_state(0) + self.__send_current_state(0, 0) def __process_button_event(self, event: _ButtonEvent) -> None: if event.code & self.__pressed_buttons: # Ранее нажатую кнопку отжимаем self.__pressed_buttons &= ~event.code - if not self.__send_current_state(0): + if not self.__send_current_state(0, 0): return if event.state: # Нажимаем если нужно self.__pressed_buttons |= event.code - self.__send_current_state(0) + self.__send_current_state(0, 0) def __process_move_event(self, event: _MoveEvent) -> None: self.__x = event.to_x self.__y = event.to_y - self.__send_current_state(0) + self.__send_current_state(0, 0) def __process_wheel_event(self, event: _WheelEvent) -> None: - self.__send_current_state(event.delta_y) + self.__send_current_state(event.delta_x, event.delta_y) - def __send_current_state(self, delta_y: int) -> bool: + def __send_current_state(self, delta_x: int, delta_y: int) -> bool: ok = False if self._ensure_device(): ok = self._write_report(self.__make_report( buttons=self.__pressed_buttons, to_x=self.__x, to_y=self.__y, + delta_x=delta_x, delta_y=delta_y, )) if not ok: @@ -163,7 +166,7 @@ class MouseProcess(DeviceProcess): self.__x = 0 self.__y = 0 - def __make_report(self, buttons: int, to_x: int, to_y: int, delta_y: int) -> bytes: + 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 - return struct.pack("<BHHb", buttons, to_x, to_y, delta_y) + return struct.pack("<BHHbb", buttons, to_x, to_y, delta_y, delta_x) diff --git a/kvmd/plugins/hid/serial.py b/kvmd/plugins/hid/serial.py index 40eb3268..9a5995e9 100644 --- a/kvmd/plugins/hid/serial.py +++ b/kvmd/plugins/hid/serial.py @@ -119,10 +119,11 @@ class _MouseButtonEvent(_BoolEvent): @dataclasses.dataclass(frozen=True) class _MouseWheelEvent(_IntEvent): def __post_init__(self) -> None: - assert self.x == 0 # Горизонтальная прокрутка пока не поддерживается + assert -127 <= self.x <= 127 assert -127 <= self.y <= 127 def make_command(self) -> bytes: + # Горизонтальная прокрутка пока не поддерживается return b"\x14\x00" + struct.pack(">b", self.y) + b"\x00\x00" @@ -253,8 +254,8 @@ class Plugin(BaseHid, multiprocessing.Process): # pylint: disable=too-many-inst async def send_mouse_button_event(self, button: str, state: bool) -> None: await self.__send_bool_event(_MouseButtonEvent(button, state), self.__pressed_mouse_buttons) - async def send_mouse_wheel_event(self, delta_y: int) -> None: - await self.__send_int_event(_MouseWheelEvent(0, delta_y)) + async def send_mouse_wheel_event(self, delta_x: int, delta_y: int) -> None: + await self.__send_int_event(_MouseWheelEvent(delta_x, delta_y)) async def clear_events(self) -> None: if not self.__stop_event.is_set(): diff --git a/web/share/js/kvm/mouse.js b/web/share/js/kvm/mouse.js index 85efa199..03d90a06 100644 --- a/web/share/js/kvm/mouse.js +++ b/web/share/js/kvm/mouse.js @@ -190,13 +190,19 @@ export function Mouse() { let delta = {x: 0, y: 0}; - __wheel_delta.y += event.deltaY; + __wheel_delta.x += event.deltaX; // Horizontal scrolling + if (Math.abs(__wheel_delta.x) >= 100) { + delta.x = __wheel_delta.x / Math.abs(__wheel_delta.x) * (-5); + __wheel_delta.x = 0; + } + + __wheel_delta.y += event.deltaY; // Vertical scrolling if (Math.abs(__wheel_delta.y) >= 100) { delta.y = __wheel_delta.y / Math.abs(__wheel_delta.y) * (-5); __wheel_delta.y = 0; } - if (delta.y) { + if (delta.x || delta.y) { tools.debug("Mouse: scrolled:", delta); if (__ws) { __ws.send(JSON.stringify({ |