2017-03-29 00:20:36 +02:00
|
|
|
/* Copyright 2016 Jack Humbert
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2016-12-09 23:49:11 +01:00
|
|
|
#include "print.h"
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
#include "process_combo.h"
|
2021-07-05 01:04:46 +02:00
|
|
|
#include "qmk_settings.h"
|
2016-12-09 23:49:11 +01:00
|
|
|
|
2021-07-04 16:11:04 +02:00
|
|
|
#ifdef VIAL_COMBO_ENABLE
|
|
|
|
#include "dynamic_keymap.h"
|
|
|
|
#endif
|
|
|
|
|
2020-03-22 14:17:26 +01:00
|
|
|
#ifndef COMBO_VARIABLE_LEN
|
|
|
|
__attribute__((weak)) combo_t key_combos[COMBO_COUNT] = {};
|
|
|
|
#else
|
2020-03-25 04:39:53 +01:00
|
|
|
extern combo_t key_combos[];
|
|
|
|
extern int COMBO_LEN;
|
2020-03-22 14:17:26 +01:00
|
|
|
#endif
|
2016-12-10 15:11:59 +01:00
|
|
|
|
2020-07-16 14:39:01 +02:00
|
|
|
__attribute__((weak)) void process_combo_event(uint16_t combo_index, bool pressed) {}
|
2016-12-16 20:50:28 +01:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
static uint16_t timer = 0;
|
2020-08-30 00:57:48 +02:00
|
|
|
static uint16_t current_combo_index = 0;
|
2019-08-30 20:19:03 +02:00
|
|
|
static bool drop_buffer = false;
|
|
|
|
static bool b_combo_enable = true; // defaults to enabled
|
2016-12-16 20:50:28 +01:00
|
|
|
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
static uint8_t buffer_size = 0;
|
2016-12-16 20:50:28 +01:00
|
|
|
#ifdef COMBO_ALLOW_ACTION_KEYS
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
static keyrecord_t key_buffer[MAX_COMBO_LENGTH];
|
2016-12-16 20:50:28 +01:00
|
|
|
#else
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
static uint16_t key_buffer[MAX_COMBO_LENGTH];
|
2016-12-10 15:11:59 +01:00
|
|
|
#endif
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
|
|
|
|
static inline void send_combo(uint16_t action, bool pressed) {
|
2019-08-30 20:19:03 +02:00
|
|
|
if (action) {
|
|
|
|
if (pressed) {
|
|
|
|
register_code16(action);
|
|
|
|
} else {
|
|
|
|
unregister_code16(action);
|
|
|
|
}
|
2016-12-10 15:11:59 +01:00
|
|
|
} else {
|
2019-08-30 20:19:03 +02:00
|
|
|
process_combo_event(current_combo_index, pressed);
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
}
|
|
|
|
}
|
2016-12-09 23:49:11 +01:00
|
|
|
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
static inline void dump_key_buffer(bool emit) {
|
2019-08-30 20:19:03 +02:00
|
|
|
if (buffer_size == 0) {
|
|
|
|
return;
|
|
|
|
}
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
if (emit) {
|
|
|
|
for (uint8_t i = 0; i < buffer_size; i++) {
|
2016-12-16 20:50:28 +01:00
|
|
|
#ifdef COMBO_ALLOW_ACTION_KEYS
|
2019-08-30 20:19:03 +02:00
|
|
|
const action_t action = store_or_get_action(key_buffer[i].event.pressed, key_buffer[i].event.key);
|
|
|
|
process_action(&(key_buffer[i]), action);
|
2016-12-16 20:50:28 +01:00
|
|
|
#else
|
2019-08-30 20:19:03 +02:00
|
|
|
register_code16(key_buffer[i]);
|
|
|
|
send_keyboard_report();
|
2016-12-10 15:11:59 +01:00
|
|
|
#endif
|
2019-08-30 20:19:03 +02:00
|
|
|
}
|
2016-12-09 23:49:11 +01:00
|
|
|
}
|
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
buffer_size = 0;
|
2016-12-09 23:49:11 +01:00
|
|
|
}
|
|
|
|
|
2021-07-04 21:07:31 +02:00
|
|
|
static void end_incomplete_combos(void) {
|
|
|
|
#ifndef COMBO_VARIABLE_LEN
|
|
|
|
for (current_combo_index = 0; current_combo_index < COMBO_COUNT; ++current_combo_index) {
|
|
|
|
#else
|
|
|
|
for (current_combo_index = 0; current_combo_index < COMBO_LEN; ++current_combo_index) {
|
|
|
|
#endif
|
|
|
|
combo_t *combo = &key_combos[current_combo_index];
|
|
|
|
if (!(combo->state & COMBO_COMPLETE))
|
|
|
|
combo->state = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define ALL_COMBO_KEYS_ARE_DOWN (((1 << count) - 1) == (combo->state & ~COMBO_COMPLETE))
|
2019-08-30 20:19:03 +02:00
|
|
|
#define KEY_STATE_DOWN(key) \
|
|
|
|
do { \
|
|
|
|
combo->state |= (1 << key); \
|
|
|
|
} while (0)
|
|
|
|
#define KEY_STATE_UP(key) \
|
|
|
|
do { \
|
|
|
|
combo->state &= ~(1 << key); \
|
|
|
|
} while (0)
|
2021-07-04 21:07:31 +02:00
|
|
|
#define KEY_STATE(key) (combo->state & (1 << (key)))
|
2019-08-30 20:19:03 +02:00
|
|
|
|
|
|
|
static bool process_single_combo(combo_t *combo, uint16_t keycode, keyrecord_t *record) {
|
2020-08-30 00:57:48 +02:00
|
|
|
uint8_t count = 0;
|
2020-07-16 14:39:01 +02:00
|
|
|
uint16_t index = -1;
|
2021-07-04 16:11:04 +02:00
|
|
|
#ifdef VIAL_COMBO_ENABLE
|
|
|
|
uint8_t combo_idx = (uintptr_t)combo->keys;
|
|
|
|
vial_combo_entry_t entry;
|
|
|
|
if (dynamic_keymap_get_combo(combo_idx, &entry) != 0)
|
|
|
|
return false;
|
|
|
|
for (count = 0; count < sizeof(entry.input)/sizeof(*entry.input); ++count) {
|
|
|
|
uint16_t key = entry.input[count];
|
|
|
|
if (key == KC_NO) break;
|
|
|
|
if (key == keycode) index = count;
|
|
|
|
}
|
|
|
|
/* must have at least 2 keys in the combo */
|
|
|
|
if (count < 2)
|
|
|
|
return false;
|
|
|
|
#else
|
2019-08-30 20:19:03 +02:00
|
|
|
/* Find index of keycode and number of combo keys */
|
|
|
|
for (const uint16_t *keys = combo->keys;; ++count) {
|
|
|
|
uint16_t key = pgm_read_word(&keys[count]);
|
|
|
|
if (keycode == key) index = count;
|
|
|
|
if (COMBO_END == key) break;
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
}
|
2021-07-04 16:11:04 +02:00
|
|
|
#endif
|
2019-08-30 20:19:03 +02:00
|
|
|
|
|
|
|
/* Continue processing if not a combo key */
|
|
|
|
if (-1 == (int8_t)index) return false;
|
|
|
|
|
2021-07-04 21:07:31 +02:00
|
|
|
bool is_combo_active = true;
|
2019-08-30 20:19:03 +02:00
|
|
|
|
|
|
|
if (record->event.pressed) {
|
|
|
|
KEY_STATE_DOWN(index);
|
|
|
|
|
|
|
|
if (is_combo_active) {
|
|
|
|
if (ALL_COMBO_KEYS_ARE_DOWN) { /* Combo was pressed */
|
2021-07-04 21:07:31 +02:00
|
|
|
dump_key_buffer(false);
|
|
|
|
combo->state |= COMBO_COMPLETE;
|
2019-08-30 20:19:03 +02:00
|
|
|
send_combo(combo->keycode, true);
|
|
|
|
drop_buffer = true;
|
|
|
|
}
|
|
|
|
}
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
} else {
|
2019-08-30 20:19:03 +02:00
|
|
|
if (ALL_COMBO_KEYS_ARE_DOWN) { /* Combo was released */
|
|
|
|
send_combo(combo->keycode, false);
|
|
|
|
}
|
|
|
|
|
2021-07-04 21:07:31 +02:00
|
|
|
/* don't consume the key unless it is a part of a completed combo */
|
|
|
|
is_combo_active = false;
|
|
|
|
if (KEY_STATE(index)) {
|
|
|
|
KEY_STATE_UP(index);
|
|
|
|
|
|
|
|
if (combo->state & COMBO_COMPLETE) {
|
|
|
|
/* Consume the key on release, only if the combo was complete */
|
|
|
|
is_combo_active = true;
|
|
|
|
|
|
|
|
/* if all keys are up, reset the combo */
|
|
|
|
if (combo->state == COMBO_COMPLETE)
|
|
|
|
combo->state = 0;
|
|
|
|
}
|
|
|
|
}
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
}
|
2016-12-09 23:49:11 +01:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
return is_combo_active;
|
2016-12-10 15:11:59 +01:00
|
|
|
}
|
|
|
|
|
2021-07-04 21:07:31 +02:00
|
|
|
#define NO_COMBO_KEYS_ARE_DOWN (0 == (combo->state & ~COMBO_COMPLETE))
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
|
|
|
|
bool process_combo(uint16_t keycode, keyrecord_t *record) {
|
2019-08-30 20:19:03 +02:00
|
|
|
bool is_combo_key = false;
|
|
|
|
bool no_combo_keys_pressed = true;
|
|
|
|
|
|
|
|
if (keycode == CMB_ON && record->event.pressed) {
|
|
|
|
combo_enable();
|
|
|
|
return true;
|
|
|
|
}
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
if (keycode == CMB_OFF && record->event.pressed) {
|
|
|
|
combo_disable();
|
|
|
|
return true;
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
}
|
2016-12-16 20:50:28 +01:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
if (keycode == CMB_TOG && record->event.pressed) {
|
|
|
|
combo_toggle();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_combo_enabled()) {
|
|
|
|
return true;
|
|
|
|
}
|
2020-03-22 14:17:26 +01:00
|
|
|
#ifndef COMBO_VARIABLE_LEN
|
2019-08-30 20:19:03 +02:00
|
|
|
for (current_combo_index = 0; current_combo_index < COMBO_COUNT; ++current_combo_index) {
|
2020-03-25 04:39:53 +01:00
|
|
|
#else
|
2020-03-22 14:17:26 +01:00
|
|
|
for (current_combo_index = 0; current_combo_index < COMBO_LEN; ++current_combo_index) {
|
|
|
|
#endif
|
2019-08-30 20:19:03 +02:00
|
|
|
combo_t *combo = &key_combos[current_combo_index];
|
|
|
|
is_combo_key |= process_single_combo(combo, keycode, record);
|
|
|
|
no_combo_keys_pressed = no_combo_keys_pressed && NO_COMBO_KEYS_ARE_DOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drop_buffer) {
|
|
|
|
/* buffer is only dropped when we complete a combo, so we refresh the timer
|
|
|
|
* here */
|
|
|
|
timer = timer_read();
|
2021-07-04 21:07:31 +02:00
|
|
|
drop_buffer = false;
|
2019-08-30 20:19:03 +02:00
|
|
|
} else if (!is_combo_key) {
|
|
|
|
/* if no combos claim the key we need to emit the keybuffer */
|
|
|
|
dump_key_buffer(true);
|
2021-07-04 21:07:31 +02:00
|
|
|
end_incomplete_combos();
|
|
|
|
} else if (record->event.pressed) {
|
2019-08-30 20:19:03 +02:00
|
|
|
/* otherwise the key is consumed and placed in the buffer */
|
|
|
|
timer = timer_read();
|
|
|
|
|
|
|
|
if (buffer_size < MAX_COMBO_LENGTH) {
|
2016-12-16 20:50:28 +01:00
|
|
|
#ifdef COMBO_ALLOW_ACTION_KEYS
|
2019-08-30 20:19:03 +02:00
|
|
|
key_buffer[buffer_size++] = *record;
|
2016-12-16 20:50:28 +01:00
|
|
|
#else
|
2019-08-30 20:19:03 +02:00
|
|
|
key_buffer[buffer_size++] = keycode;
|
2016-12-16 20:50:28 +01:00
|
|
|
#endif
|
2019-08-30 20:19:03 +02:00
|
|
|
}
|
2016-12-10 15:11:59 +01:00
|
|
|
}
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
return !is_combo_key;
|
Switch process_combo to using global register and timer (#2561)
Since combos keep local state about what keys have been previously pressed, when combos are layered, multiple keypresses will register for any key with multiple combos assigned to it. In order to fix this, I switched process_combo to use a global keycode / keyrecord register and timer. When a keypress is consumed by a combo, it gets stored in the register and the timer is updated; when the next keypress takes too long or a key is pressed that isn't part of any combo, the buffer is emitted and the timer reset. This has a few side effects. For instance, I couldn't _not_ fix combo keys printing out of order while also fixing this bug, so combo keys print in order correctly when a combo fails. since combos no longer have local timers, the logic around when combos time out has changed. now that there is a single timer pressing any combo key (including one in a different combo) will reset the timer for all combos, making combo entry a little more lenient. Since combos no longer have local keycode / keyrecord state, there is an edge case where incomplete combo keys can be consumed. if you have a combo for a+s = tab and a combo for b+n = space, if you press a+b+n, only a space will be emitted. This is because when b+n completes successfully, it drops the register.
2019-04-08 23:07:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void matrix_scan_combo(void) {
|
2021-07-05 01:04:46 +02:00
|
|
|
if (b_combo_enable && timer && timer_elapsed(timer) > QS_combo_term) {
|
2019-08-30 20:19:03 +02:00
|
|
|
/* This disables the combo, meaning key events for this
|
|
|
|
* combo will be handled by the next processors in the chain
|
|
|
|
*/
|
|
|
|
dump_key_buffer(true);
|
2021-07-04 21:07:31 +02:00
|
|
|
end_incomplete_combos();
|
2019-08-30 20:19:03 +02:00
|
|
|
}
|
2016-12-16 20:50:28 +01:00
|
|
|
}
|
2019-07-16 10:37:19 +02:00
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
void combo_enable(void) { b_combo_enable = true; }
|
2019-07-16 10:37:19 +02:00
|
|
|
|
|
|
|
void combo_disable(void) {
|
2021-07-04 21:07:31 +02:00
|
|
|
b_combo_enable = false;
|
2019-08-30 20:19:03 +02:00
|
|
|
timer = 0;
|
2019-07-16 10:37:19 +02:00
|
|
|
dump_key_buffer(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void combo_toggle(void) {
|
|
|
|
if (b_combo_enable) {
|
|
|
|
combo_disable();
|
|
|
|
} else {
|
|
|
|
combo_enable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-30 20:19:03 +02:00
|
|
|
bool is_combo_enabled(void) { return b_combo_enable; }
|