5efd2688d7
Faulty logic corrected Possibly a bug could happen whenever entering ht_press with a non-empty key_state outside of repeat context (but can it really happen?) Some formatting to make github tests happy Documentation for repeat behavior. Make the tests happy again! same...
241 lines
8.1 KiB
Python
241 lines
8.1 KiB
Python
from micropython import const
|
|
|
|
from kmk.keys import make_argumented_key
|
|
from kmk.modules import Module
|
|
from kmk.utils import Debug
|
|
|
|
debug = Debug(__name__)
|
|
|
|
|
|
class ActivationType:
|
|
PRESSED = const(0)
|
|
RELEASED = const(1)
|
|
HOLD_TIMEOUT = const(2)
|
|
INTERRUPTED = const(3)
|
|
REPEAT = const(4)
|
|
|
|
|
|
class HoldTapKeyState:
|
|
def __init__(self, timeout_key, *args, **kwargs):
|
|
self.timeout_key = timeout_key
|
|
self.args = args
|
|
self.kwargs = kwargs
|
|
self.activated = ActivationType.PRESSED
|
|
|
|
|
|
class HoldTapKeyMeta:
|
|
def __init__(
|
|
self,
|
|
tap,
|
|
hold,
|
|
prefer_hold=True,
|
|
tap_interrupted=False,
|
|
tap_time=None,
|
|
repeat=False,
|
|
):
|
|
self.tap = tap
|
|
self.hold = hold
|
|
self.prefer_hold = prefer_hold
|
|
self.tap_interrupted = tap_interrupted
|
|
self.tap_time = tap_time
|
|
self.repeat = repeat
|
|
|
|
|
|
class HoldTap(Module):
|
|
tap_time = 300
|
|
|
|
def __init__(self):
|
|
self.key_buffer = []
|
|
self.key_states = {}
|
|
make_argumented_key(
|
|
validator=HoldTapKeyMeta,
|
|
names=('HT',),
|
|
on_press=self.ht_pressed,
|
|
on_release=self.ht_released,
|
|
)
|
|
|
|
def during_bootup(self, keyboard):
|
|
return
|
|
|
|
def before_matrix_scan(self, keyboard):
|
|
return
|
|
|
|
def after_matrix_scan(self, keyboard):
|
|
return
|
|
|
|
def process_key(self, keyboard, key, is_pressed, int_coord):
|
|
'''Handle holdtap being interrupted by another key press/release.'''
|
|
current_key = key
|
|
send_buffer = False
|
|
append_buffer = False
|
|
|
|
for key, state in self.key_states.items():
|
|
if key == current_key:
|
|
continue
|
|
if state.activated != ActivationType.PRESSED:
|
|
continue
|
|
|
|
# holdtap is interrupted by another key event.
|
|
if (is_pressed and not key.meta.tap_interrupted) or (
|
|
not is_pressed and key.meta.tap_interrupted and self.key_buffer
|
|
):
|
|
|
|
keyboard.cancel_timeout(state.timeout_key)
|
|
self.key_states[key].activated = ActivationType.INTERRUPTED
|
|
self.ht_activate_on_interrupt(
|
|
key, keyboard, *state.args, **state.kwargs
|
|
)
|
|
keyboard._send_hid()
|
|
send_buffer = True
|
|
|
|
if state.activated == ActivationType.INTERRUPTED:
|
|
current_key = keyboard._find_key_in_map(int_coord)
|
|
|
|
# if interrupt on release: store interrupting keys until one of them
|
|
# is released.
|
|
if (
|
|
key.meta.tap_interrupted
|
|
and is_pressed
|
|
and not isinstance(current_key.meta, HoldTapKeyMeta)
|
|
):
|
|
append_buffer = True
|
|
|
|
# apply changes with 'side-effects' on key_states or the loop behaviour
|
|
# outside the loop.
|
|
if append_buffer:
|
|
self.key_buffer.append((int_coord, current_key))
|
|
current_key = None
|
|
elif send_buffer:
|
|
self.send_key_buffer(keyboard)
|
|
|
|
return current_key
|
|
|
|
def before_hid_send(self, keyboard):
|
|
return
|
|
|
|
def after_hid_send(self, keyboard):
|
|
return
|
|
|
|
def on_powersave_enable(self, keyboard):
|
|
return
|
|
|
|
def on_powersave_disable(self, keyboard):
|
|
return
|
|
|
|
def ht_pressed(self, key, keyboard, *args, **kwargs):
|
|
'''Unless in repeat mode, do nothing yet, action resolves when key is released, timer expires or other key is pressed.'''
|
|
state = self.key_states.get(key)
|
|
if state is not None and state.activated == ActivationType.RELEASED:
|
|
state.activated = ActivationType.REPEAT
|
|
self.ht_activate_tap(key, keyboard, *args, **kwargs)
|
|
else:
|
|
if key.meta.tap_time is None:
|
|
tap_time = self.tap_time
|
|
else:
|
|
tap_time = key.meta.tap_time
|
|
timeout_key = keyboard.set_timeout(
|
|
tap_time,
|
|
lambda: self.on_tap_time_expired(key, keyboard, *args, **kwargs),
|
|
)
|
|
self.key_states[key] = HoldTapKeyState(timeout_key, *args, **kwargs)
|
|
return keyboard
|
|
|
|
def ht_released(self, key, keyboard, *args, **kwargs):
|
|
'''On keyup, release mod or tap key.'''
|
|
if key not in self.key_states:
|
|
return keyboard
|
|
|
|
state = self.key_states[key]
|
|
keyboard.cancel_timeout(state.timeout_key)
|
|
|
|
if state.activated == ActivationType.HOLD_TIMEOUT:
|
|
# release hold
|
|
self.ht_deactivate_hold(key, keyboard, *args, **kwargs)
|
|
elif state.activated == ActivationType.INTERRUPTED:
|
|
# release tap
|
|
self.ht_deactivate_on_interrupt(key, keyboard, *args, **kwargs)
|
|
elif state.activated == ActivationType.PRESSED:
|
|
# press and release tap because key released within tap time
|
|
self.ht_activate_tap(key, keyboard, *args, **kwargs)
|
|
self.ht_deactivate_tap(key, keyboard, *args, **kwargs)
|
|
state.activated = ActivationType.RELEASED
|
|
self.send_key_buffer(keyboard)
|
|
# don't delete the key state right now in this case
|
|
if key.meta.repeat:
|
|
return keyboard
|
|
elif state.activated == ActivationType.REPEAT:
|
|
self.ht_deactivate_tap(key, keyboard, *args, **kwargs)
|
|
del self.key_states[key]
|
|
|
|
return keyboard
|
|
|
|
def on_tap_time_expired(self, key, keyboard, *args, **kwargs):
|
|
'''When tap time expires activate hold if key is still being pressed.
|
|
Remove key if ActivationType is RELEASED.'''
|
|
try:
|
|
state = self.key_states[key]
|
|
except KeyError:
|
|
if debug.enabled:
|
|
debug(f'on_tap_time_expired: no such key {key}')
|
|
return
|
|
|
|
if self.key_states[key].activated == ActivationType.PRESSED:
|
|
# press hold because timer expired after tap time
|
|
self.key_states[key].activated = ActivationType.HOLD_TIMEOUT
|
|
self.ht_activate_hold(key, keyboard, *args, **kwargs)
|
|
self.send_key_buffer(keyboard)
|
|
elif state.activated == ActivationType.RELEASED:
|
|
self.ht_deactivate_tap(key, keyboard, *args, **kwargs)
|
|
del self.key_states[key]
|
|
|
|
def send_key_buffer(self, keyboard):
|
|
for (int_coord, key) in self.key_buffer:
|
|
new_key = keyboard._find_key_in_map(int_coord)
|
|
keyboard.resume_process_key(self, new_key, True, int_coord)
|
|
|
|
keyboard._send_hid()
|
|
self.key_buffer.clear()
|
|
|
|
def ht_activate_hold(self, key, keyboard, *args, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_activate_hold')
|
|
keyboard.resume_process_key(self, key.meta.hold, True)
|
|
|
|
def ht_deactivate_hold(self, key, keyboard, *args, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_deactivate_hold')
|
|
keyboard.set_timeout(
|
|
False, lambda: keyboard.resume_process_key(self, key.meta.hold, False)
|
|
)
|
|
|
|
def ht_activate_tap(self, key, keyboard, *args, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_activate_tap')
|
|
keyboard.resume_process_key(self, key.meta.tap, True)
|
|
|
|
def ht_deactivate_tap(self, key, keyboard, *args, delayed=True, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_deactivate_tap')
|
|
if delayed:
|
|
keyboard.set_timeout(
|
|
False, lambda: keyboard.resume_process_key(self, key.meta.tap, False)
|
|
)
|
|
else:
|
|
keyboard.resume_process_key(self, key.meta.tap, False)
|
|
|
|
def ht_activate_on_interrupt(self, key, keyboard, *args, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_activate_on_interrupt')
|
|
if key.meta.prefer_hold:
|
|
self.ht_activate_hold(key, keyboard, *args, **kwargs)
|
|
else:
|
|
self.ht_activate_tap(key, keyboard, *args, **kwargs)
|
|
|
|
def ht_deactivate_on_interrupt(self, key, keyboard, *args, **kwargs):
|
|
if debug.enabled:
|
|
debug('ht_deactivate_on_interrupt')
|
|
if key.meta.prefer_hold:
|
|
self.ht_deactivate_hold(key, keyboard, *args, **kwargs)
|
|
else:
|
|
self.ht_deactivate_tap(key, keyboard, *args, delayed=False, **kwargs)
|