Remove IGNORE_MOD_TAP_INTERRUPT_PER_KEY in favour of HOLD_ON_OTHER_KEY_PRESS_PER_KEY (#15741)
This commit is contained in:
parent
ca13734f91
commit
515dd18c28
@ -32,8 +32,6 @@
|
|||||||
"FORCE_NKRO": {"info_key": "usb.force_nkro", "value_type": "bool"},
|
"FORCE_NKRO": {"info_key": "usb.force_nkro", "value_type": "bool"},
|
||||||
"DYNAMIC_KEYMAP_EEPROM_MAX_ADDR": {"info_key": "dynamic_keymap.eeprom_max_addr", "value_type": "int"},
|
"DYNAMIC_KEYMAP_EEPROM_MAX_ADDR": {"info_key": "dynamic_keymap.eeprom_max_addr", "value_type": "int"},
|
||||||
"DYNAMIC_KEYMAP_LAYER_COUNT": {"info_key": "dynamic_keymap.layer_count", "value_type": "int"},
|
"DYNAMIC_KEYMAP_LAYER_COUNT": {"info_key": "dynamic_keymap.layer_count", "value_type": "int"},
|
||||||
"IGNORE_MOD_TAP_INTERRUPT": {"info_key": "tapping.ignore_mod_tap_interrupt", "value_type": "bool"},
|
|
||||||
"IGNORE_MOD_TAP_INTERRUPT_PER_KEY": {"info_key": "tapping.ignore_mod_tap_interrupt_per_key", "value_type": "bool"},
|
|
||||||
"HOLD_ON_OTHER_KEY_PRESS": {"info_key": "tapping.hold_on_other_key_press", "value_type": "bool"},
|
"HOLD_ON_OTHER_KEY_PRESS": {"info_key": "tapping.hold_on_other_key_press", "value_type": "bool"},
|
||||||
"HOLD_ON_OTHER_KEY_PRESS_PER_KEY": {"info_key": "tapping.hold_on_other_key_press_per_key", "value_type": "bool"},
|
"HOLD_ON_OTHER_KEY_PRESS_PER_KEY": {"info_key": "tapping.hold_on_other_key_press_per_key", "value_type": "bool"},
|
||||||
"LAYOUTS": {"info_key": "layout_aliases", "value_type": "mapping"},
|
"LAYOUTS": {"info_key": "layout_aliases", "value_type": "mapping"},
|
||||||
@ -129,6 +127,8 @@
|
|||||||
"QMK_KEYS_PER_SCAN": {"info_key": "qmk.keys_per_scan", "value_type": "int", "deprecated": true},
|
"QMK_KEYS_PER_SCAN": {"info_key": "qmk.keys_per_scan", "value_type": "int", "deprecated": true},
|
||||||
"TAPPING_FORCE_HOLD": {"info_key": "tapping.force_hold", "value_type": "bool", "deprecated": true},
|
"TAPPING_FORCE_HOLD": {"info_key": "tapping.force_hold", "value_type": "bool", "deprecated": true},
|
||||||
"TAPPING_FORCE_HOLD_PER_KEY": {"info_key": "tapping.force_hold_per_key", "value_type": "bool", "deprecated": true},
|
"TAPPING_FORCE_HOLD_PER_KEY": {"info_key": "tapping.force_hold_per_key", "value_type": "bool", "deprecated": true},
|
||||||
|
"IGNORE_MOD_TAP_INTERRUPT": {"info_key": "_deprecated.ignore_mod_tap_interrupt", "value_type": "bool", "deprecated": true},
|
||||||
|
"IGNORE_MOD_TAP_INTERRUPT_PER_KEY": {"info_key": "_invalid.ignore_mod_tap_interrupt_per_key", "invalid": true}
|
||||||
|
|
||||||
// USB params, need to mark as failure when specified in config.h, rather than deprecated
|
// USB params, need to mark as failure when specified in config.h, rather than deprecated
|
||||||
"PRODUCT_ID": {"info_key": "usb.pid", "value_type": "hex", "deprecated": true, "replace_with": "`usb.pid` in info.json"},
|
"PRODUCT_ID": {"info_key": "usb.pid", "value_type": "hex", "deprecated": true, "replace_with": "`usb.pid` in info.json"},
|
||||||
|
@ -620,7 +620,6 @@
|
|||||||
"force_hold": {"type": "boolean"},
|
"force_hold": {"type": "boolean"},
|
||||||
"force_hold_per_key": {"type": "boolean"},
|
"force_hold_per_key": {"type": "boolean"},
|
||||||
"ignore_mod_tap_interrupt": {"type": "boolean"},
|
"ignore_mod_tap_interrupt": {"type": "boolean"},
|
||||||
"ignore_mod_tap_interrupt_per_key": {"type": "boolean"},
|
|
||||||
"hold_on_other_key_press": {"type": "boolean"},
|
"hold_on_other_key_press": {"type": "boolean"},
|
||||||
"hold_on_other_key_press_per_key": {"type": "boolean"},
|
"hold_on_other_key_press_per_key": {"type": "boolean"},
|
||||||
"permissive_hold": {"type": "boolean"},
|
"permissive_hold": {"type": "boolean"},
|
||||||
|
43
docs/ChangeLog/20230226/PR15741.md
Normal file
43
docs/ChangeLog/20230226/PR15741.md
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
`IGNORE_MOD_TAP_INTERRUPT_PER_KEY` has been removed and `IGNORE_MOD_TAP_INTERRUPT` deprecated as a stepping stone towards making `IGNORE_MOD_TAP_INTERRUPT` the new default behavior for mod-taps in the future.
|
||||||
|
|
||||||
|
In place of the now removed `IGNORE_MOD_TAP_INTERRUPT_PER_KEY`, one must use the pre-existing `HOLD_ON_OTHER_KEY_PRESS` option.
|
||||||
|
|
||||||
|
In most cases, updating `get_ignore_mod_tap_interrupt` to `get_hold_on_other_key_press` is simply a matter of renaming the function and swapping every `true` by `false` and vice versa. The one subtlety you may need to look out for is that the `get_ignore_mod_tap_interrupt` was only ever called with mod-taps passed in as the `keycode` argument, while the `keycode` argument of `get_hold_on_other_key_press` can be any dual-role key. This includes not only mod-taps, but also layer-taps, one shot keys, `TT(layer)` and more. This has an impact on the effect of the `default` case in a typical per-key configuration making use of a `switch(keycode)` statement.
|
||||||
|
|
||||||
|
To illustrate, let's take the example of a configuration where we'd want all mod-taps to activate the modifier if another key is pressed while held with the exception of `LCTL_T(KC_A)`, which should ignore keys pressed while it is held and activate the modifier only if it has been held for longer than the tapping term. In addition, we would like to keep the default "ignore-interrupt" behavior of layer taps.
|
||||||
|
|
||||||
|
An old way to do this would be via the following code:
|
||||||
|
|
||||||
|
```c
|
||||||
|
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
||||||
|
switch(keycode) {
|
||||||
|
case LCTL_T(KC_A):
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The correct way to update this code without accidentally changing how the layer-taps work would be the following:
|
||||||
|
|
||||||
|
```c
|
||||||
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
|
switch(keycode) {
|
||||||
|
// Capture all mod-tap keycodes.
|
||||||
|
case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
||||||
|
if (keycode == LCTL_T(KC_A)) {
|
||||||
|
// Disable HOLD_ON_OTHER_KEY_PRESS for LCTL_T(KC_A)
|
||||||
|
// aka enable IGNORE_MOD_TAP_INTERRUPT for LCTL_T(KC_A).
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
// Enable HOLD_ON_OTHER_KEY_PRESS for every other mod-tap keycode.
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
For more information, you are invited to read the sections on [IGNORE_MOD_TAP_INTERRUPT](tap_hold.md#ignore-mod-tap-interrupt) and [HOLD_ON_OTHER_KEY_PRESS](tap_hold.md#hold-on-other-key-press) in the page on [Tap-Hold configuration options](tap_hold.md).
|
@ -169,8 +169,6 @@ If you define these options you will enable the associated feature, which may in
|
|||||||
* `#define IGNORE_MOD_TAP_INTERRUPT`
|
* `#define IGNORE_MOD_TAP_INTERRUPT`
|
||||||
* makes it possible to do rolling combos (zx) with keys that convert to other keys on hold, by enforcing the `TAPPING_TERM` for both keys.
|
* makes it possible to do rolling combos (zx) with keys that convert to other keys on hold, by enforcing the `TAPPING_TERM` for both keys.
|
||||||
* See [Ignore Mod Tap Interrupt](tap_hold.md#ignore-mod-tap-interrupt) for details
|
* See [Ignore Mod Tap Interrupt](tap_hold.md#ignore-mod-tap-interrupt) for details
|
||||||
* `#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY`
|
|
||||||
* enables handling for per key `IGNORE_MOD_TAP_INTERRUPT` settings
|
|
||||||
* `#define QUICK_TAP_TERM 100`
|
* `#define QUICK_TAP_TERM 100`
|
||||||
* tap-then-hold timing to use a dual role key to repeat keycode
|
* tap-then-hold timing to use a dual role key to repeat keycode
|
||||||
* See [Quick Tap Term](tap_hold.md#quick-tap-term)
|
* See [Quick Tap Term](tap_hold.md#quick-tap-term)
|
||||||
@ -178,6 +176,11 @@ If you define these options you will enable the associated feature, which may in
|
|||||||
* Defaults to `TAPPING_TERM` if not defined
|
* Defaults to `TAPPING_TERM` if not defined
|
||||||
* `#define QUICK_TAP_TERM_PER_KEY`
|
* `#define QUICK_TAP_TERM_PER_KEY`
|
||||||
* enables handling for per key `QUICK_TAP_TERM` settings
|
* enables handling for per key `QUICK_TAP_TERM` settings
|
||||||
|
* `#define HOLD_ON_OTHER_KEY_PRESS`
|
||||||
|
* selects the hold action of a dual-role key as soon as the tap of the dual-role key is interrupted by the press of another key.
|
||||||
|
* See "[hold on other key press](tap_hold.md#hold-on-other-key-press)" for details
|
||||||
|
* `#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY`
|
||||||
|
* enables handling for per key `HOLD_ON_OTHER_KEY_PRESS` settings
|
||||||
* `#define LEADER_TIMEOUT 300`
|
* `#define LEADER_TIMEOUT 300`
|
||||||
* how long before the leader key times out
|
* how long before the leader key times out
|
||||||
* If you're having issues finishing the sequence before it times out, you may need to increase the timeout setting. Or you may want to enable the `LEADER_PER_KEY_TIMING` option, which resets the timeout after each key is tapped.
|
* If you're having issues finishing the sequence before it times out, you may need to increase the timeout setting. Or you may want to enable the `LEADER_PER_KEY_TIMING` option, which resets the timeout after each key is tapped.
|
||||||
|
@ -281,16 +281,7 @@ Tap Hold Configurations work a little differently when using Retro Shift.
|
|||||||
Referencing `TAPPING_TERM` makes little sense, as holding longer would result in
|
Referencing `TAPPING_TERM` makes little sense, as holding longer would result in
|
||||||
shifting one of the keys.
|
shifting one of the keys.
|
||||||
|
|
||||||
`IGNORE_MOD_TAP_INTERRUPT` changes *only* rolling from a mod tap (releasing it
|
`RETRO_SHIFT` enables [`PERMISSIVE_HOLD`-like behaviour](tap_hold.md#permissive-hold) (even if not explicitly enabled) on all mod-taps for which `RETRO_SHIFT` applies.
|
||||||
first), sending both keys instead of the modifier on the second. Its effects on
|
|
||||||
nested presses are ignored.
|
|
||||||
|
|
||||||
As nested taps were changed to act as though `PERMISSIVE_HOLD` is set unless only
|
|
||||||
`IGNORE_MOD_TAP_INTERRUPT` is (outside of Retro Shift), and Retro Shift ignores
|
|
||||||
`IGNORE_MOD_TAP_INTERRUPT`, `PERMISSIVE_HOLD` has no effect on Mod Taps.
|
|
||||||
|
|
||||||
Nested taps will *always* act as though the `TAPPING_TERM` was exceeded for both
|
|
||||||
Mod and Layer Tap keys.
|
|
||||||
|
|
||||||
## Using Auto Shift Setup
|
## Using Auto Shift Setup
|
||||||
|
|
||||||
|
@ -162,8 +162,6 @@ QMK での全ての利用可能な設定にはデフォルトがあります。
|
|||||||
* `#define IGNORE_MOD_TAP_INTERRUPT`
|
* `#define IGNORE_MOD_TAP_INTERRUPT`
|
||||||
* 両方のキーに `TAPPING_TERM` を適用することで、ホールド時に他のキーに変換するキーを使ってローリングコンボ (zx) をすることができるようにします
|
* 両方のキーに `TAPPING_TERM` を適用することで、ホールド時に他のキーに変換するキーを使ってローリングコンボ (zx) をすることができるようにします
|
||||||
* 詳細は [Ignore Mod Tap Interrupt](ja/tap_hold.md#ignore-mod-tap-interrupt) を見てください
|
* 詳細は [Ignore Mod Tap Interrupt](ja/tap_hold.md#ignore-mod-tap-interrupt) を見てください
|
||||||
* `#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY`
|
|
||||||
* キーごとの `IGNORE_MOD_TAP_INTERRUPT` 設定の処理を有効にします
|
|
||||||
* `#define TAPPING_FORCE_HOLD`
|
* `#define TAPPING_FORCE_HOLD`
|
||||||
* タップされた直後に、デュアルロールキーを修飾子として使用できるようにします
|
* タップされた直後に、デュアルロールキーを修飾子として使用できるようにします
|
||||||
* [Tapping Force Hold](ja/tap_hold.md#tapping-force-hold)を見てください
|
* [Tapping Force Hold](ja/tap_hold.md#tapping-force-hold)を見てください
|
||||||
|
@ -110,25 +110,6 @@ bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) {
|
|||||||
|
|
||||||
?> `許容ホールド`を有効にすると、これは両方がどのように動作するかを変更します。通常のキーには、最初のキーが最初に放された場合、あるいは両方のキーが `TAPPING_TERM` より長くホールドされた場合に、修飾キーが追加されます。
|
?> `許容ホールド`を有効にすると、これは両方がどのように動作するかを変更します。通常のキーには、最初のキーが最初に放された場合、あるいは両方のキーが `TAPPING_TERM` より長くホールドされた場合に、修飾キーが追加されます。
|
||||||
|
|
||||||
この機能をより細かく制御するために、以下を `config.h` に追加することができます:
|
|
||||||
|
|
||||||
```c
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
```
|
|
||||||
|
|
||||||
そして、以下の関数をキーマップに追加します:
|
|
||||||
|
|
||||||
```c
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
|
||||||
switch (keycode) {
|
|
||||||
case SFT_T(KC_SPC):
|
|
||||||
return true;
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## タッピング強制ホールド
|
## タッピング強制ホールド
|
||||||
|
|
||||||
`タッピング強制ホールド` を有効にするには、以下を `config.h` に追加します:
|
`タッピング強制ホールド` を有効にするには、以下を `config.h` に追加します:
|
||||||
|
@ -118,7 +118,7 @@ The reason is that `TAPPING_TERM` is a macro that expands to a constant integer
|
|||||||
|
|
||||||
The code which decides between the tap and hold actions of dual-role keys supports three different modes, in increasing order of preference for the hold action:
|
The code which decides between the tap and hold actions of dual-role keys supports three different modes, in increasing order of preference for the hold action:
|
||||||
|
|
||||||
1. The default mode selects the hold action only if the dual-role key is held down longer than the tapping term. In this mode pressing other keys while the dual-role key is held down does not influence the tap-or-hold decision.
|
1. The default mode selects the hold action only if the dual-role key is held down longer than the tapping term. In this mode pressing other keys while the dual-role key is held down does not influence the tap-or-hold decision. In other words, this mode ignores interrupts.
|
||||||
|
|
||||||
2. The “permissive hold” mode, in addition to the default behavior, immediately selects the hold action when another key is tapped (pressed and then released) while the dual-role key is held down, even if this happens earlier than the tapping term. If another key is just pressed, but then the dual-role key is released before that other key (and earlier than the tapping term), this mode will still select the tap action.
|
2. The “permissive hold” mode, in addition to the default behavior, immediately selects the hold action when another key is tapped (pressed and then released) while the dual-role key is held down, even if this happens earlier than the tapping term. If another key is just pressed, but then the dual-role key is released before that other key (and earlier than the tapping term), this mode will still select the tap action.
|
||||||
|
|
||||||
@ -126,6 +126,73 @@ The code which decides between the tap and hold actions of dual-role keys suppor
|
|||||||
|
|
||||||
Note that until the tap-or-hold decision completes (which happens when either the dual-role key is released, or the tapping term has expired, or the extra condition for the selected decision mode is satisfied), key events are delayed and not transmitted to the host immediately. The default mode gives the most delay (if the dual-role key is held down, this mode always waits for the whole tapping term), and the other modes may give less delay when other keys are pressed, because the hold action may be selected earlier.
|
Note that until the tap-or-hold decision completes (which happens when either the dual-role key is released, or the tapping term has expired, or the extra condition for the selected decision mode is satisfied), key events are delayed and not transmitted to the host immediately. The default mode gives the most delay (if the dual-role key is held down, this mode always waits for the whole tapping term), and the other modes may give less delay when other keys are pressed, because the hold action may be selected earlier.
|
||||||
|
|
||||||
|
### Comparison :id=comparison
|
||||||
|
|
||||||
|
To better illustrate the tap-or-hold decision modes, let us compare the expected output of each decision mode in a handful of tapping scenarios involving a mod-tap key (`LSFT_T(KC_A)`) and a regular key (`KC_B`) with the `TAPPING_TERM` set to 200ms.
|
||||||
|
|
||||||
|
By default, mod-taps behave like `HOLD_ON_OTHER_KEY_PRESS`, while layer-taps behave like "Ignore Interrupt" out of the box. If you want "Ignore Interrupt"-like behaviour for mod-taps, you must enable `IGNORE_MOD_TAP_INTERRUPT`, or return `false` in the `get_hold_on_other_key_press` function for all mod-taps.
|
||||||
|
|
||||||
|
Note: "`kc` held" in the "Physical key event" column means that the key wasn't physically released yet at this point in time.
|
||||||
|
|
||||||
|
#### Distinct taps (AABB) :id=distinct-taps
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 199 | `LSFT_T(KC_A)` up | a | a | a |
|
||||||
|
| 210 | `KC_B` down | ab | ab | ab |
|
||||||
|
| 220 | `KC_B` up | ab | ab | ab |
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 200 | `LSFT_T(KC_A)` held|<kbd>Shift</kbd>| <kbd>Shift</kbd> | <kbd>Shift</kbd> |
|
||||||
|
| 201 | `LSFT_T(KC_A)` up |<kbd>Shift</kbd>| <kbd>Shift</kbd> | <kbd>Shift</kbd> |
|
||||||
|
| 205 | `KC_B` down | b | b | b |
|
||||||
|
| 210 | `KC_B` up | b | b | b |
|
||||||
|
|
||||||
|
#### Nested tap (ABBA) :id=nested-tap
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 110 | `KC_B` down | | | B |
|
||||||
|
| 120 | `KC_B` up | | B | B |
|
||||||
|
| 199 | `LSFT_T(KC_A)` up | ab | B | B |
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 110 | `KC_B` down | | | B |
|
||||||
|
| 120 | `KC_B` up | | B | B |
|
||||||
|
| 200 | `LSFT_T(KC_A)` held| B | B | B |
|
||||||
|
| 210 | `LSFT_T(KC_A)` up | B | B | B |
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 200 | `LSFT_T(KC_A)` held|<kbd>Shift</kbd>| <kbd>Shift</kbd> | <kbd>Shift</kbd> |
|
||||||
|
| 205 | `KC_B` down | B | B | B |
|
||||||
|
| 210 | `KC_B` up | B | B | B |
|
||||||
|
| 220 | `LSFT_T(KC_A)` up | B | B | B |
|
||||||
|
|
||||||
|
#### Rolling keys (ABAB) :id=rolling-keys
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 110 | `KC_B` down | | | B |
|
||||||
|
| 130 | `LSFT_T(KC_A)` up | ab | ab | B |
|
||||||
|
| 140 | `KC_B` up | ab | ab | B |
|
||||||
|
|
||||||
|
| Time | Physical key event |Ignore Interrupt| `PERMISSIVE_HOLD` | `HOLD_ON_OTHER_KEY_PRESS` |
|
||||||
|
|------|--------------------|----------------|-------------------|----------------------------|
|
||||||
|
| 0 | `LSFT_T(KC_A)` down| | | |
|
||||||
|
| 110 | `KC_B` down | | | B |
|
||||||
|
| 200 | `LSFT_T(KC_A)` held| B | B | B |
|
||||||
|
| 205 | `LSFT_T(KC_A)` up | B | B | B |
|
||||||
|
| 210 | `KC_B` up | B | B | B |
|
||||||
|
|
||||||
### Default Mode
|
### Default Mode
|
||||||
Example sequence 1 (the `L` key is also mapped to `KC_RGHT` on layer 2):
|
Example sequence 1 (the `L` key is also mapped to `KC_RGHT` on layer 2):
|
||||||
|
|
||||||
@ -179,8 +246,6 @@ since `SFT_T(KC_A)` is NOT held longer than the `TAPPING_TERM`.
|
|||||||
However, the actual output would be capital `X` (`SHIFT` + `x`) due to reasons
|
However, the actual output would be capital `X` (`SHIFT` + `x`) due to reasons
|
||||||
explained under [Ignore Mod Tap Interrupt](#ignore-mod-tap-interrupt).
|
explained under [Ignore Mod Tap Interrupt](#ignore-mod-tap-interrupt).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Permissive Hold
|
### Permissive Hold
|
||||||
|
|
||||||
The “permissive hold” mode can be enabled for all dual-role keys by adding the corresponding option to `config.h`:
|
The “permissive hold” mode can be enabled for all dual-role keys by adding the corresponding option to `config.h`:
|
||||||
@ -212,7 +277,7 @@ An example of a sequence that is affected by the “permissive hold” mode:
|
|||||||
+---------------------------|--------+
|
+---------------------------|--------+
|
||||||
```
|
```
|
||||||
|
|
||||||
Normally, if you do all this within the `TAPPING_TERM` (default: 200ms), this will be registered as `al` by the firmware and host system. With the `PERMISSIVE_HOLD` option enabled, the Layer Tap key is considered as a layer switch if another key is tapped, and the above sequence would be registered as `KC_RGHT` (the mapping of `L` on layer 2). We could describe this sequence as a “nested press” (the modified key's key down and key up events are “nested” between the dual-role key's key down and key up events).
|
Normally, if you do all this within the `TAPPING_TERM` (default: 200ms), this will be registered as `al` by the firmware and host system. With the `PERMISSIVE_HOLD` option enabled, the Layer Tap key is considered as a layer switch if another key is tapped, and the above sequence would be registered as `KC_RGHT` (the mapping of `L` on layer 2). We could describe this sequence as a “nested tap” (the modified key's key down and key up events are “nested” between the dual-role key's key down and key up events).
|
||||||
|
|
||||||
However, this slightly different sequence will not be affected by the “permissive hold” mode:
|
However, this slightly different sequence will not be affected by the “permissive hold” mode:
|
||||||
|
|
||||||
@ -235,7 +300,7 @@ However, this slightly different sequence will not be affected by the “permiss
|
|||||||
|
|
||||||
In the sequence above the dual-role key is released before the other key is released, and if that happens within the tapping term, the “permissive hold” mode will still choose the tap action for the dual-role key, and the sequence will be registered as `al` by the host. We could describe this as a “rolling press” (the two keys' key down and key up events behave as if you were rolling a ball across the two keys, first pressing each key down in sequence and then releasing them in the same order).
|
In the sequence above the dual-role key is released before the other key is released, and if that happens within the tapping term, the “permissive hold” mode will still choose the tap action for the dual-role key, and the sequence will be registered as `al` by the host. We could describe this as a “rolling press” (the two keys' key down and key up events behave as if you were rolling a ball across the two keys, first pressing each key down in sequence and then releasing them in the same order).
|
||||||
|
|
||||||
?> The `PERMISSIVE_HOLD` option also affects Mod Tap keys, but this may not be noticeable if you do not also enable the `IGNORE_MOD_TAP_INTERRUPT` option for those keys, because the default handler for Mod Tap keys also considers both the “nested press” and “rolling press” sequences like shown above as a modifier hold, not the tap action. If you do not enable `IGNORE_MOD_TAP_INTERRUPT`, the effect of `PERMISSIVE_HOLD` on Mod Tap keys would be limited to reducing the delay before the key events are made visible to the host.
|
?> The `PERMISSIVE_HOLD` option is not noticeable if you also enable `HOLD_ON_OTHER_KEY_PRESS` because the latter option considers both the “nested tap” and “rolling press” sequences like shown above as a hold action, not the tap action. `HOLD_ON_OTHER_KEY_PRESS` makes the Tap-Or-Hold decision earlier in the chain of key events, thus taking a precedence over `PERMISSIVE_HOLD`. This remark also applies to default mod-taps.
|
||||||
|
|
||||||
For more granular control of this feature, you can add the following to your `config.h`:
|
For more granular control of this feature, you can add the following to your `config.h`:
|
||||||
|
|
||||||
@ -291,7 +356,7 @@ An example of a sequence that is affected by the “hold on other key press” m
|
|||||||
|
|
||||||
Normally, if you do all this within the `TAPPING_TERM` (default: 200ms), this will be registered as `al` by the firmware and host system. With the `HOLD_ON_OTHER_KEY_PRESS` option enabled, the Layer Tap key is considered as a layer switch if another key is pressed, and the above sequence would be registered as `KC_RGHT` (the mapping of `L` on layer 2).
|
Normally, if you do all this within the `TAPPING_TERM` (default: 200ms), this will be registered as `al` by the firmware and host system. With the `HOLD_ON_OTHER_KEY_PRESS` option enabled, the Layer Tap key is considered as a layer switch if another key is pressed, and the above sequence would be registered as `KC_RGHT` (the mapping of `L` on layer 2).
|
||||||
|
|
||||||
?> The `HOLD_ON_OTHER_KEY_PRESS` option also affects Mod Tap keys, but this may not be noticeable if you do not also enable the `IGNORE_MOD_TAP_INTERRUPT` option for those keys, because the default handler for Mod Tap keys also considers the “rolling press” sequence like shown above as a modifier hold, not the tap action. If you do not enable `IGNORE_MOD_TAP_INTERRUPT`, the effect of `HOLD_ON_OTHER_KEY_PRESS` on Mod Tap keys would be limited to reducing the delay before the key events are made visible to the host.
|
?> The `HOLD_ON_OTHER_KEY_PRESS` option is essentially redundant with the default mod-tap behaviour. The only notable difference is that `HOLD_ON_OTHER_KEY_PRESS` reduces the delay before the key events are made visible to the host.
|
||||||
|
|
||||||
For more granular control of this feature, you can add the following to your `config.h`:
|
For more granular control of this feature, you can add the following to your `config.h`:
|
||||||
|
|
||||||
@ -355,26 +420,32 @@ However, if the `HOLD_ON_OTHER_KEY_PRESS` option is enabled in addition to `IGNO
|
|||||||
For more granular control of this feature, you can add the following to your `config.h`:
|
For more granular control of this feature, you can add the following to your `config.h`:
|
||||||
|
|
||||||
```c
|
```c
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
```
|
```
|
||||||
|
|
||||||
|
?> This option affects *all* dual-role keys.
|
||||||
|
|
||||||
You can then add the following function to your keymap:
|
You can then add the following function to your keymap:
|
||||||
|
|
||||||
```c
|
```c
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case SFT_T(KC_SPC):
|
case SFT_T(KC_SPC):
|
||||||
// Do not force the mod-tap key press to be handled as a modifier
|
// Do not force the mod-tap key press to be handled as a modifier
|
||||||
// if any other key was pressed while the mod-tap key is held down.
|
// if any other key was pressed while the mod-tap key is held down.
|
||||||
return true;
|
|
||||||
default:
|
|
||||||
// Force the mod-tap key press to be handled as a modifier if any
|
|
||||||
// other key was pressed while the mod-tap key is held down.
|
|
||||||
return false;
|
return false;
|
||||||
|
default:
|
||||||
|
// Force the dual-role key press to be handled as a modifier if any
|
||||||
|
// other key was pressed while the mod-tap key is held down.
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Note that you must return `false` in `get_hold_on_other_key_press` in order to apply `IGNORE_MOD_TAP_INTERRUPT` for a certain mod-tap key.
|
||||||
|
|
||||||
|
?> `IGNORE_MOD_TAP_INTERRUPT[_PER_KEY]` is being progressively phased out to align the (default) behavior and configuration of mod-taps with the rest of dual-role keys.
|
||||||
|
|
||||||
## Quick Tap Term
|
## Quick Tap Term
|
||||||
|
|
||||||
When the user holds a key after tapping it, the tapping function is repeated by default, rather than activating the hold function. This allows keeping the ability to auto-repeat the tapping function of a dual-role key. `QUICK_TAP_TERM` enables fine tuning of that ability. If set to `0`, it will remove the auto-repeat ability and activate the hold function instead.
|
When the user holds a key after tapping it, the tapping function is repeated by default, rather than activating the hold function. This allows keeping the ability to auto-repeat the tapping function of a dual-role key. `QUICK_TAP_TERM` enables fine tuning of that ability. If set to `0`, it will remove the auto-repeat ability and activate the hold function instead.
|
||||||
|
@ -1,3 +1,2 @@
|
|||||||
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
#define QUICK_TAP_TERM_PER_KEY
|
#define QUICK_TAP_TERM_PER_KEY
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
|
@ -93,21 +93,15 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
|
|||||||
),
|
),
|
||||||
};
|
};
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
// Special if-condition outside the switch because `RC_QUT` overlaps with
|
||||||
case RC_QUT:
|
// the `QK_MOD_TAP ... QK_MOD_TAP_MAX` range.
|
||||||
return true;
|
if (keycode == RC_QUT) {
|
||||||
default:
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case LLS_ESC:
|
case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
||||||
case LLS_RALT:
|
case QK_LAYER_TAP ... QK_LAYER_TAP_MAX:
|
||||||
case LLE_ENT:
|
|
||||||
case LLA_DEL:
|
|
||||||
return true;
|
return true;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
|
@ -28,15 +28,12 @@
|
|||||||
|
|
||||||
#define TAPPING_TERM 200
|
#define TAPPING_TERM 200
|
||||||
|
|
||||||
// Prevent normal rollover on alphas from accidentally triggering mods.
|
|
||||||
// #define IGNORE_MOD_TAP_INTERRUPT
|
|
||||||
|
|
||||||
// Enable rapid switch from tap to hold, disables double tap hold auto-repeat.
|
// Enable rapid switch from tap to hold, disables double tap hold auto-repeat.
|
||||||
// #define QUICK_TAP_TERM 0
|
// #define QUICK_TAP_TERM 0
|
||||||
|
|
||||||
// Apply the modifier on keys that are tapped during a short hold of a modtap
|
// Apply the modifier on keys that are tapped during a short hold of a modtap
|
||||||
// #define PERMISSIVE_HOLD
|
// #define PERMISSIVE_HOLD
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
|
|
||||||
#define FORCE_NKRO
|
#define FORCE_NKRO
|
||||||
|
@ -169,12 +169,13 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
|
|||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
if (keycode == CM_SPAR) {
|
||||||
case CM_SPAR:
|
return false;
|
||||||
return true;
|
} else if (QK_MOD_TAP <= keycode && keycode <= QK_MOD_TAP_MAX) {
|
||||||
default:
|
return true;
|
||||||
return false;
|
} else {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,8 +18,7 @@
|
|||||||
|
|
||||||
#define TAPPING_TERM 200 // Delay for tap modifiers until it is considered a hold
|
#define TAPPING_TERM 200 // Delay for tap modifiers until it is considered a hold
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT // Enable ignore mod tap interrupt: https://docs.qmk.fm/#/tap_hold?id=ignore-mod-tap-interrupt
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY // Allows configuration of hold on other key press per key in keymap.c
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY // Allows configuration of ignore mod tap interrupt per key in keymap.c
|
|
||||||
|
|
||||||
#define COMBO_COUNT 2 // Number of defined combos
|
#define COMBO_COUNT 2 // Number of defined combos
|
||||||
#define COMBO_TERM 20 // Delay for combo keys to be chained together
|
#define COMBO_TERM 20 // Delay for combo keys to be chained together
|
||||||
|
@ -174,13 +174,13 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
|
|||||||
};
|
};
|
||||||
// clang-format on
|
// clang-format on
|
||||||
|
|
||||||
// Configure ignore mod tap interrupt per key
|
// Configure hold on other key press per key
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
// I don't like how mod tap interrupt feels with these keys specifically when I'm typing
|
// I don't like how ignore interrupt feels with these keys specifically when I'm typing
|
||||||
case LCTL_T(KC_ESC):
|
case LCTL_T(KC_ESC):
|
||||||
return false;
|
|
||||||
default:
|
|
||||||
return true;
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -57,7 +57,6 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
|
|
||||||
// fix for me putting alt under A and being a fast typist
|
// fix for me putting alt under A and being a fast typist
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
#define IGNORE_MOD_TAP_INTERRUPT
|
||||||
//#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
|
|
||||||
#define LAYER_STATE_8BIT
|
#define LAYER_STATE_8BIT
|
||||||
#define SPLIT_WPM_ENABLE
|
#define SPLIT_WPM_ENABLE
|
||||||
|
@ -198,12 +198,14 @@ bool process_record_user(uint16_t keycode, keyrecord_t *record) {
|
|||||||
// }
|
// }
|
||||||
//}
|
//}
|
||||||
//
|
//
|
||||||
// bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
// bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
|
// if (keycode == ALT_T(KC_A) || keycode == SH_BKSP) {
|
||||||
|
// return false;
|
||||||
|
// }
|
||||||
// switch (keycode) {
|
// switch (keycode) {
|
||||||
// case ALT_T(KC_A):
|
// case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
||||||
// case SH_BKSP:
|
|
||||||
// return true;
|
// return true;
|
||||||
// default:
|
// default:
|
||||||
// return false;
|
// return false;
|
||||||
// }
|
// }
|
||||||
//}
|
//}
|
||||||
|
@ -29,8 +29,8 @@
|
|||||||
#define PERMISSIVE_HOLD
|
#define PERMISSIVE_HOLD
|
||||||
/* #define PERMISSIVE_HOLD_PER_KEY */
|
/* #define PERMISSIVE_HOLD_PER_KEY */
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
#define HOLD_ON_OTHER_KEY_PRESS
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
|
|
||||||
#define QUICK_TAP_TERM 0
|
#define QUICK_TAP_TERM 0
|
||||||
#define QUICK_TAP_TERM_PER_KEY
|
#define QUICK_TAP_TERM_PER_KEY
|
||||||
|
@ -206,7 +206,7 @@ uint16_t get_tapping_term(uint16_t keycode, keyrecord_t *record) {
|
|||||||
return TAPPING_TERM;
|
return TAPPING_TERM;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case E_NUMBERS:
|
case E_NUMBERS:
|
||||||
case R_MOUSE:
|
case R_MOUSE:
|
||||||
@ -219,7 +219,7 @@ bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
|||||||
case RCTL_T(KC_L):
|
case RCTL_T(KC_L):
|
||||||
case RSFT_T(KC_SEMICOLON):
|
case RSFT_T(KC_SEMICOLON):
|
||||||
case ARROWS:
|
case ARROWS:
|
||||||
return true;
|
return false;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -45,10 +45,7 @@
|
|||||||
// Configure the global tapping term (default: 200ms)
|
// Configure the global tapping term (default: 200ms)
|
||||||
#define TAPPING_TERM 200
|
#define TAPPING_TERM 200
|
||||||
|
|
||||||
// Prevent normal rollover on alphas from accidentally triggering mods.
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
|
||||||
//#define IGNORE_MOD_TAP_INTERRUPT
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
|
|
||||||
// Enable rapid switch from tap to hold, disables double tap hold auto-repeat.
|
// Enable rapid switch from tap to hold, disables double tap hold auto-repeat.
|
||||||
#define QUICK_TAP_TERM 0
|
#define QUICK_TAP_TERM 0
|
||||||
|
@ -195,18 +195,18 @@ layer_state_t layer_state_set_user(layer_state_t state) {
|
|||||||
return update_tri_layer_state(state, _LOWER, _RAISE, _ADJUST);
|
return update_tri_layer_state(state, _LOWER, _RAISE, _ADJUST);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case SPC_L:
|
case SPC_L:
|
||||||
return false;
|
return true;
|
||||||
case SPC_R:
|
case SPC_R:
|
||||||
return true;
|
return false;
|
||||||
case ENT_L:
|
case ENT_L:
|
||||||
return false;
|
|
||||||
case ENT_R:
|
|
||||||
return false;
|
|
||||||
default:
|
|
||||||
return true;
|
return true;
|
||||||
|
case ENT_R:
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,4 +17,4 @@
|
|||||||
|
|
||||||
#define MASTER_LEFT
|
#define MASTER_LEFT
|
||||||
#define TAPPING_TERM 200
|
#define TAPPING_TERM 200
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
|
@ -96,9 +96,11 @@ const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
|
|||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case SY_SPC:
|
case SY_SPC:
|
||||||
|
return false;
|
||||||
|
case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
||||||
return true;
|
return true;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
|
@ -33,7 +33,7 @@
|
|||||||
// Tap-Hold Configs
|
// Tap-Hold Configs
|
||||||
#define TAPPING_TERM 180
|
#define TAPPING_TERM 180
|
||||||
#define PERMISSIVE_HOLD
|
#define PERMISSIVE_HOLD
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
#define HOLD_ON_OTHER_KEY_PRESS
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
#define QUICK_TAP_TERM 0
|
#define QUICK_TAP_TERM 0
|
||||||
#define QUICK_TAP_TERM_PER_KEY
|
#define QUICK_TAP_TERM_PER_KEY
|
||||||
|
@ -255,7 +255,7 @@ uint16_t get_quick_tap_term(uint16_t keycode, keyrecord_t *record) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case ALT__A:
|
case ALT__A:
|
||||||
case ALTSCLN:
|
case ALTSCLN:
|
||||||
@ -272,7 +272,7 @@ bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
|||||||
case CTL__J:
|
case CTL__J:
|
||||||
case CTL__T:
|
case CTL__T:
|
||||||
case CTL__N:
|
case CTL__N:
|
||||||
return true;
|
return false;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -118,7 +118,7 @@
|
|||||||
*/
|
*/
|
||||||
#define RETRO_TAPPING_PER_KEY // Control Retro-Tap individually by key
|
#define RETRO_TAPPING_PER_KEY // Control Retro-Tap individually by key
|
||||||
#define QUICK_TAP_TERM_PER_KEY // Control Quick-Tap individually by key
|
#define QUICK_TAP_TERM_PER_KEY // Control Quick-Tap individually by key
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY // Control Mod-Tap-Interrupt individually by key
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY // Control Hold-on-Other-Key-Press individually by key
|
||||||
#define PERMISSIVE_HOLD_PER_KEY // Control Permissive-Hold individually by key
|
#define PERMISSIVE_HOLD_PER_KEY // Control Permissive-Hold individually by key
|
||||||
|
|
||||||
#define MK_KINETIC_SPEED // Use kinetic acceleration for mouse-keys
|
#define MK_KINETIC_SPEED // Use kinetic acceleration for mouse-keys
|
||||||
|
@ -1369,9 +1369,11 @@ bool get_retro_tapping(uint16_t keycode, keyrecord_t* record) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Handles per-key configuration of Mod-Tap-Interrupt
|
// Handles per-key configuration of Hold-on-Other-Key-Press
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t* record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t* record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
|
case QK_MOD_TAP ... QK_MOD_TAP_MAX:
|
||||||
|
return true;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -17,5 +17,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#define TAPPING_TERM 200
|
#define TAPPING_TERM 200
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
#define PERMISSIVE_HOLD_PER_KEY
|
#define PERMISSIVE_HOLD_PER_KEY
|
||||||
|
@ -133,18 +133,18 @@ layer_state_t layer_state_set_user(layer_state_t state) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Enable `IGNORE_MOD_TAP_INTERRUPT` for all modifiers except `Shift`.
|
* Enable `HOLD_ON_OTHER_KEY_PRESS` only for `Shift`.
|
||||||
* For more info see `IGNORE_MOD_TAP_INTERRUPT_PER_KEY` in `docs/tap_hold.md`.
|
* For more info see `HOLD_ON_OTHER_KEY_PRESS_PER_KEY` in `docs/tap_hold.md`.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case S_EQL:
|
case S_EQL:
|
||||||
return false;
|
|
||||||
case S_MINS:
|
|
||||||
return false;
|
|
||||||
default:
|
|
||||||
return true;
|
return true;
|
||||||
|
case S_MINS:
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -55,8 +55,8 @@ int retro_tapping_counter = 0;
|
|||||||
# include "process_auto_shift.h"
|
# include "process_auto_shift.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
#ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
__attribute__((weak)) bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
__attribute__((weak)) bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -484,10 +484,10 @@ void process_action(keyrecord_t *record, action_t action) {
|
|||||||
default:
|
default:
|
||||||
if (event.pressed) {
|
if (event.pressed) {
|
||||||
if (tap_count > 0) {
|
if (tap_count > 0) {
|
||||||
# if !defined(IGNORE_MOD_TAP_INTERRUPT) || defined(IGNORE_MOD_TAP_INTERRUPT_PER_KEY)
|
# if !defined(IGNORE_MOD_TAP_INTERRUPT) || defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
||||||
if (
|
if (
|
||||||
# ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
!get_ignore_mod_tap_interrupt(get_event_keycode(record->event, false), record) &&
|
get_hold_on_other_key_press(get_event_keycode(record->event, false), record) &&
|
||||||
# endif
|
# endif
|
||||||
record->tap.interrupted) {
|
record->tap.interrupted) {
|
||||||
dprint("mods_tap: tap: cancel: add_mods\n");
|
dprint("mods_tap: tap: cancel: add_mods\n");
|
||||||
|
@ -15,6 +15,14 @@
|
|||||||
|
|
||||||
#ifndef NO_ACTION_TAPPING
|
#ifndef NO_ACTION_TAPPING
|
||||||
|
|
||||||
|
# if defined(IGNORE_MOD_TAP_INTERRUPT_PER_KEY)
|
||||||
|
# error "IGNORE_MOD_TAP_INTERRUPT_PER_KEY has been removed; the code needs to be ported to use HOLD_ON_OTHER_KEY_PRESS_PER_KEY instead."
|
||||||
|
# elif !defined(IGNORE_MOD_TAP_INTERRUPT)
|
||||||
|
# if !defined(PERMISSIVE_HOLD) && !defined(PERMISSIVE_HOLD_PER_KEY) && !defined(HOLD_ON_OTHER_KEY_PRESS) && !defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
||||||
|
# pragma message "The default behavior of mod-taps will change to mimic IGNORE_MOD_TAP_INTERRUPT in the future.\nIf you wish to keep the old default behavior of mod-taps, please use HOLD_ON_OTHER_KEY_PRESS."
|
||||||
|
# endif
|
||||||
|
# endif
|
||||||
|
|
||||||
# define IS_TAPPING() !IS_NOEVENT(tapping_key.event)
|
# define IS_TAPPING() !IS_NOEVENT(tapping_key.event)
|
||||||
# define IS_TAPPING_PRESSED() (IS_TAPPING() && tapping_key.event.pressed)
|
# define IS_TAPPING_PRESSED() (IS_TAPPING() && tapping_key.event.pressed)
|
||||||
# define IS_TAPPING_RELEASED() (IS_TAPPING() && !tapping_key.event.pressed)
|
# define IS_TAPPING_RELEASED() (IS_TAPPING() && !tapping_key.event.pressed)
|
||||||
@ -162,9 +170,7 @@ void action_tapping_process(keyrecord_t record) {
|
|||||||
# define TAP_GET_HOLD_ON_OTHER_KEY_PRESS false
|
# define TAP_GET_HOLD_ON_OTHER_KEY_PRESS false
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
# ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
# if defined(IGNORE_MOD_TAP_INTERRUPT)
|
||||||
# define TAP_GET_IGNORE_MOD_TAP_INTERRUPT get_ignore_mod_tap_interrupt(tapping_keycode, &tapping_key)
|
|
||||||
# elif defined(IGNORE_MOD_TAP_INTERRUPT)
|
|
||||||
# define TAP_GET_IGNORE_MOD_TAP_INTERRUPT true
|
# define TAP_GET_IGNORE_MOD_TAP_INTERRUPT true
|
||||||
# else
|
# else
|
||||||
# define TAP_GET_IGNORE_MOD_TAP_INTERRUPT false
|
# define TAP_GET_IGNORE_MOD_TAP_INTERRUPT false
|
||||||
@ -216,7 +222,7 @@ bool process_tapping(keyrecord_t *keyp) {
|
|||||||
// Rolled over the two keys.
|
// Rolled over the two keys.
|
||||||
(tapping_key.tap.interrupted == true && (
|
(tapping_key.tap.interrupted == true && (
|
||||||
(TAP_IS_LT && TAP_GET_HOLD_ON_OTHER_KEY_PRESS) ||
|
(TAP_IS_LT && TAP_GET_HOLD_ON_OTHER_KEY_PRESS) ||
|
||||||
(TAP_IS_MT && !TAP_GET_IGNORE_MOD_TAP_INTERRUPT)
|
(TAP_IS_MT && TAP_GET_HOLD_ON_OTHER_KEY_PRESS)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
// Makes Retro Shift ignore [IGNORE_MOD_TAP_INTERRUPT's
|
// Makes Retro Shift ignore [IGNORE_MOD_TAP_INTERRUPT's
|
||||||
|
@ -397,8 +397,17 @@ bool process_auto_shift(uint16_t keycode, keyrecord_t *record) {
|
|||||||
break;
|
break;
|
||||||
# endif
|
# endif
|
||||||
}
|
}
|
||||||
// If Retro Shift is disabled, possible custom actions shouldn't happen.
|
// If Retro Shift is disabled, possible custom actions shouldn't happen.
|
||||||
// clang-format off
|
// clang-format off
|
||||||
|
# if defined(RETRO_SHIFT) && !defined(NO_ACTION_TAPPING)
|
||||||
|
# if defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
||||||
|
const bool is_hold_on_interrupt = get_hold_on_other_key_press(keycode, record);
|
||||||
|
# elif defined(IGNORE_MOD_TAP_INTERRUPT)
|
||||||
|
const bool is_hold_on_interrupt = false;
|
||||||
|
# else
|
||||||
|
const bool is_hold_on_interrupt = IS_MT(keycode);
|
||||||
|
# endif
|
||||||
|
# endif
|
||||||
if (IS_RETRO(keycode)
|
if (IS_RETRO(keycode)
|
||||||
# if defined(RETRO_SHIFT) && !defined(NO_ACTION_TAPPING)
|
# if defined(RETRO_SHIFT) && !defined(NO_ACTION_TAPPING)
|
||||||
// Not tapped or #defines mean that rolls should use hold action.
|
// Not tapped or #defines mean that rolls should use hold action.
|
||||||
@ -407,27 +416,7 @@ bool process_auto_shift(uint16_t keycode, keyrecord_t *record) {
|
|||||||
# ifdef RETRO_TAPPING_PER_KEY
|
# ifdef RETRO_TAPPING_PER_KEY
|
||||||
|| !get_retro_tapping(keycode, record)
|
|| !get_retro_tapping(keycode, record)
|
||||||
# endif
|
# endif
|
||||||
|| (record->tap.interrupted && (IS_LT(keycode)
|
|| (record->tap.interrupted && is_hold_on_interrupt))
|
||||||
# if defined(HOLD_ON_OTHER_KEY_PRESS) || defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
|
||||||
# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
|
||||||
? get_hold_on_other_key_press(keycode, record)
|
|
||||||
# else
|
|
||||||
? true
|
|
||||||
# endif
|
|
||||||
# else
|
|
||||||
? false
|
|
||||||
# endif
|
|
||||||
# if defined(IGNORE_MOD_TAP_INTERRUPT) || defined(IGNORE_MOD_TAP_INTERRUPT_PER_KEY)
|
|
||||||
# ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
: !get_ignore_mod_tap_interrupt(keycode, record)
|
|
||||||
# else
|
|
||||||
: false
|
|
||||||
# endif
|
|
||||||
# else
|
|
||||||
: true
|
|
||||||
# endif
|
|
||||||
))
|
|
||||||
)
|
|
||||||
# endif
|
# endif
|
||||||
) {
|
) {
|
||||||
// clang-format on
|
// clang-format on
|
||||||
@ -454,10 +443,10 @@ bool process_auto_shift(uint16_t keycode, keyrecord_t *record) {
|
|||||||
# endif
|
# endif
|
||||||
) {
|
) {
|
||||||
// Fixes modifiers not being applied to rolls with AUTO_SHIFT_MODIFIERS set.
|
// Fixes modifiers not being applied to rolls with AUTO_SHIFT_MODIFIERS set.
|
||||||
# if !defined(IGNORE_MOD_TAP_INTERRUPT) || defined(IGNORE_MOD_TAP_INTERRUPT_PER_KEY)
|
# if !defined(IGNORE_MOD_TAP_INTERRUPT) || defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
|
||||||
if (autoshift_flags.in_progress
|
if (autoshift_flags.in_progress
|
||||||
# ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
&& !get_ignore_mod_tap_interrupt(keycode, record)
|
&& get_hold_on_other_key_press(keycode, record)
|
||||||
# endif
|
# endif
|
||||||
) {
|
) {
|
||||||
autoshift_end(KC_NO, now, false, &autoshift_lastrecord);
|
autoshift_end(KC_NO, now, false, &autoshift_lastrecord);
|
||||||
|
@ -18,4 +18,4 @@
|
|||||||
|
|
||||||
#include "test_common.h"
|
#include "test_common.h"
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
#define IGNORE_MOD_TAP_INTERRUPT
|
||||||
|
@ -66,7 +66,7 @@ TEST_F(DefaultTapHold, tap_regular_key_while_mod_tap_key_is_held) {
|
|||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
testing::Mock::VerifyAndClearExpectations(&driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(DefaultTapHold, tap_mod_tap_key_while_mod_tap_key_is_held) {
|
TEST_F(DefaultTapHold, tap_a_mod_tap_key_while_another_mod_tap_key_is_held) {
|
||||||
TestDriver driver;
|
TestDriver driver;
|
||||||
InSequence s;
|
InSequence s;
|
||||||
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
/* Copyright 2021 Stefan Kerkmann
|
|
||||||
*
|
|
||||||
* 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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "test_common.h"
|
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
|
@ -1,18 +0,0 @@
|
|||||||
# Copyright 2021 Stefan Kerkmann
|
|
||||||
#
|
|
||||||
# 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/>.
|
|
||||||
|
|
||||||
# --------------------------------------------------------------------------------
|
|
||||||
# Keep this file, even if it is empty, as a marker that this folder contains tests
|
|
||||||
# --------------------------------------------------------------------------------
|
|
@ -1,136 +0,0 @@
|
|||||||
/* Copyright 2021 Stefan Kerkmann
|
|
||||||
*
|
|
||||||
* 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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "keyboard_report_util.hpp"
|
|
||||||
#include "keycode.h"
|
|
||||||
#include "test_common.hpp"
|
|
||||||
#include "action_tapping.h"
|
|
||||||
#include "test_fixture.hpp"
|
|
||||||
#include "test_keymap_key.hpp"
|
|
||||||
|
|
||||||
using testing::_;
|
|
||||||
using testing::InSequence;
|
|
||||||
|
|
||||||
class IgnoreModTapInterrupt : public TestFixture {};
|
|
||||||
|
|
||||||
TEST_F(IgnoreModTapInterrupt, tap_regular_key_while_mod_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
|
||||||
auto regular_key = KeymapKey(0, 2, 0, KC_A);
|
|
||||||
|
|
||||||
set_keymap({mod_tap_hold_key, regular_key});
|
|
||||||
|
|
||||||
/* Press mod-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release mod-tap-hold key */
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_REPORT(driver, (KC_A, KC_P));
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(IgnoreModTapInterrupt, tap_mod_tap_key_while_mod_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
|
||||||
auto second_mod_tap_hold_key = KeymapKey(0, 2, 0, RSFT_T(KC_A));
|
|
||||||
|
|
||||||
set_keymap({first_mod_tap_hold_key, second_mod_tap_hold_key});
|
|
||||||
|
|
||||||
/* Press first mod-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
first_mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press second tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
second_mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release second tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
second_mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release first mod-tap-hold key */
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_REPORT(driver, (KC_A, KC_P));
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
first_mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(IgnoreModTapInterrupt, tap_regular_key_while_layer_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto layer_tap_hold_key = KeymapKey(0, 1, 0, LT(1, KC_P));
|
|
||||||
auto regular_key = KeymapKey(0, 2, 0, KC_A);
|
|
||||||
auto layer_key = KeymapKey(1, 2, 0, KC_B);
|
|
||||||
|
|
||||||
set_keymap({layer_tap_hold_key, regular_key, layer_key});
|
|
||||||
|
|
||||||
/* Press layer-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
layer_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release layer-tap-hold key */
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_REPORT(driver, (KC_P, regular_key.report_code));
|
|
||||||
EXPECT_REPORT(driver, (KC_P));
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
layer_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
@ -60,7 +60,7 @@ TEST_F(PermissiveHold, tap_regular_key_while_mod_tap_key_is_held) {
|
|||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
testing::Mock::VerifyAndClearExpectations(&driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(PermissiveHold, tap_mod_tap_key_while_mod_tap_key_is_held) {
|
TEST_F(PermissiveHold, tap_a_mod_tap_key_while_another_mod_tap_key_is_held) {
|
||||||
TestDriver driver;
|
TestDriver driver;
|
||||||
InSequence s;
|
InSequence s;
|
||||||
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
/* Copyright 2021 Stefan Kerkmann
|
|
||||||
*
|
|
||||||
* 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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "test_common.h"
|
|
||||||
|
|
||||||
#define IGNORE_MOD_TAP_INTERRUPT
|
|
||||||
#define PERMISSIVE_HOLD
|
|
@ -1,18 +0,0 @@
|
|||||||
# Copyright 2021 Stefan Kerkmann
|
|
||||||
#
|
|
||||||
# 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/>.
|
|
||||||
|
|
||||||
# --------------------------------------------------------------------------------
|
|
||||||
# Keep this file, even if it is empty, as a marker that this folder contains tests
|
|
||||||
# --------------------------------------------------------------------------------
|
|
@ -1,133 +0,0 @@
|
|||||||
|
|
||||||
/* Copyright 2021 Stefan Kerkmann
|
|
||||||
*
|
|
||||||
* 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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "keyboard_report_util.hpp"
|
|
||||||
#include "keycode.h"
|
|
||||||
#include "test_common.hpp"
|
|
||||||
#include "action_tapping.h"
|
|
||||||
#include "test_fixture.hpp"
|
|
||||||
#include "test_keymap_key.hpp"
|
|
||||||
|
|
||||||
using testing::_;
|
|
||||||
using testing::InSequence;
|
|
||||||
|
|
||||||
class PermissiveHold_IgnoreModTapInterrupt : public TestFixture {};
|
|
||||||
|
|
||||||
TEST_F(PermissiveHold_IgnoreModTapInterrupt, tap_regular_key_while_mod_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
|
||||||
auto regular_key = KeymapKey(0, 2, 0, KC_A);
|
|
||||||
|
|
||||||
set_keymap({mod_tap_hold_key, regular_key});
|
|
||||||
|
|
||||||
/* Press mod-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release regular key */
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT));
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT, KC_A));
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT));
|
|
||||||
regular_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release mod-tap-hold key */
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(PermissiveHold_IgnoreModTapInterrupt, tap_mod_tap_key_while_mod_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto first_mod_tap_hold_key = KeymapKey(0, 1, 0, SFT_T(KC_P));
|
|
||||||
auto second_mod_tap_hold_key = KeymapKey(0, 2, 0, RSFT_T(KC_A));
|
|
||||||
|
|
||||||
set_keymap({first_mod_tap_hold_key, second_mod_tap_hold_key});
|
|
||||||
|
|
||||||
/* Press first mod-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
first_mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press second tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
second_mod_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release second tap-hold key */
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT));
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT, KC_A));
|
|
||||||
EXPECT_REPORT(driver, (KC_LSFT));
|
|
||||||
second_mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release first mod-tap-hold key */
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
first_mod_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_F(PermissiveHold_IgnoreModTapInterrupt, tap_regular_key_while_layer_tap_key_is_held) {
|
|
||||||
TestDriver driver;
|
|
||||||
InSequence s;
|
|
||||||
auto layer_tap_hold_key = KeymapKey(0, 1, 0, LT(1, KC_P));
|
|
||||||
auto regular_key = KeymapKey(0, 2, 0, KC_A);
|
|
||||||
auto layer_key = KeymapKey(1, 2, 0, KC_B);
|
|
||||||
|
|
||||||
set_keymap({layer_tap_hold_key, regular_key, layer_key});
|
|
||||||
|
|
||||||
/* Press layer-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
layer_tap_hold_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Press regular key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
regular_key.press();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release regular key */
|
|
||||||
EXPECT_REPORT(driver, (KC_B));
|
|
||||||
EXPECT_EMPTY_REPORT(driver);
|
|
||||||
regular_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
|
|
||||||
/* Release layer-tap-hold key */
|
|
||||||
EXPECT_NO_REPORT(driver);
|
|
||||||
layer_tap_hold_key.release();
|
|
||||||
run_one_scan_loop();
|
|
||||||
testing::Mock::VerifyAndClearExpectations(&driver);
|
|
||||||
}
|
|
@ -81,7 +81,6 @@
|
|||||||
#endif // !ONESHOT_TIMEOUT
|
#endif // !ONESHOT_TIMEOUT
|
||||||
|
|
||||||
#if defined(PER_KEY_TAPPING)
|
#if defined(PER_KEY_TAPPING)
|
||||||
# define IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
# define PERMISSIVE_HOLD_PER_KEY
|
# define PERMISSIVE_HOLD_PER_KEY
|
||||||
# define QUICK_TAP_TERM_PER_KEY
|
# define QUICK_TAP_TERM_PER_KEY
|
||||||
# define HOLD_ON_OTHER_KEY
|
# define HOLD_ON_OTHER_KEY
|
||||||
|
@ -42,21 +42,6 @@ __attribute__((weak)) bool get_hold_on_other_key_press(uint16_t keycode, keyreco
|
|||||||
}
|
}
|
||||||
#endif // HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
#endif // HOLD_ON_OTHER_KEY_PRESS_PER_KEY
|
||||||
|
|
||||||
#ifdef IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
__attribute__((weak)) bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
|
|
||||||
// Do not force the mod-tap key press to be handled as a modifier
|
|
||||||
// if any other key was pressed while the mod-tap key is held down.
|
|
||||||
// return true;
|
|
||||||
// Force the mod-tap key press to be handled as a modifier if any
|
|
||||||
// other key was pressed while the mod-tap key is held down.
|
|
||||||
// return false;
|
|
||||||
switch (keycode) {
|
|
||||||
default:
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // IGNORE_MOD_TAP_INTERRUPT_PER_KEY
|
|
||||||
|
|
||||||
#ifdef QUICK_TAP_TERM_PER_KEY
|
#ifdef QUICK_TAP_TERM_PER_KEY
|
||||||
__attribute__((weak)) uint16_t get_quick_tap_term(uint16_t keycode, keyrecord_t *record) {
|
__attribute__((weak)) uint16_t get_quick_tap_term(uint16_t keycode, keyrecord_t *record) {
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
|
Loading…
Reference in New Issue
Block a user