From a39ec87a55dad8637e7eddb96614de86abe11c45 Mon Sep 17 00:00:00 2001 From: Mikael Dallaire Cote <110583667+0mdc@users.noreply.github.com> Date: Sun, 17 Mar 2024 23:01:39 -0400 Subject: [PATCH 1/4] Rename MouseKeyCode to MouseButton. --- .../habitat_hitl/_internal/gui_application.py | 37 ++++++++-------- habitat-hitl/habitat_hitl/core/gui_input.py | 22 +++------- habitat-hitl/habitat_hitl/core/key_mapping.py | 27 +++++++++++- .../habitat_hitl/core/remote_client_state.py | 43 +++++++++++++++---- 4 files changed, 85 insertions(+), 44 deletions(-) diff --git a/habitat-hitl/habitat_hitl/_internal/gui_application.py b/habitat-hitl/habitat_hitl/_internal/gui_application.py index 8b85744251..fb1a722422 100644 --- a/habitat-hitl/habitat_hitl/_internal/gui_application.py +++ b/habitat-hitl/habitat_hitl/_internal/gui_application.py @@ -7,6 +7,7 @@ import abc import math import time +from typing import List import magnum as mn from magnum.platform.glfw import Application @@ -32,13 +33,13 @@ def unproject(self, viewport_pos): class InputHandlerApplication(Application): def __init__(self, config): super().__init__(config) - self._gui_inputs = [] + self._gui_inputs: List[GuiInput] = [] - def add_gui_input(self, gui_input): + def add_gui_input(self, gui_input: GuiInput) -> None: self._gui_inputs.append(gui_input) def key_press_event(self, event: Application.KeyEvent) -> None: - key = MagnumKeyConverter.convert(event.key) + key = MagnumKeyConverter.convert_key(event.key) if key: for wrapper in self._gui_inputs: # If the key is already held, this is a repeat press event and we should @@ -48,7 +49,7 @@ def key_press_event(self, event: Application.KeyEvent) -> None: wrapper._key_down.add(key) def key_release_event(self, event: Application.KeyEvent) -> None: - key = MagnumKeyConverter.convert(event.key) + key = MagnumKeyConverter.convert_key(event.key) if key: for wrapper in self._gui_inputs: if key in wrapper._key_held: @@ -56,22 +57,22 @@ def key_release_event(self, event: Application.KeyEvent) -> None: wrapper._key_up.add(key) def mouse_press_event(self, event: Application.MouseEvent) -> None: - mouse_button = event.button - GuiInput.validate_mouse_button(mouse_button) - for wrapper in self._gui_inputs: - wrapper._mouse_button_held.add(mouse_button) - wrapper._mouse_button_down.add(mouse_button) + key = MagnumKeyConverter.convert_mouse_key(event.button) + if key: + for wrapper in self._gui_inputs: + # If the key is already held, this is a repeat press event and we should + # ignore it. + if key not in wrapper._mouse_button_held: + wrapper._mouse_button_held.add(key) + wrapper._mouse_button_down.add(key) def mouse_release_event(self, event: Application.MouseEvent) -> None: - mouse_button = event.button - GuiInput.validate_mouse_button(mouse_button) - for wrapper in self._gui_inputs: - # In theory, mouse_button should always be present in _mouse_button_held. - # In practice, we seem to get spurious release events due to the app - # losing focus (e.g. switching to VS code debugger while mouse-clicking) - if mouse_button in wrapper._mouse_button_held: - wrapper._mouse_button_held.remove(mouse_button) - wrapper._mouse_button_up.add(mouse_button) + key = MagnumKeyConverter.convert_mouse_key(event.button) + if key: + for wrapper in self._gui_inputs: + if key in wrapper._mouse_button_held: + wrapper._mouse_button_held.remove(key) + wrapper._mouse_button_up.add(key) def mouse_scroll_event(self, event: Application.MouseEvent) -> None: # shift+scroll is forced into x direction on mac, seemingly at OS level, diff --git a/habitat-hitl/habitat_hitl/core/gui_input.py b/habitat-hitl/habitat_hitl/core/gui_input.py index 184525759f..29401a2bda 100644 --- a/habitat-hitl/habitat_hitl/core/gui_input.py +++ b/habitat-hitl/habitat_hitl/core/gui_input.py @@ -4,17 +4,7 @@ # This source code is licensed under the MIT license found in the # LICENSE file in the root directory of this source tree. -from habitat_hitl.core.key_mapping import KeyCode - - -class StubNSMeta(type): - def __getattr__(cls, name): - return None - - -# Stub version of Application.MouseEvent.Button -class StubMouseNS(metaclass=StubNSMeta): - pass +from habitat_hitl.core.key_mapping import KeyCode, MouseKeyCode class GuiInput: @@ -25,7 +15,7 @@ class GuiInput: """ KeyNS = KeyCode - MouseNS = StubMouseNS + MouseNS = MouseKeyCode def __init__(self): self._key_held = set() @@ -37,7 +27,7 @@ def __init__(self): self._mouse_button_down = set() self._mouse_button_up = set() self._relative_mouse_position = [0, 0] - self._mouse_scroll_offset = 0 + self._mouse_scroll_offset = 0.0 self._mouse_ray = None def validate_key(key): @@ -59,9 +49,7 @@ def get_key_up(self, key): return key in self._key_up def validate_mouse_button(mouse_button): - # if not do_agnostic_gui_input: - # assert isinstance(mouse_button, Application.MouseEvent.Button) - pass + assert isinstance(mouse_button, MouseKeyCode) def get_mouse_button(self, mouse_button): GuiInput.validate_mouse_button(mouse_button) @@ -99,4 +87,4 @@ def on_frame_end(self): self._mouse_button_down.clear() self._mouse_button_up.clear() self._relative_mouse_position = [0, 0] - self._mouse_scroll_offset = 0 + self._mouse_scroll_offset = 0.0 diff --git a/habitat-hitl/habitat_hitl/core/key_mapping.py b/habitat-hitl/habitat_hitl/core/key_mapping.py index bf2e641b60..ba56d63b85 100644 --- a/habitat-hitl/habitat_hitl/core/key_mapping.py +++ b/habitat-hitl/habitat_hitl/core/key_mapping.py @@ -69,6 +69,18 @@ class KeyCode(IntEnum, metaclass=KeyCodeMetaEnum): # fmt: on +class MouseKeyCode(IntEnum, metaclass=KeyCodeMetaEnum): + """ + Mouse buttons available to control habitat-hitl. + """ + + # fmt: off + LEFT = 0 + RIGHT = 1 + MIDDLE = 2 + # fmt: on + + # On headless systems, we may be unable to import magnum.platform.glfw.Application. try: from magnum.platform.glfw import Application @@ -124,9 +136,22 @@ class KeyCode(IntEnum, metaclass=KeyCodeMetaEnum): # fmt: on } + magnum_mouse_keymap: Dict[Application.KeyEvent.Key, MouseKeyCode] = { + # fmt: off + Application.MouseEvent.Button.LEFT : MouseKeyCode.LEFT , + Application.MouseEvent.Button.RIGHT : MouseKeyCode.RIGHT , + Application.MouseEvent.Button.MIDDLE : MouseKeyCode.MIDDLE, + # fmt: on + } + class MagnumKeyConverter: - def convert(key: Any) -> Optional[KeyCode]: + def convert_key(key: Any) -> Optional[KeyCode]: if magnum_enabled and key in magnum_keymap: return magnum_keymap[key] return None + + def convert_mouse_key(key: Any) -> Optional[MouseKeyCode]: + if magnum_enabled and key in magnum_mouse_keymap: + return magnum_mouse_keymap[key] + return None diff --git a/habitat-hitl/habitat_hitl/core/remote_client_state.py b/habitat-hitl/habitat_hitl/core/remote_client_state.py index 690cf1ac8c..df82d5f1a5 100644 --- a/habitat-hitl/habitat_hitl/core/remote_client_state.py +++ b/habitat-hitl/habitat_hitl/core/remote_client_state.py @@ -156,12 +156,11 @@ def _update_input_state(self, client_states): input_json = ( client_state["input"] if "input" in client_state else None ) - # TODO: Add mouse support - # mouse_json = ( - # client_state["mouse"] if "mouse" in client_state else None - # ) + mouse_json = ( + client_state["mouse"] if "mouse" in client_state else None + ) - if input_json is not None: + if input_json != None: for button in input_json["buttonDown"]: if button not in KeyCode: continue @@ -171,26 +170,54 @@ def _update_input_state(self, client_states): continue self._gui_input._key_up.add(KeyCode(button)) + if mouse_json != None: + mouse_buttons = mouse_json["buttons"] + for button in mouse_buttons["buttonDown"]: + if button not in KeyCode: + continue + self._gui_input._mouse_button_down.add(KeyCode(button)) + for button in mouse_buttons["buttonUp"]: + if button not in KeyCode: + continue + self._gui_input._mouse_button_up.add(KeyCode(button)) + + delta: List[Any] = mouse_json["scrollDelta"] + if len(delta) == 2: + self._gui_input._mouse_scroll_offset += ( + delta[0] if abs(delta[0]) > abs(delta[1]) else delta[1] + ) + # todo: think about ambiguous GuiInput states (key-down and key-up events in the same # frame and other ways that keyHeld, keyDown, and keyUp can be inconsistent. last_client_state = client_states[-1] + input_json = ( last_client_state["input"] if "input" in last_client_state else None ) - # TODO: Add mouse support - # mouse_json = last_client_state["mouse"] if "mouse" in last_client_state else None + mouse_json = ( + last_client_state["mouse"] + if "mouse" in last_client_state + else None + ) self._gui_input._key_held.clear() - if input_json is not None: + if input_json != None: for button in input_json["buttonHeld"]: if button not in KeyCode: continue self._gui_input._key_held.add(KeyCode(button)) + if mouse_json != None: + mouse_buttons = mouse_json["buttons"] + for button in mouse_buttons["buttonHeld"]: + if button not in KeyCode: + continue + self._gui_input._mouse_button_held.add(KeyCode(button)) + def debug_visualize_client(self): """Visualize the received VR inputs (head and hands).""" # Sloppy: Use internal debug_line_render to render on server only. From 48a5a1fafd1c3fd00b56e3f6303029b1fa0480ec Mon Sep 17 00:00:00 2001 From: Mikael Dallaire Cote <110583667+0mdc@users.noreply.github.com> Date: Sun, 17 Mar 2024 23:02:53 -0400 Subject: [PATCH 2/4] Rename MouseKeyCode to MouseButton. --- habitat-hitl/habitat_hitl/core/gui_input.py | 6 +++--- habitat-hitl/habitat_hitl/core/key_mapping.py | 12 ++++++------ 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/habitat-hitl/habitat_hitl/core/gui_input.py b/habitat-hitl/habitat_hitl/core/gui_input.py index 29401a2bda..caf3eb660c 100644 --- a/habitat-hitl/habitat_hitl/core/gui_input.py +++ b/habitat-hitl/habitat_hitl/core/gui_input.py @@ -4,7 +4,7 @@ # This source code is licensed under the MIT license found in the # LICENSE file in the root directory of this source tree. -from habitat_hitl.core.key_mapping import KeyCode, MouseKeyCode +from habitat_hitl.core.key_mapping import KeyCode, MouseButton class GuiInput: @@ -15,7 +15,7 @@ class GuiInput: """ KeyNS = KeyCode - MouseNS = MouseKeyCode + MouseNS = MouseButton def __init__(self): self._key_held = set() @@ -49,7 +49,7 @@ def get_key_up(self, key): return key in self._key_up def validate_mouse_button(mouse_button): - assert isinstance(mouse_button, MouseKeyCode) + assert isinstance(mouse_button, MouseButton) def get_mouse_button(self, mouse_button): GuiInput.validate_mouse_button(mouse_button) diff --git a/habitat-hitl/habitat_hitl/core/key_mapping.py b/habitat-hitl/habitat_hitl/core/key_mapping.py index ba56d63b85..3047a0531a 100644 --- a/habitat-hitl/habitat_hitl/core/key_mapping.py +++ b/habitat-hitl/habitat_hitl/core/key_mapping.py @@ -69,7 +69,7 @@ class KeyCode(IntEnum, metaclass=KeyCodeMetaEnum): # fmt: on -class MouseKeyCode(IntEnum, metaclass=KeyCodeMetaEnum): +class MouseButton(IntEnum, metaclass=KeyCodeMetaEnum): """ Mouse buttons available to control habitat-hitl. """ @@ -136,11 +136,11 @@ class MouseKeyCode(IntEnum, metaclass=KeyCodeMetaEnum): # fmt: on } - magnum_mouse_keymap: Dict[Application.KeyEvent.Key, MouseKeyCode] = { + magnum_mouse_keymap: Dict[Application.KeyEvent.Key, MouseButton] = { # fmt: off - Application.MouseEvent.Button.LEFT : MouseKeyCode.LEFT , - Application.MouseEvent.Button.RIGHT : MouseKeyCode.RIGHT , - Application.MouseEvent.Button.MIDDLE : MouseKeyCode.MIDDLE, + Application.MouseEvent.Button.LEFT : MouseButton.LEFT , + Application.MouseEvent.Button.RIGHT : MouseButton.RIGHT , + Application.MouseEvent.Button.MIDDLE : MouseButton.MIDDLE, # fmt: on } @@ -151,7 +151,7 @@ def convert_key(key: Any) -> Optional[KeyCode]: return magnum_keymap[key] return None - def convert_mouse_key(key: Any) -> Optional[MouseKeyCode]: + def convert_mouse_key(key: Any) -> Optional[MouseButton]: if magnum_enabled and key in magnum_mouse_keymap: return magnum_mouse_keymap[key] return None From f3267569c14d819ef4ae440acbee4d60f00ea1d1 Mon Sep 17 00:00:00 2001 From: Mikael Dallaire Cote <110583667+0mdc@users.noreply.github.com> Date: Tue, 19 Mar 2024 12:29:42 -0400 Subject: [PATCH 3/4] Formatting fix. --- habitat-hitl/habitat_hitl/core/remote_client_state.py | 1 - 1 file changed, 1 deletion(-) diff --git a/habitat-hitl/habitat_hitl/core/remote_client_state.py b/habitat-hitl/habitat_hitl/core/remote_client_state.py index df82d5f1a5..00744a1744 100644 --- a/habitat-hitl/habitat_hitl/core/remote_client_state.py +++ b/habitat-hitl/habitat_hitl/core/remote_client_state.py @@ -191,7 +191,6 @@ def _update_input_state(self, client_states): # frame and other ways that keyHeld, keyDown, and keyUp can be inconsistent. last_client_state = client_states[-1] - input_json = ( last_client_state["input"] if "input" in last_client_state From e284b01fb47465947b847bc1b8a8104dec029ede Mon Sep 17 00:00:00 2001 From: Mikael Dallaire Cote <110583667+0mdc@users.noreply.github.com> Date: Wed, 20 Mar 2024 17:59:35 -0400 Subject: [PATCH 4/4] Review pass. --- habitat-hitl/habitat_hitl/_internal/gui_application.py | 4 ++-- habitat-hitl/habitat_hitl/core/key_mapping.py | 6 +++--- habitat-hitl/habitat_hitl/core/remote_client_state.py | 8 ++++---- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/habitat-hitl/habitat_hitl/_internal/gui_application.py b/habitat-hitl/habitat_hitl/_internal/gui_application.py index fb1a722422..98afc42fab 100644 --- a/habitat-hitl/habitat_hitl/_internal/gui_application.py +++ b/habitat-hitl/habitat_hitl/_internal/gui_application.py @@ -57,7 +57,7 @@ def key_release_event(self, event: Application.KeyEvent) -> None: wrapper._key_up.add(key) def mouse_press_event(self, event: Application.MouseEvent) -> None: - key = MagnumKeyConverter.convert_mouse_key(event.button) + key = MagnumKeyConverter.convert_mouse_button(event.button) if key: for wrapper in self._gui_inputs: # If the key is already held, this is a repeat press event and we should @@ -67,7 +67,7 @@ def mouse_press_event(self, event: Application.MouseEvent) -> None: wrapper._mouse_button_down.add(key) def mouse_release_event(self, event: Application.MouseEvent) -> None: - key = MagnumKeyConverter.convert_mouse_key(event.button) + key = MagnumKeyConverter.convert_mouse_button(event.button) if key: for wrapper in self._gui_inputs: if key in wrapper._mouse_button_held: diff --git a/habitat-hitl/habitat_hitl/core/key_mapping.py b/habitat-hitl/habitat_hitl/core/key_mapping.py index 3047a0531a..e1f8fb3204 100644 --- a/habitat-hitl/habitat_hitl/core/key_mapping.py +++ b/habitat-hitl/habitat_hitl/core/key_mapping.py @@ -151,7 +151,7 @@ def convert_key(key: Any) -> Optional[KeyCode]: return magnum_keymap[key] return None - def convert_mouse_key(key: Any) -> Optional[MouseButton]: - if magnum_enabled and key in magnum_mouse_keymap: - return magnum_mouse_keymap[key] + def convert_mouse_button(button: Any) -> Optional[MouseButton]: + if magnum_enabled and button in magnum_mouse_keymap: + return magnum_mouse_keymap[button] return None diff --git a/habitat-hitl/habitat_hitl/core/remote_client_state.py b/habitat-hitl/habitat_hitl/core/remote_client_state.py index 00744a1744..a84227d0bb 100644 --- a/habitat-hitl/habitat_hitl/core/remote_client_state.py +++ b/habitat-hitl/habitat_hitl/core/remote_client_state.py @@ -160,7 +160,7 @@ def _update_input_state(self, client_states): client_state["mouse"] if "mouse" in client_state else None ) - if input_json != None: + if input_json is not None: for button in input_json["buttonDown"]: if button not in KeyCode: continue @@ -170,7 +170,7 @@ def _update_input_state(self, client_states): continue self._gui_input._key_up.add(KeyCode(button)) - if mouse_json != None: + if mouse_json is not None: mouse_buttons = mouse_json["buttons"] for button in mouse_buttons["buttonDown"]: if button not in KeyCode: @@ -204,13 +204,13 @@ def _update_input_state(self, client_states): self._gui_input._key_held.clear() - if input_json != None: + if input_json is not None: for button in input_json["buttonHeld"]: if button not in KeyCode: continue self._gui_input._key_held.add(KeyCode(button)) - if mouse_json != None: + if mouse_json is not None: mouse_buttons = mouse_json["buttons"] for button in mouse_buttons["buttonHeld"]: if button not in KeyCode: