from kmk.consts import LeaderMode from kmk.keys import KC from kmk.kmktime import ticks_ms from kmk.types import TapDanceKeyMeta from kmk.util import intify_coordinate class InternalState: keys_pressed = set() coord_keys_pressed = {} leader_pending = None leader_last_len = 0 hid_pending = False leader_mode_history = [] active_layers = [0] reversed_active_layers = list(reversed(active_layers)) start_time = { 'lt': None, 'tg': None, 'tt': None, 'lm': None, 'leader': None, } timeouts = {} tapping = False tap_dance_counts = {} tap_side_effects = {} def __init__(self, config): self.config = config def __repr__(self): return 'InternalState({})'.format(self._to_dict()) def _to_dict(self): ret = { 'keys_pressed': self.keys_pressed, 'active_layers': self.active_layers, 'leader_mode_history': self.leader_mode_history, 'leader_mode': self.config.leader_mode, 'start_time': self.start_time, 'tapping': self.tapping, 'tap_dance_counts': self.tap_dance_counts, 'timeouts': self.timeouts, } return ret def _find_key_in_map(self, row, col): # Later-added layers have priority. Sift through the layers # in reverse order until we find a valid keycode object for layer in self.reversed_active_layers: layer_key = self.config.keymap[layer][row][col] if not layer_key or layer_key == KC.TRNS: continue if self.config.debug_enabled: print('Resolved key: {}'.format(layer_key)) return layer_key def set_timeout(self, after_ticks, callback): if after_ticks is False: # We allow passing False as an implicit "run this on the next process timeouts cycle" timeout_key = ticks_ms() else: timeout_key = ticks_ms() + after_ticks self.timeouts[timeout_key] = callback return self def process_timeouts(self): if not self.timeouts: return self current_time = ticks_ms() # cast this to a tuple to ensure that if a callback itself sets # timeouts, we do not handle them on the current cycle timeouts = tuple(self.timeouts.items()) for k, v in timeouts: if k <= current_time: v() del self.timeouts[k] return self def matrix_changed(self, row, col, is_pressed): if self.config.debug_enabled: print('Matrix changed (col, row, pressed?): {}, {}, {}'.format( col, row, is_pressed, )) int_coord = intify_coordinate(row, col) kc_changed = self._find_key_in_map(row, col) if kc_changed is None: print('No key accessible for col, row: {}, {}'.format(row, col)) return self return self.process_key(kc_changed, is_pressed, int_coord, (row, col)) def process_key(self, key, is_pressed, coord_int=None, coord_raw=None): if self.tapping and not isinstance(key.meta, TapDanceKeyMeta): self._process_tap_dance(key, is_pressed) else: if is_pressed: key.on_press(self, coord_int, coord_raw) else: key.on_release(self, coord_int, coord_raw) if self.config.leader_mode % 2 == 1: self._process_leader_mode() return self def remove_key(self, keycode): self.keys_pressed.discard(keycode) return self.process_key(keycode, False) def add_key(self, keycode): self.keys_pressed.add(keycode) return self.process_key(keycode, True) def tap_key(self, keycode): self.add_key(keycode) # On the next cycle, we'll remove the key. This is way more clean than # the `pending_keys` implementation that we used to rely on in # firmware.py self.set_timeout(False, lambda: self.remove_key(keycode)) return self def resolve_hid(self): self.hid_pending = False return self def _process_internal_key_event(self, changed_key, is_pressed): # Since the key objects can be chained into new objects # with, for example, no_press set, always check against # the underlying code rather than comparing Key # objects return self.internal_key_handlers[changed_key.code]( changed_key, is_pressed, ) def _process_tap_dance(self, changed_key, is_pressed): if is_pressed: if not isinstance(changed_key.meta, TapDanceKeyMeta): # If we get here, changed_key is not a TapDanceKey and thus # the user kept typing elsewhere (presumably). End ALL of the # currently outstanding tap dance runs. for k, v in self.tap_dance_counts.items(): if v: self._end_tap_dance(k) return self if ( changed_key not in self.tap_dance_counts or not self.tap_dance_counts[changed_key] ): self.tap_dance_counts[changed_key] = 1 self.set_timeout(self.config.tap_time, lambda: self._end_tap_dance(changed_key)) self.tapping = True else: self.tap_dance_counts[changed_key] += 1 if changed_key not in self.tap_side_effects: self.tap_side_effects[changed_key] = None else: if ( self.tap_side_effects[changed_key] is not None or self.tap_dance_counts[changed_key] == len(changed_key.codes) ): self._end_tap_dance(changed_key) return self def _end_tap_dance(self, td_key): v = self.tap_dance_counts[td_key] - 1 if v >= 0: if td_key in self.keys_pressed: key_to_press = td_key.codes[v] self.add_key(key_to_press) self.tap_side_effects[td_key] = key_to_press self.hid_pending = True else: if self.tap_side_effects[td_key]: self.remove_key(self.tap_side_effects[td_key]) self.tap_side_effects[td_key] = None self.hid_pending = True self._cleanup_tap_dance(td_key) else: self.tap_key(td_key.codes[v]) self._cleanup_tap_dance(td_key) return self def _cleanup_tap_dance(self, td_key): self.tap_dance_counts[td_key] = 0 self.tapping = any(count > 0 for count in self.tap_dance_counts.values()) return self def _begin_leader_mode(self): if self.config.leader_mode % 2 == 0: self.keys_pressed.discard(KC.LEAD) # All leader modes are one number higher when activating self.config.leader_mode += 1 if self.config.leader_mode == LeaderMode.TIMEOUT_ACTIVE: self.set_timeout(self.config.leader_timeout, self._handle_leader_sequence) return self def _handle_leader_sequence(self): lmh = tuple(self.leader_mode_history) if lmh in self.config.leader_dictionary: self.process_key(self.config.leader_dictionary[lmh], True) return self._exit_leader_mode() def _process_leader_mode(self): keys_pressed = self.keys_pressed if self.leader_last_len and self.leader_mode_history: history_set = set(self.leader_mode_history) keys_pressed = keys_pressed - history_set self.leader_last_len = len(self.keys_pressed) for key in keys_pressed: if ( self.config.leader_mode == LeaderMode.ENTER_ACTIVE and key == KC.ENT ): self._handle_leader_sequence() break elif key == KC.ESC or key == KC.GESC: # Clean self and turn leader mode off. self._exit_leader_mode() break elif key == KC.LEAD: break else: # Add key if not needing to escape # This needs replaced later with a proper debounce self.leader_mode_history.append(key) self.hid_pending = False return self def _exit_leader_mode(self): self.leader_mode_history.clear() self.config.leader_mode -= 1 self.leader_last_len = 0 self.keys_pressed.clear() return self