#keyboard-firmware #key #keyboard-layout #usb #event-stream #send #traits

nightly keytokey

A keyboard firmware library turning event streams of key presses into reports for USB

5 releases (breaking)

0.5.0 Sep 23, 2024
0.4.0 Sep 19, 2024
0.3.0 Jul 8, 2020
0.2.0 Aug 14, 2019
0.1.0 Aug 5, 2019

#705 in Hardware support

MIT license

210KB
4.5K SLoC

KeyToKey

KeyToKey (K2K) is a Rust libary for building keyboard firmwares.

Basically, a keyboard firmware a) reads key-presses, b) translates them c) and outputs the results to a computer.

KeyToKey's role is in b - it takes a stream of events, such as key presses, releases and timeouts, and translates them using a series of handler trait objects into reports that can easily be converted to the format USB expects.

This is inspired by the QMK mechanical keyboard firmware, which probably is the most feature complete keyboard firmware to date. Alas it's C, and an amazing ball of ifdefs that usually needs more flash than the micros it's targeting offer.

The trait object oriented approach chosen here allows composition of arbitrarily complex keyboard layouts - and unit testing them without hardware.

To get started, maybe check out the reference firmware.

KeyToKey operates on key codes that are u32 representing unicode key points. (The USB keycodes are nested in the first 'private area' of the unicode code set, and the rest of the private area is free to be used by the keyboard implementor.)

A basic keyboard uses two handlers - a handlers::UnicodeKeyboard, which sends the OS-specific magic 'enter-an-arbitrary-unicode-keypoint' for any key code outside of the private areas, and a handlers::USBKeyboard which handles all the usual 'tell the computer which buttons are pressed' functionality including modifiers.

USBKeyboard does not limit the number of simultanious keys, but the downstream translation into USB might restrict to the usual 6 key rollover.

Basic features

  • works as a regular USB keyboard
  • arbirtrary unicode input in linux and windows

Advanced Features working

  • Layers (which can rewrite key codes, conditionally rewrite them based on shift status or send arbitrary strings, again dependeant on shift)
  • RewriteLayers (which can only rewrite key codes, but are much more memory efficient)
  • PressReleaseMacros (callbacks on key press / key release)
  • StickyMacros (Tap once to activate, again to deactivate){
  • OneShots (Press -> activate, deactivates after next any-not-one-shot key press - useful for modifiers or temporarily activated layers)
  • SpaceCadet (Do one thing on press-and-hold, a different thing on tap. For example a shift key that also outputs a '('))
  • Sequences (e.g. t e h -> the which don't intercept the keycodes, but then send a set of backspace presses, and then your action)

Advanced features planned

  • Leader sequences (e.g. hit leader h e a r t to enter a heart emoji, or an arbitrary string)
  • TapDance (count the number of taps on a key, pass the final count to a callback)
  • AutoShift - Short tap: lower case, longer tap: uppercase. Removes key repeat though.
  • send keys later (e.g. for a macro that sends a key after a delay)

K2K strives to seperate the triggers (when does something happen) from the actual actions. Currently actions may be arbitrary (USB) KeyCodes being registered, &str being send or callbacks (implemented via traits)

Other rust firmwares / keyboard libraries

Dependencies

~2.5MB
~62K SLoC