Add support for key override introspection. (#24120)
This commit is contained in:
19
docs/ChangeLog/PR24120.md
Normal file
19
docs/ChangeLog/PR24120.md
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
## Changes requiring user action
|
||||||
|
|
||||||
|
### Key Override Introspection
|
||||||
|
|
||||||
|
Changes were made to key overrides in order to hook them into the keymap introspection system.
|
||||||
|
|
||||||
|
Key override signature changed from:
|
||||||
|
|
||||||
|
```c
|
||||||
|
const key_override_t **key_overrides = (const key_override_t *[]){
|
||||||
|
```
|
||||||
|
|
||||||
|
to:
|
||||||
|
|
||||||
|
```c
|
||||||
|
const key_override_t *key_overrides[] = {
|
||||||
|
```
|
||||||
|
|
||||||
|
The list of key overrides now does not need to be `NULL`-terminated.
|
@ -14,7 +14,7 @@ You can use key overrides in a similar way to momentary layer/fn keys to activat
|
|||||||
|
|
||||||
To enable this feature, you need to add `KEY_OVERRIDE_ENABLE = yes` to your `rules.mk`.
|
To enable this feature, you need to add `KEY_OVERRIDE_ENABLE = yes` to your `rules.mk`.
|
||||||
|
|
||||||
Then, in your `keymap.c` file, you'll need to define the array `key_overrides`, which defines all key overrides to be used. Each override is a value of type `key_override_t`. The array `key_overrides` is `NULL`-terminated and contains pointers to `key_override_t` values (`const key_override_t **`).
|
Then, in your `keymap.c` file, you'll need to define the array `key_overrides`, which defines all key overrides to be used. Each override is a value of type `key_override_t`. The array `key_overrides`contains pointers to `key_override_t` values (`const key_override_t **`).
|
||||||
|
|
||||||
## Creating Key Overrides {#creating-key-overrides}
|
## Creating Key Overrides {#creating-key-overrides}
|
||||||
|
|
||||||
@ -42,9 +42,8 @@ This shows how the mentioned example of sending `delete` when `shift` + `backspa
|
|||||||
const key_override_t delete_key_override = ko_make_basic(MOD_MASK_SHIFT, KC_BSPC, KC_DEL);
|
const key_override_t delete_key_override = ko_make_basic(MOD_MASK_SHIFT, KC_BSPC, KC_DEL);
|
||||||
|
|
||||||
// This globally defines all key overrides to be used
|
// This globally defines all key overrides to be used
|
||||||
const key_override_t **key_overrides = (const key_override_t *[]){
|
const key_override_t *key_overrides[] = {
|
||||||
&delete_key_override,
|
&delete_key_override
|
||||||
NULL // Null terminate the array of overrides!
|
|
||||||
};
|
};
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -91,14 +90,13 @@ const key_override_t brightness_up_override = ko_make_with_layers_negmods_and_op
|
|||||||
const key_override_t brightness_down_override = ko_make_basic(MOD_MASK_CSA, KC_MPLY, KC_BRID);
|
const key_override_t brightness_down_override = ko_make_basic(MOD_MASK_CSA, KC_MPLY, KC_BRID);
|
||||||
|
|
||||||
// This globally defines all key overrides to be used
|
// This globally defines all key overrides to be used
|
||||||
const key_override_t **key_overrides = (const key_override_t *[]){
|
const key_override_t *key_overrides[] = {
|
||||||
&next_track_override,
|
&next_track_override,
|
||||||
&prev_track_override,
|
&prev_track_override,
|
||||||
&vol_up_override,
|
&vol_up_override,
|
||||||
&vol_down_override,
|
&vol_down_override,
|
||||||
&brightness_up_override,
|
&brightness_up_override,
|
||||||
&brightness_down_override,
|
&brightness_down_override
|
||||||
NULL
|
|
||||||
};
|
};
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -112,10 +110,9 @@ const key_override_t tilde_esc_override = ko_make_basic(MOD_MASK_SHIFT, KC_ESC,
|
|||||||
// GUI + esc = `
|
// GUI + esc = `
|
||||||
const key_override_t grave_esc_override = ko_make_basic(MOD_MASK_GUI, KC_ESC, KC_GRV);
|
const key_override_t grave_esc_override = ko_make_basic(MOD_MASK_GUI, KC_ESC, KC_GRV);
|
||||||
|
|
||||||
const key_override_t **key_overrides = (const key_override_t *[]){
|
const key_override_t *key_overrides[] = {
|
||||||
&tilde_esc_override,
|
&tilde_esc_override,
|
||||||
&grave_esc_override,
|
&grave_esc_override
|
||||||
NULL
|
|
||||||
};
|
};
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#endif // INTROSPECTION_KEYMAP_C
|
#endif // INTROSPECTION_KEYMAP_C
|
||||||
|
|
||||||
#include "keymap_introspection.h"
|
#include "keymap_introspection.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Key mapping
|
// Key mapping
|
||||||
@ -83,7 +84,7 @@ uint16_t keycode_at_dip_switch_map_location_raw(uint8_t switch_idx, bool on) {
|
|||||||
return KC_TRNS;
|
return KC_TRNS;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t keycode_at_dip_switch_map_location(uint8_t switch_idx, bool on) {
|
__attribute__((weak)) uint16_t keycode_at_dip_switch_map_location(uint8_t switch_idx, bool on) {
|
||||||
return keycode_at_dip_switch_map_location_raw(switch_idx, on);
|
return keycode_at_dip_switch_map_location_raw(switch_idx, on);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,13 +96,16 @@ uint16_t keycode_at_dip_switch_map_location(uint8_t switch_idx, bool on) {
|
|||||||
#if defined(COMBO_ENABLE)
|
#if defined(COMBO_ENABLE)
|
||||||
|
|
||||||
uint16_t combo_count_raw(void) {
|
uint16_t combo_count_raw(void) {
|
||||||
return sizeof(key_combos) / sizeof(combo_t);
|
return ARRAY_SIZE(key_combos);
|
||||||
}
|
}
|
||||||
__attribute__((weak)) uint16_t combo_count(void) {
|
__attribute__((weak)) uint16_t combo_count(void) {
|
||||||
return combo_count_raw();
|
return combo_count_raw();
|
||||||
}
|
}
|
||||||
|
|
||||||
combo_t* combo_get_raw(uint16_t combo_idx) {
|
combo_t* combo_get_raw(uint16_t combo_idx) {
|
||||||
|
if (combo_idx >= combo_count_raw()) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
return &key_combos[combo_idx];
|
return &key_combos[combo_idx];
|
||||||
}
|
}
|
||||||
__attribute__((weak)) combo_t* combo_get(uint16_t combo_idx) {
|
__attribute__((weak)) combo_t* combo_get(uint16_t combo_idx) {
|
||||||
@ -116,19 +120,48 @@ __attribute__((weak)) combo_t* combo_get(uint16_t combo_idx) {
|
|||||||
#if defined(TAP_DANCE_ENABLE)
|
#if defined(TAP_DANCE_ENABLE)
|
||||||
|
|
||||||
uint16_t tap_dance_count_raw(void) {
|
uint16_t tap_dance_count_raw(void) {
|
||||||
return sizeof(tap_dance_actions) / sizeof(tap_dance_action_t);
|
return ARRAY_SIZE(tap_dance_actions);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t tap_dance_count(void) {
|
__attribute__((weak)) uint16_t tap_dance_count(void) {
|
||||||
return tap_dance_count_raw();
|
return tap_dance_count_raw();
|
||||||
}
|
}
|
||||||
|
|
||||||
tap_dance_action_t* tap_dance_get_raw(uint16_t tap_dance_idx) {
|
tap_dance_action_t* tap_dance_get_raw(uint16_t tap_dance_idx) {
|
||||||
|
if (tap_dance_idx >= tap_dance_count_raw()) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
return &tap_dance_actions[tap_dance_idx];
|
return &tap_dance_actions[tap_dance_idx];
|
||||||
}
|
}
|
||||||
|
|
||||||
tap_dance_action_t* tap_dance_get(uint16_t tap_dance_idx) {
|
__attribute__((weak)) tap_dance_action_t* tap_dance_get(uint16_t tap_dance_idx) {
|
||||||
return tap_dance_get_raw(tap_dance_idx);
|
return tap_dance_get_raw(tap_dance_idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // defined(TAP_DANCE_ENABLE)
|
#endif // defined(TAP_DANCE_ENABLE)
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Key Overrides
|
||||||
|
|
||||||
|
#if defined(KEY_OVERRIDE_ENABLE)
|
||||||
|
|
||||||
|
uint16_t key_override_count_raw(void) {
|
||||||
|
return ARRAY_SIZE(key_overrides);
|
||||||
|
}
|
||||||
|
|
||||||
|
__attribute__((weak)) uint16_t key_override_count(void) {
|
||||||
|
return key_override_count_raw();
|
||||||
|
}
|
||||||
|
|
||||||
|
const key_override_t* key_override_get_raw(uint16_t key_override_idx) {
|
||||||
|
if (key_override_idx >= key_override_count_raw()) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return key_overrides[key_override_idx];
|
||||||
|
}
|
||||||
|
|
||||||
|
__attribute__((weak)) const key_override_t* key_override_get(uint16_t key_override_idx) {
|
||||||
|
return key_override_get_raw(key_override_idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // defined(KEY_OVERRIDE_ENABLE)
|
||||||
|
@ -88,3 +88,24 @@ tap_dance_action_t* tap_dance_get_raw(uint16_t tap_dance_idx);
|
|||||||
tap_dance_action_t* tap_dance_get(uint16_t tap_dance_idx);
|
tap_dance_action_t* tap_dance_get(uint16_t tap_dance_idx);
|
||||||
|
|
||||||
#endif // defined(TAP_DANCE_ENABLE)
|
#endif // defined(TAP_DANCE_ENABLE)
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Key Overrides
|
||||||
|
|
||||||
|
#if defined(KEY_OVERRIDE_ENABLE)
|
||||||
|
|
||||||
|
// Forward declaration of key_override_t so we don't need to deal with header reordering
|
||||||
|
struct key_override_t;
|
||||||
|
typedef struct key_override_t key_override_t;
|
||||||
|
|
||||||
|
// Get the number of key overrides defined in the user's keymap, stored in firmware rather than any other persistent storage
|
||||||
|
uint16_t key_override_count_raw(void);
|
||||||
|
// Get the number of key overrides defined in the user's keymap, potentially stored dynamically
|
||||||
|
uint16_t key_override_count(void);
|
||||||
|
|
||||||
|
// Get the key override definitions, stored in firmware rather than any other persistent storage
|
||||||
|
const key_override_t* key_override_get_raw(uint16_t key_override_idx);
|
||||||
|
// Get the key override definitions, potentially stored dynamically
|
||||||
|
const key_override_t* key_override_get(uint16_t key_override_idx);
|
||||||
|
|
||||||
|
#endif // defined(KEY_OVERRIDE_ENABLE)
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "action_util.h"
|
#include "action_util.h"
|
||||||
#include "quantum.h"
|
#include "quantum.h"
|
||||||
#include "quantum_keycodes.h"
|
#include "quantum_keycodes.h"
|
||||||
|
#include "keymap_introspection.h"
|
||||||
|
|
||||||
#ifndef KEY_OVERRIDE_REPEAT_DELAY
|
#ifndef KEY_OVERRIDE_REPEAT_DELAY
|
||||||
# define KEY_OVERRIDE_REPEAT_DELAY 500
|
# define KEY_OVERRIDE_REPEAT_DELAY 500
|
||||||
@ -83,9 +84,6 @@ static uint16_t deferred_register = 0;
|
|||||||
// TODO: in future maybe save in EEPROM?
|
// TODO: in future maybe save in EEPROM?
|
||||||
static bool enabled = true;
|
static bool enabled = true;
|
||||||
|
|
||||||
// Public variables
|
|
||||||
__attribute__((weak)) const key_override_t **key_overrides = NULL;
|
|
||||||
|
|
||||||
// Forward decls
|
// Forward decls
|
||||||
static const key_override_t *clear_active_override(const bool allow_reregister);
|
static const key_override_t *clear_active_override(const bool allow_reregister);
|
||||||
|
|
||||||
@ -247,12 +245,12 @@ static bool check_activation_event(const key_override_t *override, const bool ke
|
|||||||
|
|
||||||
/** Iterates through the list of key overrides and tries activating each, until it finds one that activates or reaches the end of overrides. Returns true if the key action for `keycode` should be sent */
|
/** Iterates through the list of key overrides and tries activating each, until it finds one that activates or reaches the end of overrides. Returns true if the key action for `keycode` should be sent */
|
||||||
static bool try_activating_override(const uint16_t keycode, const uint8_t layer, const bool key_down, const bool is_mod, const uint8_t active_mods, bool *activated) {
|
static bool try_activating_override(const uint16_t keycode, const uint8_t layer, const bool key_down, const bool is_mod, const uint8_t active_mods, bool *activated) {
|
||||||
if (key_overrides == NULL) {
|
if (key_override_count() == 0) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (uint8_t i = 0;; i++) {
|
for (uint8_t i = 0; i < key_override_count(); i++) {
|
||||||
const key_override_t *const override = key_overrides[i];
|
const key_override_t *const override = key_override_get(i);
|
||||||
|
|
||||||
// End of array
|
// End of array
|
||||||
if (override == NULL) {
|
if (override == NULL) {
|
||||||
|
@ -55,7 +55,7 @@ typedef enum {
|
|||||||
} ko_option_t;
|
} ko_option_t;
|
||||||
|
|
||||||
/** Defines a single key override */
|
/** Defines a single key override */
|
||||||
typedef struct {
|
typedef struct key_override_t {
|
||||||
// The non-modifier keycode that triggers the override. This keycode, and the necessary modifiers (trigger_mods) must be pressed to activate this override. Set this to the keycode of the key that should activate the override. Set to KC_NO to require only the necessary modifiers to be pressed and no non-modifier.
|
// The non-modifier keycode that triggers the override. This keycode, and the necessary modifiers (trigger_mods) must be pressed to activate this override. Set this to the keycode of the key that should activate the override. Set to KC_NO to require only the necessary modifiers to be pressed and no non-modifier.
|
||||||
uint16_t trigger;
|
uint16_t trigger;
|
||||||
|
|
||||||
@ -87,9 +87,6 @@ typedef struct {
|
|||||||
bool *enabled;
|
bool *enabled;
|
||||||
} key_override_t;
|
} key_override_t;
|
||||||
|
|
||||||
/** Define this as a null-terminated array of pointers to key overrides. These key overrides will be used by qmk. */
|
|
||||||
extern const key_override_t **key_overrides;
|
|
||||||
|
|
||||||
/** Turns key overrides on */
|
/** Turns key overrides on */
|
||||||
void key_override_on(void);
|
void key_override_on(void);
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user