aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSergey Vlasov <sigprof@gmail.com>2021-08-07 02:16:26 +0300
committerGitHub <noreply@github.com>2021-08-07 09:16:26 +1000
commit610035dce8f4f7010779e61e1cb590eebc8cea46 (patch)
treecd36a37e468c90e64c54fc5371205e186b1be08d
parent13b94b468d728f43b7edcd91224ea217851b80bc (diff)
downloadqmk_firmware-610035dce8f4f7010779e61e1cb590eebc8cea46.tar.gz
qmk_firmware-610035dce8f4f7010779e61e1cb590eebc8cea46.zip
Add HOLD_ON_OTHER_KEY_PRESS option for dual-role keys (#9404)
* Add HOLD_ON_OTHER_KEY_PRESS option for dual-role keys Implement an additional option for dual-role keys which converts the dual-role key press into a hold action immediately when another key is pressed (this is different from the existing PERMISSIVE_HOLD option, which selects the hold action when another key is tapped (pressed and then released) while the dual-role key is pressed). The Mod-Tap keys already behave in a similar way, unless the IGNORE_MOD_TAP_INTERRUPT option is enabled (but with some additional delays); the added option makes this behavior available for all other kinds of dual-role keys. * [Docs] Update tap-hold docs for HOLD_ON_OTHER_KEY_PRESS Document the newly added HOLD_ON_OTHER_KEY_PRESS option and update the documentation for closely related options (PERMISSIVE_HOLD and IGNORE_MOD_TAP_INTERRUPT). Use Layer Tap instead of Mod Tap in examples for PERMISSIVE_HOLD and HOLD_ON_OTHER_KEY_PRESS, because the effect of using these options with Mod Tap keys is mostly invisible without IGNORE_MOD_TAP_INTERRUPT. Add comments before return statements in sample implementations of `get_ignore_mod_tap_interrupt()`, `get_hold_on_other_key_press()` and `get_permissive_hold()`. Thanks to @Erovia and @precondition for comments and suggestions to improve the documentation.
-rw-r--r--docs/tap_hold.md108
-rw-r--r--tmk_core/common/action_tapping.c17
2 files changed, 106 insertions, 19 deletions
diff --git a/docs/tap_hold.md b/docs/tap_hold.md
index 085bbde16..71bff30ba 100644
--- a/docs/tap_hold.md
+++ b/docs/tap_hold.md
@@ -36,29 +36,49 @@ uint16_t get_tapping_term(uint16_t keycode, keyrecord_t *record) {
36} 36}
37``` 37```
38 38
39## Tap-Or-Hold Decision Modes
39 40
40## Permissive Hold 41The 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:
41 42
42As of [PR#1359](https://github.com/qmk/qmk_firmware/pull/1359/), there is a new `config.h` option: 431. 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.
44
452. 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.
46
473. The “hold on other key press” mode, in addition to the default behavior, immediately selects the hold action when another key is pressed while the dual-role key is held down, even if this happens earlier than the tapping term.
48
49Note 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.
50
51### Permissive Hold
52
53The “permissive hold” mode can be enabled for all dual-role keys by adding the corresponding option to `config.h`:
43 54
44```c 55```c
45#define PERMISSIVE_HOLD 56#define PERMISSIVE_HOLD
46``` 57```
47 58
48This makes tap and hold keys (like Mod Tap) work better for fast typists, or for high `TAPPING_TERM` settings. 59This makes tap and hold keys (like Layer Tap) work better for fast typists, or for high `TAPPING_TERM` settings.
49 60
50If you press a Mod Tap key, tap another key (press and release) and then release the Mod Tap key, all within the tapping term, it will output the tapping function for both keys. 61If you press a dual-role key, tap another key (press and release) and then release the dual-role key, all within the tapping term, by default the dual-role key will perform its tap action. If the `PERMISSIVE_HOLD` option is enabled, the dual-role key will perform its hold action instead.
51 62
52For Instance: 63An example of a sequence which is affected by the “permissive hold” mode:
53 64
54- `SFT_T(KC_A)` Down 65- `LT(2, KC_A)` Down
55- `KC_X` Down 66- `KC_L` Down (the `L` key is also mapped to `KC_RGHT` on layer 2)
56- `KC_X` Up 67- `KC_L` Up
57- `SFT_T(KC_A)` Up 68- `LT(2, KC_A)` Up
69
70Normally, 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).
71
72However, this slightly different sequence will not be affected by the “permissive hold” mode:
58 73
59Normally, if you do all this within the `TAPPING_TERM` (default: 200ms) this will be registered as `ax` by the firmware and host system. With permissive hold enabled, this modifies how this is handled by considering the Mod Tap keys as a Mod if another key is tapped, and would registered as `X` (`SHIFT`+`x`). 74- `LT(2, KC_A)` Down
75- `KC_L` Down (the `L` key is also mapped to `KC_RGHT` on layer 2)
76- `LT(2, KC_A)` Up
77- `KC_L` Up
60 78
61?> If you have `Ignore Mod Tap Interrupt` enabled, as well, this will modify how both work. The regular key has the modifier added if the first key is released first or if both keys are held longer than the `TAPPING_TERM`. 79In 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.
80
81?> 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.
62 82
63For more granular control of this feature, you can add the following to your `config.h`: 83For more granular control of this feature, you can add the following to your `config.h`:
64 84
@@ -72,13 +92,60 @@ You can then add the following function to your keymap:
72bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) { 92bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) {
73 switch (keycode) { 93 switch (keycode) {
74 case LT(1, KC_BSPC): 94 case LT(1, KC_BSPC):
95 // Immediately select the hold action when another key is tapped.
96 return true;
97 default:
98 // Do not select the hold action when another key is tapped.
99 return false;
100 }
101}
102```
103
104### Hold On Other Key Press
105
106The “hold on other key press” mode can be enabled for all dual-role keys by adding the corresponding option to `config.h`:
107
108```c
109#define HOLD_ON_OTHER_KEY_PRESS
110```
111
112This mode makes tap and hold keys (like Layer Tap) work better for fast typists, or for high `TAPPING_TERM` settings. Compared to the “permissive hold” mode, this mode selects the hold action in more cases.
113
114If you press a dual-role key, press another key, and then release the dual-role key, all within the tapping term, by default the dual-role key will perform its tap action. If the `HOLD_ON_OTHER_KEY_PRESS` option is enabled, the dual-role key will perform its hold action instead.
115
116An example of a sequence which is affected by the “hold on other key press” mode, but not by the “permissive hold” mode:
117
118- `LT(2, KC_A)` Down
119- `KC_L` Down (the `L` key is also mapped to `KC_RGHT` on layer 2)
120- `LT(2, KC_A)` Up
121- `KC_L` Up
122
123Normally, 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).
124
125?> 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.
126
127For more granular control of this feature, you can add the following to your `config.h`:
128
129```c
130#define HOLD_ON_OTHER_KEY_PRESS_PER_KEY
131```
132
133You can then add the following function to your keymap:
134
135```c
136bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) {
137 switch (keycode) {
138 case LT(1, KC_BSPC):
139 // Immediately select the hold action when another key is pressed.
75 return true; 140 return true;
76 default: 141 default:
142 // Do not select the hold action when another key is pressed.
77 return false; 143 return false;
78 } 144 }
79} 145}
80``` 146```
81 147
148
82## Ignore Mod Tap Interrupt 149## Ignore Mod Tap Interrupt
83 150
84To enable this setting, add this to your `config.h`: 151To enable this setting, add this to your `config.h`:
@@ -87,23 +154,22 @@ To enable this setting, add this to your `config.h`:
87#define IGNORE_MOD_TAP_INTERRUPT 154#define IGNORE_MOD_TAP_INTERRUPT
88``` 155```
89 156
90Similar to Permissive Hold, this alters how the firmware processes inputs for fast typists. If you press a Mod Tap key, press another key, release the Mod Tap key, and then release the normal key, it would normally output the Mod plus the normal key, even if pressed within the `TAPPING_TERM`. This may not be desirable for rolling combo keys, or for fast typists who have a Mod Tap on a frequently used key (`RCTL_T(KC_QUOT)`, for example). 157?> This option affects only the Mod Tap keys; it does not affect other dual-role keys such as Layer Tap.
158
159By default the tap-or-hold decision for Mod Tap keys strongly prefers the hold action. If you press a Mod Tap key, then press another key while still holding the Mod Tap key down, the Mod Tap press will be handled as a modifier hold even if the Mod Tap key is then released within the tapping term, and irrespective of the order in which those keys are released. Using options such as `PERMISSIVE_HOLD` or `HOLD_ON_OTHER_KEY_PRESS` will not affect the functionality of Mod Tap keys in a major way (these options would still affect the delay until the common code for dual-role keys finishes its tap-or-hold decision, but then the special code for Mod Tap keys will override the result of that decision and choose the hold action if another key was pressed). In fact, by default the tap-or-hold decision for Mod Tap keys is done in the same way as if the `HOLD_ON_OTHER_KEY_PRESS` option was enabled, but without the decreased delay provided by `HOLD_ON_OTHER_KEY_PRESS`.
91 160
92Setting `Ignore Mod Tap Interrupt` requires holding both keys for the `TAPPING_TERM` to trigger the hold function (the mod). 161If the `IGNORE_MOD_TAP_INTERRUPT` option is enabled, Mod Tap keys are no longer treated as a special case, and their behavior will match the behavior of other dual-role keys such as Layer Tap. Then the behavior of Mod Tap keys can be further tuned using other options such as `PERMISSIVE_HOLD` or `HOLD_ON_OTHER_KEY_PRESS`.
93 162
94For Instance: 163An example of a sequence which will be affected by the `IGNORE_MOD_TAP_INTERRUPT` option (assuming that options like `PERMISSIVE_HOLD` or `HOLD_ON_OTHER_KEY_PRESS` are not enabled):
95 164
96- `SFT_T(KC_A)` Down 165- `SFT_T(KC_A)` Down
97- `KC_X` Down 166- `KC_X` Down
98- `SFT_T(KC_A)` Up 167- `SFT_T(KC_A)` Up
99- `KC_X` Up 168- `KC_X` Up
100 169
101Normally, this would send a capital `X` (`SHIFT`+`x`), or, Mod + key. With `Ignore Mod Tap Interrupt` enabled, holding both keys are required for the `TAPPING_TERM` to register the hold action. A quick tap will output `ax` in this case, while a hold on both will still output capital `X` (`SHIFT`+`x`). 170Normally, this would send a capital `X` (`SHIFT`+`x`), even if the sequence is performed faster than the `TAPPING_TERM`. However, if the `IGNORE_MOD_TAP_INTERRUPT` option is enabled, the `SFT_T(KC_A)` key must be held longer than the `TAPPING_TERM` to register the hold action. A quick tap will output `ax` in this case, while a hold will still output a capital `X` (`SHIFT`+`x`).
102
103
104?> __Note__: This only concerns modifiers and not layer switching keys.
105 171
106?> If you have `Permissive Hold` enabled, as well, this will modify how both work. The regular key has the modifier added if the first key is released first or if both keys are held longer than the `TAPPING_TERM`. 172However, if the `HOLD_ON_OTHER_KEY_PRESS` option is enabled in addition to `IGNORE_MOD_TAP_INTERRUPT`, the above sequence will again send a capital `X` (`SHIFT`+`x`) even if performed faster that the `TAPPING_TERM`. The difference from the default configuration is that by default the host will receive the key events only after the `SFT_T(KC_A)` key is released, but with the `HOLD_ON_OTHER_KEY_PRESS` option the host will start receiving key events when the `KC_X` key is pressed.
107 173
108For more granular control of this feature, you can add the following to your `config.h`: 174For more granular control of this feature, you can add the following to your `config.h`:
109 175
@@ -117,8 +183,12 @@ You can then add the following function to your keymap:
117bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) { 183bool get_ignore_mod_tap_interrupt(uint16_t keycode, keyrecord_t *record) {
118 switch (keycode) { 184 switch (keycode) {
119 case SFT_T(KC_SPC): 185 case SFT_T(KC_SPC):
186 // Do not force the mod-tap key press to be handled as a modifier
187 // if any other key was pressed while the mod-tap key is held down.
120 return true; 188 return true;
121 default: 189 default:
190 // Force the mod-tap key press to be handled as a modifier if any
191 // other key was pressed while the mod-tap key is held down.
122 return false; 192 return false;
123 } 193 }
124} 194}
diff --git a/tmk_core/common/action_tapping.c b/tmk_core/common/action_tapping.c
index 1701ae471..36839f9fa 100644
--- a/tmk_core/common/action_tapping.c
+++ b/tmk_core/common/action_tapping.c
@@ -40,6 +40,10 @@ __attribute__((weak)) bool get_tapping_force_hold(uint16_t keycode, keyrecord_t
40__attribute__((weak)) bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) { return false; } 40__attribute__((weak)) bool get_permissive_hold(uint16_t keycode, keyrecord_t *record) { return false; }
41# endif 41# endif
42 42
43# ifdef HOLD_ON_OTHER_KEY_PRESS_PER_KEY
44__attribute__((weak)) bool get_hold_on_other_key_press(uint16_t keycode, keyrecord_t *record) { return false; }
45# endif
46
43static keyrecord_t tapping_key = {}; 47static keyrecord_t tapping_key = {};
44static keyrecord_t waiting_buffer[WAITING_BUFFER_SIZE] = {}; 48static keyrecord_t waiting_buffer[WAITING_BUFFER_SIZE] = {};
45static uint8_t waiting_buffer_head = 0; 49static uint8_t waiting_buffer_head = 0;
@@ -175,6 +179,19 @@ bool process_tapping(keyrecord_t *keyp) {
175 // set interrupted flag when other key preesed during tapping 179 // set interrupted flag when other key preesed during tapping
176 if (event.pressed) { 180 if (event.pressed) {
177 tapping_key.tap.interrupted = true; 181 tapping_key.tap.interrupted = true;
182# if defined(HOLD_ON_OTHER_KEY_PRESS) || defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
183# if defined(HOLD_ON_OTHER_KEY_PRESS_PER_KEY)
184 if (get_hold_on_other_key_press(get_record_keycode(&tapping_key, false), keyp))
185# endif
186 {
187 debug("Tapping: End. No tap. Interfered by pressed key\n");
188 process_record(&tapping_key);
189 tapping_key = (keyrecord_t){};
190 debug_tapping_key();
191 // enqueue
192 return false;
193 }
194# endif
178 } 195 }
179 // enqueue 196 // enqueue
180 return false; 197 return false;