/* Main goals: * * Have alphanumerics in the "normal" QWERTY keyboard spots. * * Move shift and enter to thumbs instead of pinkies. * * Shuffle modifier keys as necessary to make common combos comfortable. This * needs to cover Windows, Linux, and macOS key combos that I use. * * Support typing numbers and symbols using the main three rows (with * modifiers), for reduced finger reach. This includes a standard numpad * layout under the right hand. * * Support typing arrow, enter, and space characters with the left hand (using * modifiers) for when the right hand is on the mouse. * * Include common "media" keys, volume control, brightness, and screenlock. * * Support recording and playback of short key sequences. * * Support basic mouse cursor movement and button clicks. * * Make it hard to hit "dangerous" keys/combos by accident. */ /* Notes about "shooter mode": * * This is a mode to make the lefthand keyboard a bit nicer for mouse-and- * keyboard games, especially "first person shooters". * * Shooter mode can be entered or exited by pressing both GUI keys * simultaneously. * * Entering shooter mode while the mouse/numpad layer is active will deactivate * the mouse/numpad layer. The symbols/special layer can still be activated, * locked, and deactivated while in shooter mode, but the mouse/numpad layer * cannot. * * While in shooter mode, the left GUI key can be held to remap some of the * left-hand keys to right-hand keys. 1-5 become 6-0, Q-T become F6-F10, and * the top/bottom SYS keys become F11/F12. It does not perform its usual * left-GUI keycode. (The right GUI key still behaves normally.) */ /* Notes about LEDs: * * LED 1 is closest to the USB cables, and separated a bit from the other two. * LED 2 is in the middle and LED 3 is rightmost. * * In my setup, LED 1 = red, 2 = green, 3 = blue. * * When the keyboard powers on, the LEDs quickly turn on in sequence, turn off * in sequence, then all blink together once. (This is default hotdox firmware * behavior.) * * When the key combo is used to reset the keyboard for flashing, the LEDs all * turn on briefly then off. * * When holding down the key combo for the "power" or "sleep" keycode, the * LEDs cycle during the countdown until the keycode is triggered. * * While "shooter mode" is enabled, the red LED (#1) is lit. * * While the mouse/numpad layer is locked, the green LED (#2) is lit. * * While a macro is being recorded, the blue LED (#3) is lit. */ #include QMK_KEYBOARD_H #include "version.h" #define ALT_TAB_TIMEOUT_MS 1500 #define POWER_TIMEOUT_MS 1500 bool mousenum_lock = false; bool grave_down = false; bool sys_chord = false; bool sys_chord_flash = false; bool alt_tab_active = false; bool waiting_for_power_code= false; bool waiting_for_sleep_code= false; bool recording = false; uint16_t alt_tab_timer = 0; uint16_t power_timer = 0; uint16_t sleep_timer = 0; uint8_t led_cycle = 0; enum custom_keycodes { J_ALT_TAB = SAFE_RANGE, J_ALT_TIK, J_MN_L, J_MN_R, J_CW, J_SYS, J_SYSFL, J_PWR, J_SLEP }; enum layers{ _MAIN = 0, _SYM, _MNUM, _FPS, _FPS2 }; const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = { /* Keymap _MAIN: default/base layer * * - alphanum keys in "normal" QWERTY positions, plus some other symbol keys * - Ergodox-ish thumb key placement for shift, enter, space, backspace * - common thumb/pinky modifier keys where I like them * - clusters of cursor-movement keys * - keys to activate or lock other layers * - volume control * - screenlock combo for Windows and (some) Linux * * ,--------------------------------------------------. ,--------------------------------------------------. * | Esc | 1 | 2 | 3 | 4 | 5 | - | | = | 6 | 7 | 8 | 9 | 0 | F11 | * |--------+------+------+------+------+-------------| |------+------+------+------+------+------+--------| * | Tab | Q | W | E | R | T | SYS | | [ | Y | U | I | O | P | \ | * |--------+------+------+------+------+------| | | |------+------+------+------+------+--------| * | LCtrl | A | S | D | F | G |------| |------| H | J | K | L | ; | ' | * |--------+------+------+------+------+------| SYS | | ] |------+------+------+------+------+--------| * | MNUM | Z | X | C | V | B | | | | N | M | , | . | / | MNUM | * `--------+------+------+------+------+-------------' `-------------+------+------+------+------+--------' * |CPWD/`| PgUp | PgDn | LAlt | SYM | | SYM | Left | Down | Up | Right| * `----------------------------------' `----------------------------------' * ,-------------. ,-------------. * | WLOCK| Mute | | Vol- | Vol+ | * ,------|------|------| |------+------+------. * | | | M1 | | M2 | | | * |LShift| Bksp |------| |------| Enter| Space| * | | | LGui | | RGui | | | * `--------------------' `--------------------' * * * SYM key activates the symbols/special layer while held. * * MNUM key activates the mouse/numpad layer. Normally is only active while * held; press both MNUM keys together to lock the layer. Press one MNUM key * again to unlock the layer. When the mouse/numpad layer is locked, LED 2 * will be lit. * * CPWD activates "caps word" mode; letters will be capitalized and hyphens * changed to underlines until the space key is pressed. However if any * modifier key is held (Ctrl, Alt, Shift, GUI) then this key will behave like * the backtick/tilde key instead. * * WLOCK sends a macro (LGui-L) to lock the screen. This screen lock macro * works for Windows and various Linux setups; for macOS see the corresponding * key on the symbols/special layer. * * The M1 and M2 keys play back dynamic macros 1 and 2 respectively. * * Pressing the two SYS keys together will normally type out information about * the build and the currently active layers. If SYSFL is already held however * (by pressing the left-hand SYM or MNUM key while the other layer is active), * then pressing the two SYS keys will reset the keyboard for flashing. */ [_MAIN] = LAYOUT_ergodox( // left hand KC_ESC, KC_1, KC_2, KC_3, KC_4, KC_5, KC_MINS, KC_TAB, KC_Q, KC_W, KC_E, KC_R, KC_T, J_SYS, KC_LCTL, KC_A, KC_S, KC_D, KC_F, KC_G, J_MN_L, KC_Z, KC_X, KC_C, KC_V, KC_B, J_SYS, J_CW, KC_PGUP, KC_PGDN, KC_LALT, MO(_SYM), LGUI(KC_L), KC_MUTE, DM_PLY1, KC_LSFT, KC_BSPC, KC_LGUI, // right hand KC_EQL, KC_6, KC_7, KC_8, KC_9, KC_0, KC_F11, KC_LBRC, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_BSLS, KC_H, KC_J, KC_K, KC_L, KC_SCLN, KC_QUOT, KC_RBRC, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, J_MN_R, MO(_SYM), KC_LEFT, KC_DOWN, KC_UP, KC_RGHT, KC_VOLD, KC_VOLU, DM_PLY2, KC_RGUI, KC_ENT, KC_SPC ), /* Keymap _SYM: symbols/special layer (mutually exclusive with _MNUM) * * - function keys * - power/sleep/pause * - access to symbols and arrow keys on the main three rows * - media player * - screen brightness * - less-used "entry state" keys (caps/del/ins) * - screenlock combo for macOS * * ,--------------------------------------------------. ,--------------------------------------------------. * | Power | F1 | F2 | F3 | F4 | F5 | Sleep| | Pause| F6 | F7 | F8 | F9 | F10 | F12 | * |--------+------+------+------+------+-------------| |------+------+------+------+------+------+--------| * | AltTab | Space| Up | Enter| _ | ~ | | | PrSrc| " | + | | | { | } | Null | * |--------+------+------+------+------+------| | | |------+------+------+------+------+--------| * | | Left | Down | Right| - | ` |------| |------| ' | = | \ | [ | ] | Null | * |--------+------+------+------+------+------| | | Ins |------+------+------+------+------+--------| * | SYSFL | ! | @ | # | $ | % | | | | ^ | & | * | ( | ) | Null | * `--------+------+------+------+------+-------------' `-------------+------+------+------+------+--------' * |AltGrv| Home | End | | | | | << | Stop | Play | >> | * `----------------------------------' `----------------------------------' * ,-------------. ,-------------. * | MLOCK| | | Brt- | Brt+ | * ,------|------|------| |------+------+------. * | | | RecM1| | RecM2| | | * | Caps | Del |------| |------| Prev | Next | * | | | | | | | | * `--------------------' `--------------------' * * The POWER or SLEEP key needs to be held for 1.5 seconds to activate. LEDs * will cycle during this period. * * MLOCK sends a macro (LCtl-LGui-Q) to lock the screen. This screen lock macro * works for macOS; for Windows and various Linux setups see the corresponding * key on the base layer. * * AltTab sends Alt-Tab; repeated presses continue to send Tab while holding * Alt. AltGrv is similar for Alt-Backtick. These are just to make the window * cycling combos slightly more comfortable (using the SYM key instead of the * Alt key as the modifier). * * The RecM1 and RecM2 keys start/stop recording of dynamic macros 1 and 2 * respectively. (Nested macros are not allowed.) * * SYSFL unlocks the keyboard-flashing combo; see base layer comments for more. */ [_SYM] = LAYOUT_ergodox( // left hand J_PWR, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, J_SLEP, J_ALT_TAB, KC_SPC, KC_UP, KC_ENT, KC_UNDS, KC_TILD, KC_TRNS, KC_TRNS, KC_LEFT, KC_DOWN, KC_RGHT, KC_MINS, KC_GRV, J_SYSFL, KC_EXLM, KC_AT, KC_HASH, KC_DLR, KC_PERC, KC_TRNS, J_ALT_TIK, KC_HOME, KC_END, KC_TRNS, KC_TRNS, LCTL(LGUI(KC_Q)), KC_TRNS, DM_REC1, KC_CAPS, KC_DEL, KC_TRNS, // right hand KC_PAUS, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F12, KC_PSCR, KC_DQUO, KC_PLUS, KC_PIPE, KC_LCBR, KC_RCBR, KC_NO, KC_QUOT, KC_EQL, KC_BSLS, KC_LBRC, KC_RBRC, KC_NO, KC_INS, KC_CIRC, KC_AMPR, KC_ASTR, KC_LPRN, KC_RPRN, KC_NO, KC_TRNS, KC_MRWD, KC_MSTP, KC_MPLY, KC_MFFD, KC_BRID, KC_BRIU, DM_REC2, KC_TRNS, KC_MPRV, KC_MNXT ), /* Keymap _MNUM: mouse/numpad layer (mutually exclusive with _SYM) * * - left hand keys for mouse movement, wheel, and buttons * - right hand keys for numpad * * ,--------------------------------------------------. ,--------------------------------------------------. * | | Null | Null | Null | Null | Null | Null | | Null | Null | KP= | KP/ | KP* | KP- | NumLck | * |--------+------+------+------+------+-------------| |------+------+------+------+------+------+--------| * | |LClick| MsUp |RClick| WhUp | Null | | | Null | Null | KP7 | KP8 | KP9 | KP+ | Tab | * |--------+------+------+------+------+------| | | |------+------+------+------+------+--------| * | |MsLeft|MsDown|MsRght|WhDown| Null |------| |------| Null | KP4 | KP5 | KP6 | KP+ | Bksp | * |--------+------+------+------+------+------| | | Null |------+------+------+------+------+--------| * | | Btn3 | Btn4 | Btn5 | Null | Null | | | | Null | KP1 | KP2 | KP3 | KPEnt| | * `--------+------+------+------+------+-------------' `-------------+------+------+------+------+--------' * | Null | | | | SYSFL| | KP0 | KP0 | KP. | KPEnt| Null | * `----------------------------------' `----------------------------------' * ,-------------. ,-------------. * | | | | | | * ,------|------|------| |------+------+------. * | | | | | | | | * | | |------| |------| | | * | | | | | | | | * `--------------------' `--------------------' * * SYSFL unlocks the keyboard-flashing combo; see base layer comments for more. */ [_MNUM] = LAYOUT_ergodox( // left hand KC_TRNS, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_TRNS, KC_BTN1, KC_MS_U, KC_BTN2, KC_WH_U, KC_NO, KC_TRNS, KC_TRNS, KC_MS_L, KC_MS_D, KC_MS_R, KC_WH_D, KC_NO, KC_TRNS, KC_BTN3, KC_BTN4, KC_BTN5, KC_NO, KC_NO, KC_TRNS, KC_NO, KC_TRNS, KC_TRNS, KC_TRNS, J_SYSFL, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, // right hand KC_NO, KC_NO, KC_PEQL, KC_PSLS, KC_PAST, KC_PMNS, KC_NUM, KC_NO, KC_NO, KC_P7, KC_P8, KC_P9, KC_PPLS, KC_TAB, KC_NO, KC_P4, KC_P5, KC_P6, KC_PPLS, KC_BSPC, KC_NO, KC_NO, KC_P1, KC_P2, KC_P3, KC_PENT, KC_TRNS, KC_P0, KC_P0, KC_PDOT, KC_PENT, KC_NO, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS), /* Keymap _FPS: "shooter mode" layer (mutually exclusive with _MNUM) * * - enable or disable by pressing both GUI keys * - some left-hand keys remapped to serve normal FPS key bindings * - left and right MNUM-layer keys replaced by left and right shift * - hold left GUI to overlay the _FPS2 keymap * * ,--------------------------------------------------. * | | | | | | | | * |--------+------+------+------+------+-------------| * | ` | | | | | | | * |--------+------+------+------+------+------| | * | Tab | | | | | |------| * |--------+------+------+------+------+------| | * | LShift | | | | | | | * `--------+------+------+------+------+-------------' * | LCtrl| | | | | * `----------------------------------' * * ,-------------. * | | | * ,------|------|------| * | | | | * | Space| |------| * | | | FPS2 | * `--------------------' * * All other keys transparent, except RShift in the corresponding position on * the other hand. */ [_FPS] = LAYOUT_ergodox( // left hand KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_GRV, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TAB, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_LSFT, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_LCTL, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_SPC, KC_TRNS, KC_TRNS, // right hand KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_RSFT, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS), /* Keymap _FPS2: some right-hand keys overlaid on "shooter mode" layer * * - 1-5 mapped to 6-0 * - Q-T mapped to F6-F10 * - top/bottom SYS keys mapped to F11/F12 * * ,--------------------------------------------------. * | | 6 | 7 | 8 | 9 | 0 | | * |--------+------+------+------+------+-------------| * | ` | F6 | F7 | F8 | F9 | F10 | F11 | * |--------+------+------+------+------+------| | * | | | | | | |------| * |--------+------+------+------+------+------| F12 | * | | | | | | | | * `--------+------+------+------+------+-------------' * | | | | | | * `----------------------------------' * * ,-------------. * | | | * ,------|------|------| * | | | | * | | |------| * | | | | * `--------------------' * * All other keys transparent. */ [_FPS2] = LAYOUT_ergodox( // left hand KC_TRNS, KC_6, KC_7, KC_8, KC_9, KC_0, KC_TRNS, KC_TRNS, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_F12, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, // right hand KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS) }; // Set LEDs according to current state. Used to recover after an LED-cycling // behavior. void reset_leds(void) { ergodox_led_all_off(); if (IS_LAYER_ON(_FPS)) { ergodox_right_led_1_on(); } if (mousenum_lock) { ergodox_right_led_2_on(); } if (recording) { ergodox_right_led_3_on(); } } // Our hook for special actions on key events. Currently this handles the // custom J_* keycodes as well as the shooter-mode lock and unlock behavior // added to KC_LGUI and KC_RGUI. bool process_record_user(uint16_t keycode, keyrecord_t *record) { switch (keycode) { // This key normally activates Caps Word, but if any modifier key is // pressed, it is the backtick/tilde key. case J_CW: if (record->event.pressed) { // Key Down // Checking for modifier-key-pressed seems like a perfect // usecase for key overrides at first glance. However, to // preserve the modifier(s) in the resulting emitted key event // would require defining at least 8 overrides that need to // be cycled through and checked on each press. This is faster // and simpler. if (get_mods() == 0) { caps_word_on(); } else { register_code(KC_GRV); grave_down = true; } } else { // Key Up if (grave_down) { // Key was pressed as a backtick/tilde key, so should // release that now. unregister_code(KC_GRV); grave_down = false; } } return false; // Skip all further processing of this key // These keys send tab or backtick, but also on first press will // register the alt keycode. matrix_scan_user will take care of // releasing alt if these keys' layer is deactivated or if // ALT_TAB_TIMEOUT_MS elapses w/o more presses of these keys. case J_ALT_TAB: case J_ALT_TIK: if (record->event.pressed) { if (!alt_tab_active) { alt_tab_active = true; register_code(KC_LALT); } alt_tab_timer = timer_read(); if (keycode == J_ALT_TAB) { register_code(KC_TAB); } else { register_code(KC_GRV); } } else { if (keycode == J_ALT_TAB) { unregister_code(KC_TAB); } else { unregister_code(KC_GRV); } } return false; // Skip all further processing of this key // It is assumed that there will be two _MNUM control keys (one each // for left and right hand boards). Depressing a key will do a layer // enable while the key is held. Simultaneously depressing both of // them locks the layer on. If the layer is locked on, tap the same // kind of key again to unlock. case J_MN_L: case J_MN_R: if (record->event.pressed) { // Key Down if (IS_LAYER_OFF(_MNUM)) { // _MNUM isn't on yet, so turn it on. layer_on(_MNUM); } else { // _MNUM is already on; we are locking or unlocking it. mousenum_lock = !mousenum_lock; if (mousenum_lock) { ergodox_right_led_2_on(); } else { ergodox_right_led_2_off(); } } } else { // Key Up if (!mousenum_lock) { // If not locked, clear _MNUM. layer_off(_MNUM); } } return false; // Skip all further processing of this key // The GUI keys typically just emit left/right GUI key events. // However if both keys are pressed, toggle "shooter mode". (Enabling // shooter mode will disable the mouse/numpad layer if it is on.) // While shooter mode is enabled, left GUI serves to activate the // _FPS2 overlay instead of sending its usual code. case KC_LGUI: case KC_RGUI: { uint8_t mod_state = get_mods(); bool process_key = true; if (record->event.pressed) { // Key Down bool both_pressed = false; if ((mod_state & MOD_MASK_GUI) != 0) { // This covers the both-pressed state while FPS mode // is disabled, as well as LGUI-after-RGUI while FPS // mode is enabled. both_pressed = true; } else { if ((keycode == KC_RGUI) && IS_LAYER_ON(_FPS2)) { // RGUI-after-LGUI while FPS mode is enabled. both_pressed = true; } } if (both_pressed) { // Toggle shooter mode. if (IS_LAYER_OFF(_FPS)) { if (IS_LAYER_ON(_MNUM)) { if (mousenum_lock) { mousenum_lock = false; ergodox_right_led_2_off(); } layer_off(_MNUM); } layer_on(_FPS); ergodox_right_led_1_on(); } else { layer_off(_FPS); ergodox_right_led_1_off(); } // Never send the GUI keycode for the second key // pressed. process_key = false; } else { // If FPS layer active, LGUI activates FPS2 layer. if (IS_LAYER_ON(_FPS)) { if (keycode == KC_LGUI) { layer_on(_FPS2); process_key = false; } } } } else { // Key Up // Only process the keycode for GUI up if we previously // processed it for key down. if (keycode == KC_LGUI) { process_key = ((mod_state & MOD_BIT(KC_LGUI)) != 0); // For LGUI, key up also disables FPS2 overlay. if (IS_LAYER_ON(_FPS2)) { layer_off(_FPS2); } } else { process_key = ((mod_state & MOD_BIT(KC_RGUI)) != 0); } } return process_key; } // There should be two keys with this keycode assigned. If both are // depressed together, normally this will cause a string macro to be // emitted that prints out keyboard info. If a J_SYSFL key is also // being held down, then reset the keyboard for flashing. case J_SYS: if (record->event.pressed) { // Key Down if (sys_chord) { // The other J_SYS key is already pressed. if (sys_chord_flash) { // A FLASH-enable key is pressed too, so let's flash. ergodox_right_led_1_on(); ergodox_right_led_2_on(); ergodox_right_led_3_on(); _delay_ms(200); reset_keyboard(); } else { // Just do the normal sys chord behavior: dump info. bool shooter_mode = IS_LAYER_ON(_FPS); send_string_with_delay_P(PSTR(QMK_KEYBOARD "/" QMK_KEYMAP " @ " QMK_VERSION " locked layers/modes: "), 10); if (recording || mousenum_lock || shooter_mode) { if (recording) { send_string_with_delay_P(PSTR("RECORDING "), 10); } if (mousenum_lock) { send_string_with_delay_P(PSTR("MOUSE/NUMPAD "), 10); } if (shooter_mode) { send_string_with_delay_P(PSTR("SHOOTER"), 10); } } else { send_string_with_delay_P(PSTR("none"), 10); } } } else { sys_chord = true; } } else { // Key Up sys_chord = false; } return false; // Skip all further processing of this key // This keycode is entirely used just to require another keypress // before enabling the flash behavior of the J_SYS keys. Since this // key itself is not on a base layer, that means that flash-enable // is a four key combo. case J_SYSFL: sys_chord_flash = record->event.pressed; return false; // Skip all further processing of this key // For the power or sleep key, just record whether it is pressed and // mark the start time. matrix_scan_user will send the actual power or // sleep keycode if POWER_TIMEOUT_MS elapses while holding the key. case J_PWR: if (record->event.pressed) { // Key Down waiting_for_power_code = true; power_timer = timer_read(); // Blank LEDs to prep for cycling. ergodox_led_all_off(); } else { // Key Up if (!waiting_for_power_code) { // Only do key-up if matrix_scan_user did key-down. unregister_code(KC_PWR); } waiting_for_power_code = false; // Reset the LEDs to non-cycling state. reset_leds(); } return false; // Skip all further processing of this key case J_SLEP: if (record->event.pressed) { // Key Down waiting_for_sleep_code = true; sleep_timer = timer_read(); // Blank LEDs to prep for cycling. ergodox_led_all_off(); } else { // Key Up if (!waiting_for_sleep_code) { // Only do key-up if matrix_scan_user did key-down. unregister_code(KC_SLEP); } waiting_for_sleep_code = false; // Reset the LEDs to non-cycling state. reset_leds(); } return false; // Skip all further processing of this key default: return true; // Process all other keycodes normally } } // Deactivate the alt-hold for the special Alt-Tab and Alt-Backtick keys if // necessary. Also send the power or sleep keycode if the relevant key has // been held long enough. void matrix_scan_user(void) { if (alt_tab_active) { if (IS_LAYER_OFF(_SYM) || (timer_elapsed(alt_tab_timer) > ALT_TAB_TIMEOUT_MS)) { unregister_code(KC_LALT); alt_tab_active = false; } } if (waiting_for_power_code) { if (timer_elapsed(power_timer) > POWER_TIMEOUT_MS) { register_code(KC_PWR); waiting_for_power_code = false; } } if (waiting_for_sleep_code) { if (timer_elapsed(sleep_timer) > POWER_TIMEOUT_MS) { register_code(KC_SLEP); waiting_for_sleep_code = false; } } if (waiting_for_power_code || waiting_for_sleep_code) { switch (led_cycle) { case 0: ergodox_right_led_3_off(); ergodox_right_led_1_on(); led_cycle++; break; case 1: ergodox_right_led_1_off(); ergodox_right_led_2_on(); led_cycle++; break; case 2: ergodox_right_led_2_off(); ergodox_right_led_3_on(); led_cycle = 0; break; } _delay_ms(50); } } // When either macro starts recording, turn on the recording LED. void dynamic_macro_record_start_user(void) { recording = true; ergodox_right_led_3_on(); } // When current macro recording stops, turn off the recording LED. void dynamic_macro_record_end_user(int8_t direction) { recording = false; ergodox_right_led_3_off(); }