2019-02-19 01:28:50 +01:00
|
|
|
# Keys
|
|
|
|
|
|
|
|
> NOTE: This is not a lookup table of key objects provided by KMK. That listing
|
2022-09-11 21:32:11 +02:00
|
|
|
> can be found in [`keycodes.md`](/docs/en/keycodes.md). It's probably worth a look at the raw source if
|
2022-05-20 15:26:32 +02:00
|
|
|
> you're stumped: [`kmk/keys.py`](/kmk/keys.py).
|
2019-02-19 01:28:50 +01:00
|
|
|
|
2019-07-13 02:11:36 +02:00
|
|
|
This is a bunch of documentation about how a physical keypress translates to
|
2019-02-19 01:28:50 +01:00
|
|
|
events (and the lifecycle of said events) in KMK. It's somewhat technical, but
|
|
|
|
if you're looking to extend your keyboard's functionality with extra code,
|
|
|
|
you'll need at least some of this technical knowledge.
|
|
|
|
|
|
|
|
The first few steps in the process aren't all that interesting for most
|
|
|
|
workflows, which is why they're buried deep in KMK: we scan a bunch of GPIO
|
|
|
|
lanes (about as quickly as CircuitPython will let us) to see where, in a matrix
|
|
|
|
of keys, a key has been pressed. The technical details about this process [are
|
|
|
|
probably best left to
|
|
|
|
Wikipedia](https://en.wikipedia.org/wiki/Keyboard_matrix_circuit). Then, we scan
|
|
|
|
through the defined keymap, finding the first valid key at this index based on
|
|
|
|
the stack of currently active layers (this logic, if you want to read through
|
2022-05-20 15:26:32 +02:00
|
|
|
the code, is in [`kmk/kmk_keyboard.py`](/kmk/kmk_keyboard.py), method `_find_key_in_map`).
|
2019-02-19 01:28:50 +01:00
|
|
|
|
|
|
|
The next few steps are the interesting part, but to understand them, we need to
|
2022-05-20 15:26:32 +02:00
|
|
|
understand a bit about what a `Key` object is (found in [`kmk/keys.py`](/kmk/keys.py)). `Key`
|
2019-02-19 01:28:50 +01:00
|
|
|
objects have a few core pieces of information:
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- Their `code`, which can be any integer. Integers below
|
2019-02-19 01:28:50 +01:00
|
|
|
`FIRST_KMK_INTERNAL_KEY` are sent through to the HID stack (and thus the
|
|
|
|
computer, which will translate that integer to something meaningful - for
|
|
|
|
example, `code=4` becomes `a` on a US QWERTY/Dvorak keyboard).
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- Their attached modifiers (to implement things like shifted keys or `KC.HYPR`,
|
2019-02-19 01:28:50 +01:00
|
|
|
which are single key presses sending along more than one key in a single HID
|
|
|
|
report. This is a distinct concept from Sequences, which are a KMK feature
|
2022-09-11 21:32:11 +02:00
|
|
|
documented in [`sequences.md`](/docs/en/sequences.md)). For almost all purposes outside of KMK core,
|
2019-02-19 01:28:50 +01:00
|
|
|
this field should be ignored - it can be safely populated through far more
|
|
|
|
sane means than futzing with it by hand.
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- Some data on whether the key should actually be pressed or released - this is
|
2019-02-19 01:28:50 +01:00
|
|
|
mostly an implementation detail of how Sequences work, where, for example,
|
|
|
|
`KC.RALT` may need to be held down for the entirety of a sequence, rather than
|
2022-04-25 19:38:17 +02:00
|
|
|
being released immediately before moving to the next character. Usually end
|
2019-02-19 01:28:50 +01:00
|
|
|
users shouldn't need to mess with this, but the fields are called `no_press`
|
|
|
|
and `no_release` and are referenced in a few places in the codebase if you
|
|
|
|
need examples.
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- Handlers for "press" (sometimes known as "keydown") and "release" (sometimes
|
2019-02-19 01:28:50 +01:00
|
|
|
known as "keyup") events. KMK provides handlers for standard keyboard
|
|
|
|
functions and some special override keys (like `KC.GESC`, which is an enhanced
|
2022-05-20 15:26:32 +02:00
|
|
|
form of existing ANSI keys) in [`kmk/handlers/stock.py`](/kmk/handlers/stock.py), for layer switching in
|
|
|
|
[`kmk/modules/layers.py`](/kmk/modules/layers.py), and for everything related to Sequences (see
|
2022-09-11 21:32:11 +02:00
|
|
|
[`sequences.md`](/docs/en/sequences.md) again) in [`kmk/handlers/sequences.py`](/kmk/handlers/sequences.py). We'll discuss these more
|
2019-02-19 01:28:50 +01:00
|
|
|
shortly.
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- Optional callbacks to be run before and/or after the above handlers. More on
|
2019-02-19 01:28:50 +01:00
|
|
|
that soon.
|
|
|
|
|
2022-04-25 19:38:17 +02:00
|
|
|
- A generic `meta` field, which is most commonly used for "argumented" keys -
|
2019-02-19 01:28:50 +01:00
|
|
|
objects in the `KC` object which are actually functions that return `Key`
|
|
|
|
instances, which often need to access the arguments passed into the "outer"
|
|
|
|
function. Many of these examples are related to layer switching - for example,
|
|
|
|
`KC.MO` is implemented as an argumented key - when the user adds `KC.MO(1)` to
|
|
|
|
their keymap, the function call returns a `Key` object with `meta` set to an
|
|
|
|
object containing `layer` and `kc` properties, for example. There's other uses
|
2022-05-20 15:26:32 +02:00
|
|
|
for `meta`, and examples can be found in [`kmk/types.py`](/kmk/types.py)
|
2019-02-19 01:28:50 +01:00
|
|
|
|
|
|
|
`Key` objects can also be chained together by calling them! To create a key
|
|
|
|
which holds Control and Shift simultaneously, we can simply do:
|
|
|
|
|
|
|
|
```python
|
|
|
|
CTRLSHFT = KC.LCTL(KC.LSFT)
|
|
|
|
|
|
|
|
keyboard.keymap = [ ... CTRLSHFT ... ]
|
|
|
|
```
|
|
|
|
|
|
|
|
When a key is pressed and we've pulled a `Key` object out of the keymap, the
|
|
|
|
following will happen:
|
|
|
|
|
|
|
|
- Pre-press callbacks will be run in the order they were assigned, with their
|
|
|
|
return values discarded (unless the user attached these, they will almost
|
|
|
|
never exist)
|
|
|
|
- The assigned press handler will be run (most commonly, this is provided by
|
|
|
|
KMK)
|
|
|
|
- Post-press callbacks will be run in the order they were assigned, with their
|
|
|
|
return values discarded (unless the user attached these, they will almost
|
|
|
|
never exist)
|
|
|
|
|
|
|
|
These same steps are run for when a key is released.
|
|
|
|
|
|
|
|
_So now... what's a handler, and what's a pre/post callback?!_
|
|
|
|
|
2022-04-24 19:30:10 +02:00
|
|
|
All of these serve roughly the same purpose: to _do something_ with the key's
|
2019-02-19 01:28:50 +01:00
|
|
|
data, or to fire off side effects. Most handlers are provided by KMK internally
|
|
|
|
and modify the `InternalState` in some way - adding the key to the HID queue,
|
|
|
|
changing layers, etc. The pre/post handlers are designed to allow functionality
|
|
|
|
to be bolted on at these points in the event flow without having to reimplement
|
|
|
|
(or import and manually call) the internal handlers.
|
|
|
|
|
|
|
|
All of these methods take the same arguments, and for this, I'll lift a
|
|
|
|
docstring straight out of the source:
|
|
|
|
|
|
|
|
> Receives the following:
|
2022-04-25 18:51:00 +02:00
|
|
|
>
|
2019-02-19 01:28:50 +01:00
|
|
|
> - self (this Key instance)
|
|
|
|
> - state (the current InternalState)
|
|
|
|
> - KC (the global KC lookup table, for convenience)
|
|
|
|
> - `coord_int` (an internal integer representation of the matrix coordinate
|
|
|
|
> for the pressed key - this is likely not useful to end users, but is
|
|
|
|
> provided for consistency with the internal handlers)
|
|
|
|
> - `coord_raw` (an X,Y tuple of the matrix coordinate - also likely not useful)
|
2022-04-25 18:51:00 +02:00
|
|
|
>
|
2019-02-19 01:28:50 +01:00
|
|
|
> The return value of the provided callback is discarded. Exceptions are _not_
|
|
|
|
> caught, and will likely crash KMK if not handled within your function.
|
2022-04-25 18:51:00 +02:00
|
|
|
>
|
2019-02-19 01:28:50 +01:00
|
|
|
> These handlers are run in attachment order: handlers provided by earlier
|
|
|
|
> calls of this method will be executed before those provided by later calls.
|
|
|
|
|
|
|
|
This means if you want to add things like underglow/LED support, or have a
|
|
|
|
button that triggers your GSM modem to call someone, or whatever else you can
|
|
|
|
hack up in CircuitPython, which also retaining layer-switching abilities or
|
|
|
|
whatever the stock handler is, you're covered. This also means you can add
|
|
|
|
completely new functionality to KMK by writing your own handler.
|
|
|
|
|
2022-02-28 02:38:19 +01:00
|
|
|
Here's an example of an after_press_handler to change the RGB lights with a layer change:
|
2022-04-25 19:38:17 +02:00
|
|
|
|
2022-02-28 02:38:19 +01:00
|
|
|
```python
|
|
|
|
LOWER = KC.DF(LYR_LOWER) #Set layer to LOWER
|
|
|
|
|
|
|
|
def low_lights(key, keyboard, *args):
|
|
|
|
print('Lower Layer') #serial feedback
|
|
|
|
keyboard.pixels.set_hsv_fill(0, 100, 255) #RGB extension call to set (H,S,V) values
|
|
|
|
|
|
|
|
LOWER.after_press_handler(low_lights) #call the key with the after_press_handler
|
|
|
|
```
|
2022-04-25 19:38:17 +02:00
|
|
|
|
2019-02-19 01:28:50 +01:00
|
|
|
Here's an example of a lifecycle hook to print a giant Shrek ASCII art. It
|
|
|
|
doesn't care about any of the arguments passed into it, because it has no
|
|
|
|
intentions of modifying the internal state. It is purely a [side
|
2022-04-25 19:38:17 +02:00
|
|
|
effect](<https://en.wikipedia.org/wiki/Side_effect_(computer_science)>) run every
|
2019-02-19 01:28:50 +01:00
|
|
|
time Left Alt is pressed:
|
|
|
|
|
|
|
|
```python
|
|
|
|
def shrek(*args, **kwargs):
|
|
|
|
print('⢀⡴⠑⡄⠀⠀⠀⠀⠀⠀⠀⣀⣀⣤⣤⣤⣀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠸⡇⠀⠿⡀⠀⠀⠀⣀⡴⢿⣿⣿⣿⣿⣿⣿⣿⣷⣦⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠑⢄⣠⠾⠁⣀⣄⡈⠙⣿⣿⣿⣿⣿⣿⣿⣿⣆⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⢀⡀⠁⠀⠀⠈⠙⠛⠂⠈⣿⣿⣿⣿⣿⠿⡿⢿⣆⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⢀⡾⣁⣀⠀⠴⠂⠙⣗⡀⠀⢻⣿⣿⠭⢤⣴⣦⣤⣹⠀⠀⠀⢀⢴⣶⣆')
|
|
|
|
print('⠀⠀⢀⣾⣿⣿⣿⣷⣮⣽⣾⣿⣥⣴⣿⣿⡿⢂⠔⢚⡿⢿⣿⣦⣴⣾⠁⠸⣼⡿')
|
|
|
|
print('⠀⢀⡞⠁⠙⠻⠿⠟⠉⠀⠛⢹⣿⣿⣿⣿⣿⣌⢤⣼⣿⣾⣿⡟⠉⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⣾⣷⣶⠇⠀⠀⣤⣄⣀⡀⠈⠻⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠉⠈⠉⠀⠀⢦⡈⢻⣿⣿⣿⣶⣶⣶⣶⣤⣽⡹⣿⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠉⠲⣽⡻⢿⣿⣿⣿⣿⣿⣿⣷⣜⣿⣿⣿⡇⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠀⢸⣿⣿⣷⣶⣮⣭⣽⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⣀⣀⣈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠇⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠃⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀')
|
|
|
|
print('⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠛⠻⠿⠿⠿⠿⠛⠉')
|
2022-04-25 18:51:00 +02:00
|
|
|
|
2021-12-07 00:31:42 +01:00
|
|
|
return False #Returning True will follow thru the normal handlers sending the ALT key to the OS
|
2019-02-19 01:28:50 +01:00
|
|
|
KC.LALT.before_press_handler(shrek)
|
|
|
|
```
|
|
|
|
|
|
|
|
You can also copy a key without any pre/post handlers attached with `.clone()`,
|
|
|
|
so for example, if I've already added Shrek to my `LALT` but want a Shrek-less
|
|
|
|
`LALT` key elsewhere in my keymap, I can just clone it, and the new key won't
|
|
|
|
have my handlers attached:
|
|
|
|
|
|
|
|
```python
|
|
|
|
SHREKLESS_ALT = KC.LALT.clone()
|
|
|
|
```
|
2022-04-25 18:51:00 +02:00
|
|
|
|
|
|
|
You can also refer to a key by index:
|
2022-04-25 19:38:17 +02:00
|
|
|
|
|
|
|
- `KC['A']`
|
|
|
|
- `KC['NO']`
|
|
|
|
- `KC['LALT']`
|
|
|
|
|
|
|
|
Or the `KC.get` function which has an optional default argument, which will
|
|
|
|
be returned if the key is not found (`default=None` unless otherwise specified):
|
|
|
|
|
|
|
|
- `KC.get('A')`
|
|
|
|
- `KC.get('NO', None)`
|
|
|
|
- `KC.get('NOT DEFINED', KC.RALT)`
|
|
|
|
|
|
|
|
Key names are case-sensitive. `KC['NO']` is different from `KC['no']`. It is recommended
|
|
|
|
that names are normally UPPER_CASE. The exception to this are alpha keys; `KC['A']` and
|
|
|
|
`KC['a']` will both return the same, unshifted, key.
|