This is my personal ZMK firmware configuration. It consists of a 34-keys base layout that is re-used for various boards, including my Corneish Zen and my Planck.
My configuration builds against v0.1
of upstream ZMK plus a backport of the
merged pointer PR. Custom functionality is added through various
ZMK modules.
The state of the entire firmware is pinned in my west
manifest.
- "Timeless" homerow mods
- Combos instead of symbol layer
- Auto-toggle off numbers and mouse layers
- Leader key sequences for Unicode input and system commands
- Arrow-cluster doubles as home, end, begin/end of document on long-press
- Shifted actions that make sense: , ↦ ;, . ↦ : and ? ↦ !
- Simpler Devicetree syntax using helper macros from zmk-helpers
- Fully automated, nix-based local build environment
(Click here for a breakdown by layer - powered by keymap-drawer.)
Homerow mods (aka "HRMs") can be
a game changer -- at least in theory. In practice, they require some finicky
timing: In its most naive implementation, in order to produce a "mod", they must
be held longer than tapping-term-ms
. In order to produce a "tap", they must
be held less than tapping-term-ms
. This requires very consistent typing
speeds that, alas, I do not possess. Hence my quest for a "timer-less" HRM
setup.
After months of tweaking, I eventually ended up with a HRM setup that is essentially timer-less, resulting in virtually no misfires.1 Yet it provides a fluent typing experience with mostly no delays.
Let's suppose for a moment we set tapping-term-ms
to something ridiculously
large, say 5 seconds. This makes the configuration timer-less of sorts. But it
has two problems: (1) To activate a mod we will have to hold the HRM keys for
what feels like eternity. (2) During regular typing, there are delays between
the press of a key and the time it appears on the screen.2 Enter two of my
favorite ZMK features:
- To address the first problem, I use ZMK's
balanced
flavor, which produces a "hold" if another key is both pressed and released within the tapping-term. Because that is exactly what I normally do with HRMs, there is virtually never a need to wait past my long tapping term (see below for two exceptions). - To address the typing delay, I use ZMK's
require-prior-idle-ms
property, which immediately resolves a HRM as "tap" when it is pressed shortly after another key has been tapped. This all but completely eliminates the delay.
This is great but there are still a few rough edges:
- When rolling keys, I sometimes unintentionally end up with "nested" key
sequences:
key-1
down,key-2
down and up,key-1
up. Because of thebalanced
flavor, this would falsely registerkey-1
as a mod. As a remedy, I use ZMK's "positional hold-tap" feature to force HRMs to always resolve as "tap" when the next key is on the same side of the keyboard. Problem solved. - ... or at least almost. By default, positional-hold-tap performs the
positional check when the next key is pressed. This is not ideal, because it
prevents combining multiple modifiers on the same hand. To fix this, I use the
hold-trigger-on-release
setting, which delays the positional-hold-tap decision until the next key's release. With this, mods can be combined when held while positional hold-tap continues to work as expected when keys are tapped. - So far, nothing of the configuration depends on the duration of
tapping-term-ms
. In practice, there are two reasons why I don't set it to infinity:- Sometimes, in rare circumstances, I want to combine a mod with a alpha-key
on the same hand (e.g., when using the mouse with the other hand). My
positional hold-tap configuration prevents this within the tapping term.
By setting the tapping term to something large but not crazy large (I use
280ms), I can still use same-hand
mod
+alpha
shortcuts by holding the mod for just a little while before tapping the alpha-key. - Sometimes, I want to press a modifier without another key (e.g., on
Windows, tapping
Win
opens the search menu). Because thebalanced
flavour only kicks in when another key is pressed, this also requires waiting pasttapping-term-ms
.
- Sometimes, in rare circumstances, I want to combine a mod with a alpha-key
on the same hand (e.g., when using the mouse with the other hand). My
positional hold-tap configuration prevents this within the tapping term.
By setting the tapping term to something large but not crazy large (I use
280ms), I can still use same-hand
- Finally, it is worth noting that this setup works best in combination with a
dedicated shift for capitalization during normal typing (I like sticky-shift
on a home-thumb). This is because shifting alphas is the one scenario where
pressing a mod may conflict with
require-prior-idle-ms
, which may result in false negatives for fast typers.
Here's my configuration (I use a bunch of helper macros to simplify the syntax, but they are not necessary):
#include "zmk-helpers/key-labels/36.h" // Source key-labels.
#define KEYS_L LT0 LT1 LT2 LT3 LT4 LM0 LM1 LM2 LM3 LM4 LB0 LB1 LB2 LB3 LB4 // Left-hand keys.
#define KEYS_R RT0 RT1 RT2 RT3 RT4 RM0 RM1 RM2 RM3 RM4 RB0 RB1 RB2 RB3 RB4 // Right-hand keys.
#define THUMBS LH2 LH1 LH0 RH0 RH1 RH2 // Thumb keys.
/* Left-hand HRMs. */
ZMK_HOLD_TAP(hml,
flavor = "balanced";
tapping-term-ms = <280>;
quick-tap-ms = <175>;
require-prior-idle-ms = <150>;
bindings = <&kp>, <&kp>;
hold-trigger-key-positions = <KEYS_R THUMBS>;
hold-trigger-on-release;
)
/* Right-hand HRMs. */
ZMK_HOLD_TAP(hmr,
flavor = "balanced";
tapping-term-ms = <280>;
quick-tap-ms = <175>;
require-prior-idle-ms = <150>;
bindings = <&kp>, <&kp>;
hold-trigger-key-positions = <KEYS_L THUMBS>;
hold-trigger-on-release;
)
Hopefully, the above configuration "just works". If it doesn't, here's a few smaller (and larger) things to try.
- Noticeable delay when tapping HRMs: Increase
require-prior-idle-ms
. As a rule of thumb, you want to set it to at least10500/x
wherex
is your (relaxed) WPM for English prose.3 - False negatives (same-hand): Reduce
tapping-term-ms
(or disablehold-trigger-key-positions
) - False negatives (cross-hand): Reduce
require-prior-idle-ms
(or set flavor tohold-preferred
-- to continue usinghold-trigger-on-release
, you must apply this patch to ZMK - False positives (same-hand): Increase
tapping-term-ms
- False positives (cross-hand): Increase
require-prior-idle-ms
(or set flavor totap-preferred
, which requires holding HRMs past tapping term to activate)
I am a big fan of combos for all sort of things. In terms of comfort, I much prefer them over accessing layers that involve lateral thumb movements to be activated, especially when switching between layers in rapid succession.
One common concern about overloading the layout with combos is that they lead to
misfires. Fortunately, the above-mentioned require-prior-idle-ms
option also
works for combos, which in my experience all but completely eliminates misfires
-- even when rolling keys on the home row!
My combo layout aims to place the most used symbols in easy-to-access locations while also making them easy to remember. Specifically:
- the top vertical-combo row replicates the symbols on a standard numbers row
(except
+
and&
being swapped) - the bottom vertical-combo row is symmetric to the top row (subscript
_
aligns with superscript^
; minus-
aligns with+
; division/
aligns with multiplication*
; logical-or|
aligns with logical-and&
) - parenthesis, braces, brackets are set up symmetrically as horizontal combos
with
<
,>
,{
and}
being accessed from the Navigation layer (or when combined withShift
) - left-hand side combos for
tap
,esc
,cut
(on X + D),copy
andpaste
that go well with right-handed mouse usage
Inspired by Jonas Hietala's Numword for QMK, I implemented my own Auto-layer behavior for ZMK to set up Numword. It is triggered via a single tap on "Smart-Num". Numword continues to be activated as long as I type numbers, and deactivates automatically on any other keypress (holding it activates a non-sticky num layer).
After using Numword for more than a year now, I have been overall very happy with it. When typing single digits, it effectively is a sticky-layer but with the added advantage that I can also use it to type multiple digits.
The main downside is that if a sequence of numbers is immediately followed by
any of the letters on which my numpad is located (WFPRSTXCD), then the automatic
deactivation won't work. But this is rare -- most number sequences are
terminated by space
, return
or some form of punctuation/delimination. To
deal with the rare cases where they aren't, there is a CANCEL
key on the
navigation-layer that deactivates Numword, Capsword and Smart-mouse. (It also
toggles off when pressing Numword
again, but I find it cognitively easier to
have a dedicated "off-switch" than keeping track of which modes are currently
active.)
Similarly to Numword, I have a smart-mouse layer (activated by comboing W + P), which replaces the navigation cluster with scroll and mouse-movements, and replaces the right thumbs with mouse buttons. Pressing any other key automatically deactivates the layer.
My right thumb triggers three variations of shift: Tapping yields sticky-shift (used to capitalize alphas), holding activates a regular shift, and double-tapping (or equivalently shift + tap) activates ZMK's Caps-word behavior.
One minor technical detail: While it would be possible to implement the double-tap functionality as a tap-dance, this would add a delay when using single taps. To avoid the delays, I instead implemented the double-tap functionality as a mod-morph.
To economize on keys, I am using hold-taps on my navigation cluster, which yield
home
, end
, begin/end of document
, and delete word forward/backward
on
long-presses. The exact implementation is tweaked so that Ctrl
is silently
absorbed in combination with home
and end
to avoid accidental document-wide
operations (which are accessible via the dedicated begin/end document keys
.)
I am using Nick Conway's fantastic
tri-state behavior for a
one-handed Alt-Tab switcher (PWin
and NWin
).
I am using my own implementation of a
Leader key (activated by comboing
S + T) to bind various behaviors to my layout without
reserving dedicated keys. Currently, I am using them to bind German Umlauts,
Greek letters for math usage, and various system commands (e.g., to toggle
Bluetooth). See
leader.dtsi
for the full list of leader key sequences.
I streamline my local build process using nix
, direnv
and just
. This
automatically sets up a virtual development environment with west
, the
zephyr-sdk
and all its dependencies when cd
-ing into the ZMK-workspace. The
environment is completely isolated and won't pollute your system.
-
Install the
nix
package manager:# Install Nix with flake support enabled curl --proto '=https' --tlsv1.2 -sSf -L https://install.determinate.systems/nix | sh -s -- install --no-confirm # Start the nix daemon without restarting the shell . /nix/var/nix/profiles/default/etc/profile.d/nix-daemon.sh
-
Install
direnv
(and optionally but recommendednix-direnv
4) using your package manager of choice. E.g., using thenix
package manager that we just installed5:nix profile install nixpkgs#direnv nixpkgs#nix-direnv
-
Set up the
direnv
shell-hook for your shell. E.g., forbash
:# Install the shell-hook echo 'eval "$(direnv hook bash)"' >> ~/.bashrc # Enable nix-direnv (if installed in the previous step) mkdir -p ~/.config/direnv echo 'source $HOME/.nix-profile/share/nix-direnv/direnvrc' >> ~/.config/direnv/direnvrc # Optional: make direnv less verbose echo '[global]\nwarn_timeout = "2m"\nhide_env_diff = true' >> ~/.config/direnv/direnv.toml # Source the bashrc to activate the hook (or start a new shell) source ~/.bashrc
-
Clone your fork of this repository. I like to name my local clone
zmk-workspace
as it will be the toplevel of the development environment.# Replace `urob` with your username git clone https://github.com/urob/zmk-config zmk-workspace
-
Enter the workspace and set up the environment.
# The first time you enter the workspace, you will be prompted to allow direnv cd zmk-workspace # Allow direnv for the workspace, which will set up the environment direnv allow # Initialize the Zephyr workspace and pull in the ZMK dependencies # (same as `west init -l config && west update && west zephyr-export`) just init
After following the steps above your workspace should look like this:
zmk-workspace
├── config
├── firmware (created after building)
├── modules
├── zephyr
└── zmk
To build the firmware, simply type just build all
from anywhere in the
workspace. This will parse build.yaml
and build the firmware for all board and
shield combinations listed there.
To only build the firmware for a specific target, use just build <target>
.
This will build the firmware for all matching board and shield combinations. For
instance, to build the firmware for my Corneish Zen, I can type
just build zen
, which builds both corneish_zen_v2_left
and
corneish_zen_v2_right
. (just list
shows all valid build targets.)
Additional arguments to just build
are passed on to west
. For instance, a
pristine build can be triggered with just build all -p
.
(For this particular example, there is also a just clean
recipe, which clears
the build cache. To list all available recipes, type just
. Bonus tip: just
provides
completion scripts
for many shells.)
The build environment packages
keymap-drawer. just draw
parses
base.keymap
and draws it to draw/base.svg
. I haven't gotten around to
tweaking the output yet, so for now this is just a demonstration of how to set
things up.
To make changes to the ZMK source or any of the modules, simply edit the files
or use git
to pull in changes.
To switch to any remote branches or tags, use git fetch
inside a module
directory to make the remote refs locally available. Then switch to the desired
branch with git checkout <branch>
as usual. You may also want to register
additional remotes to work with or consider making them the default in
config/west.yml
.
To update the ZMK dependencies, use just update
. This will pull in the latest
version of ZMK and all modules specified in config/west.yml
. Make sure to
commit and push all local changes you have made to ZMK and the modules before
running this command, as this will overwrite them.
To upgrade the Zephyr SDK and Python build dependencies, use just upgrade-sdk
.
Using the same Nix-based environment, I have set up a drop-in replacement for the default ZMK Github Actions build workflow. While mainly a proof-of-concept, it does run moderately faster, especially with a cold cache.
By default, the workflow is configured to run "on demand" (head to the Actions
tab in your fork and click "Run workflow" to trigger it). To enable automatic
runs, remove the commented lines near the top of
.github/workflows/build-nix.yml
. (You may
want to comment the corresponding lines of the default build.yml
to avoid
running both workflows.)
Since I switched from QMK to ZMK I have been very impressed with how easy it is to set up relatively complex layouts in ZMK. For the most parts I don't miss any functionality (to the contrary, I found that ZMK supports many features natively that would require complex user-space implementations in QMK). Below are a few remaining issues:
- ZMK does not yet support "tap-only" combos (#544), requiring a brief pause when wanting to chord HRMs that overlap with combo positions. As a workaround, I implemented all homerow combos as homerow-mod-combos. This is good enough for day-to-day, but does not address all edge cases (eg changing active mods).
- Very minor:
&bootloader
doesn't work with stm32 boards like the Planck (#1086)
- The collection of ZMK modules used in this configuration.
- A ZMK-centric introduction to Git (useful for maintaining your own ZMK fork with a custom selection of PRs).
Footnotes
-
I call it "timer-less", because the large tapping-term makes the behavior insensitive to the precise timings. One may say that there is still the
require-prior-idle
timeout. However, with both a large tapping-term and positional-hold-taps, the behavior is not actually sensitive to therequire-prior-idle
timing: All it does is reduce the delay in typing; i.e., variations in typing speed won't affect what is being typed but merely how fast it appears on the screen. ↩ -
The delay is determined by how quickly a key is released and is not directly related to the tapping-term. But regardless of its duration, most people still find it noticeable and disruptive. ↩
-
E.g, if your WPM is 70 or larger, then the default of 150ms (=10500/70) should work well. The rule of thumb is based on an average character length of 4.7 for English words. Taking into account 1 extra tap for
space
, this yields a minimumrequire-prior-idle-ms
of (60 _ 1000) / (5.7 _ x) ≈ 10500 / x milliseconds. The approximation errs on the safe side, as in practice home row taps tend to be faster than average. ↩ -
nix-direnv
provides a vastly improved caching experience compared to only havingdirenv
, making entering and exiting the workspace instantaneous after the first time. ↩ -
This will permanently install the packages into your local profile, forgoing many of the benefits that make Nix uniquely powerful. A better approach, though beyond the scope of this document, is to use
home-manager
to maintain your user environment. ↩