Skip to content

Commit

Permalink
Listen to changes in the keyboard mapping on X
Browse files Browse the repository at this point in the history
  • Loading branch information
user202729 committed Apr 6, 2021
1 parent 42fd6c0 commit d1f7624
Showing 1 changed file with 31 additions and 2 deletions.
33 changes: 31 additions & 2 deletions plover/oslayer/xkeyboardcontrol.py
Original file line number Diff line number Diff line change
Expand Up @@ -1135,7 +1135,7 @@ def keysym_to_string(keysym):
return chr(code)


class KeyboardEmulation:
class KeyboardEmulation(XEventLoop):
"""Emulate keyboard events."""

class Mapping:
Expand Down Expand Up @@ -1164,9 +1164,31 @@ def __str__(self):

def __init__(self):
"""Prepare to emulate keyboard events."""
self._display = display.Display()
super().__init__(name='emulation')
self._update_keymap()
self._update_modifiers()
self._self_change = []
# List of keycodes of keyboard mapping changes created by this KeyboardEmulation object
super().start()

def _on_event(self, event):
if event.type == X.MappingNotify:
if event.request == X.MappingKeyboard:
if event.count == 1:
try:
self._self_change.remove(event.first_keycode)
# the operation above is atomic and will not cause a race condition
# with the append to `self._self_change` below
# see http://effbot.org/pyfaq/what-kinds-of-global-value-mutation-are-thread-safe.htm
return
except ValueError:
pass
self._update_keymap()

elif event.request == X.MappingModifier:
self._update_modifiers()

@with_display_lock
def _update_keymap(self):
'''Analyse keymap, build a mapping of keysym to (keycode + modifiers),
and find unused keycodes that can be used for unmapped keysyms.
Expand Down Expand Up @@ -1222,9 +1244,13 @@ def _update_keymap(self):
self._backspace_mapping = self._get_mapping(backspace_keysym)
assert self._backspace_mapping is not None
assert self._backspace_mapping.custom_mapping is None

@with_display_lock
def _update_modifiers(self):
# Get modifier mapping.
self.modifier_mapping = self._display.get_modifier_mapping()

@with_display_lock
def send_backspaces(self, number_of_backspaces):
"""Emulate the given number of backspaces.
Expand All @@ -1240,6 +1266,7 @@ def send_backspaces(self, number_of_backspaces):
self._backspace_mapping.modifiers)
self._display.sync()

@with_display_lock
def send_string(self, s):
"""Emulate the given string.
Expand All @@ -1259,6 +1286,7 @@ def send_string(self, s):
mapping.modifiers)
self._display.sync()

@with_display_lock
def send_key_combination(self, combo_string):
"""Emulate a sequence of key combinations.
Expand Down Expand Up @@ -1359,6 +1387,7 @@ def _get_mapping(self, keysym, automatically_map=True):
keysym_index = mapping.custom_mapping.index(previous_keysym)
# Update X11 keymap.
mapping.custom_mapping[keysym_index] = keysym
self._self_change.append(mapping.keycode)
self._display.change_keyboard_mapping(mapping.keycode, [mapping.custom_mapping])
# Update our keymap.
if previous_keysym in self._keymap:
Expand Down

0 comments on commit d1f7624

Please sign in to comment.