25a86df5c1
* Changed from multiple calls of `keyboard.remove_key` to a for loop when releasing modifiers on match
224 lines
8.1 KiB
Python
224 lines
8.1 KiB
Python
try:
|
|
from typing import Optional
|
|
except ImportError:
|
|
# we're not in a dev environment, so we don't need to worry about typing
|
|
pass
|
|
from micropython import const
|
|
|
|
from kmk.keys import KC, Key, ModifierKey
|
|
from kmk.modules import Module
|
|
|
|
|
|
class State:
|
|
LISTENING = const(0)
|
|
DELETING = const(1)
|
|
SENDING = const(2)
|
|
IGNORING = const(3)
|
|
|
|
|
|
class Character:
|
|
'''Helper class for making a left-shifted key identical to a right-shifted key'''
|
|
|
|
is_shifted: bool = False
|
|
|
|
def __init__(self, key_code: Key, is_shifted: bool) -> None:
|
|
self.is_shifted = is_shifted
|
|
self.key_code = KC.LSHIFT(key_code) if is_shifted else key_code
|
|
|
|
def __eq__(self, other: any) -> bool: # type: ignore
|
|
try:
|
|
return (
|
|
self.key_code.code == other.key_code.code
|
|
and self.is_shifted == other.is_shifted
|
|
)
|
|
except AttributeError:
|
|
return False
|
|
|
|
|
|
class Phrase:
|
|
'''Manages a collection of characters and keeps an index of them so that potential matches can be tracked'''
|
|
|
|
def __init__(self, string: str) -> None:
|
|
self._characters: list[Character] = []
|
|
self._index: int = 0
|
|
for char in string:
|
|
try:
|
|
key_code = KC[char]
|
|
shifted = char.isupper() or key_code.has_modifiers == {2}
|
|
self._characters.append(Character(key_code, shifted))
|
|
except ValueError:
|
|
raise ValueError(f'Invalid character in dictionary: {char}')
|
|
|
|
def next_character(self) -> None:
|
|
'''Increment the current index for this phrase'''
|
|
if not self.index_at_end():
|
|
self._index += 1
|
|
|
|
def get_character_at_index(self, index: int) -> Character:
|
|
'''Returns the character at the given index'''
|
|
return self._characters[index]
|
|
|
|
def get_character_at_current_index(self) -> Character:
|
|
'''Returns the character at the current index for this phrase'''
|
|
return self._characters[self._index]
|
|
|
|
def reset_index(self) -> None:
|
|
'''Reset the index to the start of the phrase'''
|
|
self._index = 0
|
|
|
|
def index_at_end(self) -> bool:
|
|
'''Returns True if the index is at the end of the phrase'''
|
|
return self._index == len(self._characters)
|
|
|
|
def character_is_at_current_index(self, character) -> bool:
|
|
'''Returns True if the given character is the next character in the phrase'''
|
|
return self.get_character_at_current_index() == character
|
|
|
|
|
|
class Rule:
|
|
'''Represents the relationship between a phrase to be substituted and its substitution'''
|
|
|
|
def __init__(self, to_substitute: Phrase, substitution: Phrase) -> None:
|
|
self.to_substitute: Phrase = to_substitute
|
|
self.substitution: Phrase = substitution
|
|
|
|
def restart(self) -> None:
|
|
'''Resets this rule's to_substitute and substitution phrases'''
|
|
self.to_substitute.reset_index()
|
|
self.substitution.reset_index()
|
|
|
|
|
|
class StringSubstitution(Module):
|
|
_shifted: bool = False
|
|
_rules: list = []
|
|
_state: State = State.LISTENING
|
|
_matched_rule: Optional[Phrase] = None
|
|
_active_modifiers: list[ModifierKey] = []
|
|
|
|
def __init__(
|
|
self,
|
|
dictionary: dict,
|
|
):
|
|
for key, value in dictionary.items():
|
|
self._rules.append(Rule(Phrase(key), Phrase(value)))
|
|
|
|
def process_key(self, keyboard, key, is_pressed, int_coord):
|
|
if key is KC.LSFT or key is KC.RSFT:
|
|
if is_pressed:
|
|
self._shifted = True
|
|
else:
|
|
self._shifted = False
|
|
|
|
# control ignoring state if the key is a non-shift modifier
|
|
elif type(key) is ModifierKey:
|
|
if is_pressed and key not in self._active_modifiers:
|
|
self._active_modifiers.append(key)
|
|
self._state = State.IGNORING
|
|
elif key in self._active_modifiers:
|
|
self._active_modifiers.remove(key)
|
|
if not self._active_modifiers:
|
|
self._state = State.LISTENING
|
|
# reset rules because pressing a modifier combination
|
|
# should interrupt any current matches
|
|
for rule in self._rules:
|
|
rule.restart()
|
|
|
|
if not self._state == State.LISTENING:
|
|
return key
|
|
|
|
if is_pressed:
|
|
character = Character(key, self._shifted)
|
|
|
|
# run through the dictionary to check for a possible match on each new keypress
|
|
for rule in self._rules:
|
|
if rule.to_substitute.character_is_at_current_index(character):
|
|
rule.to_substitute.next_character()
|
|
else:
|
|
rule.restart()
|
|
# if character is not a match at the current index,
|
|
# it could still be a match at the start of the sequence
|
|
# so redo the check after resetting the sequence
|
|
if rule.to_substitute.character_is_at_current_index(character):
|
|
rule.to_substitute.next_character()
|
|
# we've matched all of the characters in a phrase to be substituted
|
|
if rule.to_substitute.index_at_end():
|
|
rule.restart()
|
|
# set the phrase indexes to where they differ
|
|
# so that only the characters that differ are replaced
|
|
for character in rule.to_substitute._characters:
|
|
if (
|
|
character
|
|
== rule.substitution.get_character_at_current_index()
|
|
):
|
|
rule.to_substitute.next_character()
|
|
rule.substitution.next_character()
|
|
else:
|
|
break
|
|
if rule.to_substitute.index_at_end():
|
|
break
|
|
self._matched_rule = rule
|
|
self._state = State.DELETING
|
|
# if we have a match there's no reason to continue the full key processing, so return out
|
|
return
|
|
return key
|
|
|
|
def during_bootup(self, keyboard):
|
|
return
|
|
|
|
def before_matrix_scan(self, keyboard):
|
|
return
|
|
|
|
def before_hid_send(self, keyboard):
|
|
|
|
if self._state == State.LISTENING:
|
|
return
|
|
|
|
if self._state == State.DELETING:
|
|
# force-release modifiers so sending the replacement text doesn't interact with them
|
|
# it should not be possible for any modifiers other than shift to be held upon rule activation
|
|
# as a modified key won't send a keycode that is matched against the user's dictionary,
|
|
# but, just in case, we'll release those too
|
|
modifiers_to_release = [
|
|
KC.LSFT,
|
|
KC.RSFT,
|
|
KC.LCTL,
|
|
KC.LGUI,
|
|
KC.LALT,
|
|
KC.RCTL,
|
|
KC.RGUI,
|
|
KC.RALT,
|
|
]
|
|
for modifier in modifiers_to_release:
|
|
keyboard.remove_key(modifier)
|
|
|
|
# send backspace taps equivalent to the length of the phrase to be substituted
|
|
to_substitute: Phrase = self._matched_rule.to_substitute # type: ignore
|
|
to_substitute.next_character()
|
|
if not to_substitute.index_at_end():
|
|
keyboard.tap_key(KC.BSPC)
|
|
else:
|
|
self._state = State.SENDING
|
|
|
|
if self._state == State.SENDING:
|
|
substitution = self._matched_rule.substitution # type: ignore
|
|
if not substitution.index_at_end():
|
|
keyboard.tap_key(substitution.get_character_at_current_index().key_code)
|
|
substitution.next_character()
|
|
else:
|
|
self._state = State.LISTENING
|
|
self._matched_rule = None
|
|
for rule in self._rules:
|
|
rule.restart()
|
|
|
|
def after_hid_send(self, keyboard):
|
|
return
|
|
|
|
def on_powersave_enable(self, keyboard):
|
|
return
|
|
|
|
def on_powersave_disable(self, keyboard):
|
|
return
|
|
|
|
def after_matrix_scan(self, keyboard):
|
|
return
|