123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520 |
- #include "quantum.h"
- #include "report.h"
- #include "timer.h"
- #include "process_key_override.h"
- #include <debug.h>
- #ifndef KEY_OVERRIDE_REPEAT_DELAY
- # define KEY_OVERRIDE_REPEAT_DELAY 500
- #endif
- #ifdef DEBUG_KEY_OVERRIDE
- # define key_override_printf dprintf
- #else
- # define key_override_printf(str, ...) \
- {}
- #endif
- extern uint8_t extract_mod_bits(uint16_t code);
- extern void set_weak_override_mods(uint8_t mods);
- extern void clear_weak_override_mods(void);
- extern void set_suppressed_override_mods(uint8_t mods);
- extern void clear_suppressed_override_mods(void);
- static uint16_t clear_mods_from(uint16_t keycode) {
- switch (keycode) {
- case QK_MODS ... QK_MODS_MAX:
- break;
- default:
- return keycode;
- }
- static const uint16_t all_mods = QK_LCTL | QK_LSFT | QK_LALT | QK_LGUI | QK_RCTL | QK_RSFT | QK_RALT | QK_RGUI;
- return (keycode & ~(all_mods));
- }
- static const key_override_t *active_override = NULL;
- static bool active_override_trigger_is_down = false;
- static uint16_t last_key_down = 0;
- static uint32_t last_key_down_time = 0;
- static uint32_t defer_reference_time = 0;
- static uint32_t defer_delay = 0;
- static uint16_t deferred_register = 0;
- static bool enabled = true;
- __attribute__((weak)) const key_override_t **key_overrides = NULL;
- static const key_override_t *clear_active_override(const bool allow_reregister);
- void key_override_on(void) {
- enabled = true;
- key_override_printf("Key override ON\n");
- }
- void key_override_off(void) {
- enabled = false;
- clear_active_override(false);
- key_override_printf("Key override OFF\n");
- }
- void key_override_toggle(void) {
- if (key_override_is_enabled()) {
- key_override_off();
- } else {
- key_override_on();
- }
- }
- bool key_override_is_enabled(void) {
- return enabled;
- }
- static bool key_override_matches_active_modifiers(const key_override_t *override, const uint8_t mods) {
-
- if ((override->negative_mod_mask & mods) != 0) {
- return false;
- }
-
- if (override->trigger_mods == 0) {
- return true;
- }
- if ((override->options & ko_option_one_mod) != 0) {
-
- return (override->trigger_mods & mods) != 0;
- } else {
-
-
- uint8_t one_sided_required_mods = (override->trigger_mods & 0b1111) | (override->trigger_mods >> 4);
-
- uint8_t active_required_mods = override->trigger_mods & mods;
-
- uint8_t one_sided_active_required_mods = (active_required_mods & 0b1111) | (active_required_mods >> 4);
-
- return one_sided_active_required_mods == one_sided_required_mods;
- }
- return false;
- }
- static void schedule_deferred_register(const uint16_t keycode) {
- if (timer_elapsed32(last_key_down_time) < KEY_OVERRIDE_REPEAT_DELAY) {
-
- defer_reference_time = last_key_down_time;
- defer_delay = KEY_OVERRIDE_REPEAT_DELAY;
- } else {
-
- defer_reference_time = timer_read32();
- defer_delay = 50;
- }
- deferred_register = keycode;
- }
- const key_override_t *clear_active_override(const bool allow_reregister) {
- if (active_override == NULL) {
- return NULL;
- }
- key_override_printf("Deactivating override\n");
- deferred_register = 0;
-
- clear_suppressed_override_mods();
-
- clear_weak_override_mods();
- const key_override_t *const old = active_override;
- const uint8_t mod_free_replacement = clear_mods_from(active_override->replacement);
- bool unregister_replacement = mod_free_replacement != KC_NO &&
- mod_free_replacement < SAFE_RANGE;
-
- if (active_override->custom_action != NULL) {
- unregister_replacement &= active_override->custom_action(false, active_override->context);
- }
-
- if (unregister_replacement) {
- if (IS_KEY(mod_free_replacement)) {
- del_key(mod_free_replacement);
- } else {
- key_override_printf("NOT KEY 1\n");
- send_keyboard_report();
- unregister_code(mod_free_replacement);
- }
- }
- const uint16_t trigger = active_override->trigger;
- const bool reregister_trigger = allow_reregister &&
- (active_override->options & ko_option_no_reregister_trigger) == 0 &&
- active_override_trigger_is_down &&
- trigger != KC_NO &&
- trigger < SAFE_RANGE;
-
- if (reregister_trigger) {
- key_override_printf("Re-registering trigger deferred: %u\n", trigger);
-
- schedule_deferred_register(trigger);
- }
- send_keyboard_report();
- active_override = NULL;
- active_override_trigger_is_down = false;
- return old;
- }
- static bool check_activation_event(const key_override_t *override, const bool key_down, const bool is_mod) {
- ko_option_t options = override->options;
- if ((options & ko_options_all_activations) == 0) {
-
- options = ko_options_all_activations;
- }
- if (is_mod) {
- if (key_down) {
- return (options & ko_option_activation_required_mod_down) != 0;
- } else {
- return (options & ko_option_activation_negative_mod_up) != 0;
- }
- } else {
- if (key_down) {
- return (options & ko_option_activation_trigger_down) != 0;
- } else {
- return false;
- }
- }
- }
- static bool try_activating_override(const uint16_t keycode, const uint8_t layer, const bool key_down, const bool is_mod, const uint8_t active_mods, bool *activated) {
- if (key_overrides == NULL) {
- return true;
- }
- for (uint8_t i = 0;; i++) {
- const key_override_t *const override = key_overrides[i];
-
- if (override == NULL) {
- break;
- }
-
- if (active_mods == 0 && override->trigger_mods != 0) {
- key_override_printf("Not activating override: Modifiers don't match\n");
- continue;
- }
-
- if ((override->layers & (1 << layer)) == 0) {
- key_override_printf("Not activating override: Not set to activate on pressed layer\n");
- continue;
- }
-
- if (!check_activation_event(override, key_down, is_mod)) {
- key_override_printf("Not activating override: Activation event not allowed\n");
- continue;
- }
- const bool is_trigger = override->trigger == keycode;
-
- if (is_trigger && !key_down) {
- key_override_printf("Not activating override: Trigger lifted\n");
- continue;
- }
-
- const bool no_trigger = override->trigger == KC_NO;
-
- if (override == active_override) {
- key_override_printf("Not activating override: Alerady actived\n");
- continue;
- }
-
- if (override->enabled != NULL && !((*(override->enabled) & 1))) {
- key_override_printf("Not activating override: Not enabled\n");
- continue;
- }
-
- if (!key_override_matches_active_modifiers(override, active_mods)) {
- key_override_printf("Not activating override: Modifiers don't match\n");
- continue;
- }
-
- const bool trigger_down = is_trigger && key_down;
-
-
-
-
- bool should_activate = no_trigger || trigger_down || last_key_down == override->trigger;
- if (!should_activate) {
- key_override_printf("Not activating override. Trigger not down\n");
- continue;
- }
- key_override_printf("Activating override\n");
- clear_active_override(false);
- active_override = override;
- active_override_trigger_is_down = true;
- set_suppressed_override_mods(override->suppressed_mods);
- if (!trigger_down && !no_trigger) {
-
- if (IS_KEY(override->trigger)) {
- del_key(override->trigger);
- } else {
- unregister_code(override->trigger);
- }
- }
- const uint16_t mod_free_replacement = clear_mods_from(override->replacement);
- bool register_replacement = mod_free_replacement != KC_NO &&
- mod_free_replacement < SAFE_RANGE;
-
- if (override->custom_action != NULL) {
- register_replacement &= override->custom_action(true, override->context);
- }
- if (register_replacement) {
- const uint8_t override_mods = extract_mod_bits(override->replacement);
- set_weak_override_mods(override_mods);
-
- if (is_mod) {
- key_override_printf("Deferring register replacement key\n");
- schedule_deferred_register(mod_free_replacement);
- send_keyboard_report();
- } else {
- if (IS_KEY(mod_free_replacement)) {
- add_key(mod_free_replacement);
- } else {
- key_override_printf("NOT KEY 2\n");
- send_keyboard_report();
-
- wait_ms(10);
- register_code(mod_free_replacement);
- }
- }
- } else {
-
- send_keyboard_report();
- }
- *activated = true;
-
- return !trigger_down;
- }
- *activated = false;
- return true;
- }
- void key_override_task(void) {
- if (deferred_register == 0) {
- return;
- }
- if (timer_elapsed32(defer_reference_time) >= defer_delay) {
- key_override_printf("Registering deferred key\n");
- register_code16(deferred_register);
- deferred_register = 0;
- defer_reference_time = 0;
- defer_delay = 0;
- }
- }
- bool process_key_override(const uint16_t keycode, const keyrecord_t *const record) {
- #ifdef BENCH_KEY_OVERRIDE
- uint16_t start = timer_read();
- #endif
- const bool key_down = record->event.pressed;
- const bool is_mod = IS_MOD(keycode);
- if (key_down) {
- switch (keycode) {
- case KEY_OVERRIDE_TOGGLE:
- key_override_toggle();
- return false;
- case KEY_OVERRIDE_ON:
- key_override_on();
- return false;
- case KEY_OVERRIDE_OFF:
- key_override_off();
- return false;
- default:
- break;
- }
- }
- if (!enabled) {
- return true;
- }
- uint8_t effective_mods = get_mods();
- #ifdef KEY_OVERRIDE_INCLUDE_WEAK_MODS
- effective_mods |= get_weak_mods();
- #endif
- #ifndef NO_ACTION_ONESHOT
-
- effective_mods |= get_oneshot_locked_mods() | get_oneshot_mods();
- #endif
- if (is_mod) {
-
- if (key_down) {
- effective_mods |= MOD_BIT(keycode);
- } else {
- effective_mods &= ~MOD_BIT(keycode);
- }
- } else {
- if (key_down) {
- last_key_down = keycode;
- last_key_down_time = timer_read32();
- deferred_register = 0;
- }
-
- if (!key_down && keycode == last_key_down) {
- last_key_down = 0;
- last_key_down_time = 0;
-
- deferred_register = 0;
- }
- }
- key_override_printf("key down: %u keycode: %u is mod: %u effective mods: %u\n", key_down, keycode, is_mod, effective_mods);
- bool send_key_action = true;
- bool activated = false;
-
- if (is_mod || key_down) {
-
- const uint8_t layer = read_source_layers_cache(record->event.key);
-
- send_key_action = try_activating_override(keycode, layer, key_down, is_mod, effective_mods, &activated);
- if (!send_key_action) {
- send_keyboard_report();
- }
- }
- if (!activated && active_override != NULL) {
- if (is_mod) {
-
- if (!key_override_matches_active_modifiers(active_override, effective_mods)) {
- key_override_printf("Deactivating override because necessary modifier lifted or negative mod pressed\n");
- clear_active_override(true);
- }
- } else {
-
- const bool is_trigger = keycode == active_override->trigger;
- bool should_deactivate = false;
-
- if (is_trigger && !key_down) {
- should_deactivate = true;
- active_override_trigger_is_down = false;
- key_override_printf("Deactivating override because trigger key up\n");
- }
-
- if (key_down && (active_override->options & ko_option_no_unregister_on_other_key_down) == 0) {
- should_deactivate = true;
- key_override_printf("Deactivating override because another key was pressed\n");
- }
- if (should_deactivate) {
- clear_active_override(false);
- }
- }
- }
- #ifdef BENCH_KEY_OVERRIDE
- uint16_t elapsed = timer_elapsed(start);
- dprintf("Processing key overrides took: %u ms\n", elapsed);
- #endif
- return send_key_action;
- }
|