You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

329 lines
22 KiB

  1. // Copyright 2022 Dane Skalski (@Daneski13)
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include QMK_KEYBOARD_H
  4. // Layers enum
  5. enum junco_layers { _QWERTY, _COLEMAK_DH, _SYMB, _EXT, _ADJUST };
  6. // Custom keycodes
  7. enum custom_keycodes {
  8. // Keycode for toggling between macOS and Windows key mappings
  9. // Actually just an alias to the GUI and Alt swap Magic keycode
  10. KC_OS = MAGIC_TOGGLE_ALT_GUI,
  11. // Keycode for swapping the base layer between QWERTY and Colemak-DH
  12. KC_TOGGLE_BASE = SAFE_RANGE,
  13. // Keycode for redo action (Ctrl + Y on windows, Ctrl + Shift + Z on macOS)
  14. KC_REDO,
  15. // Keycodes for next/previous word
  16. KC_WNXT,
  17. KC_WPRV,
  18. // Keycode for sticking/unsticking the adjust layer
  19. KC_ADJST
  20. };
  21. /* LED indicators */
  22. bool is_caps_lock_enabled(void) { // Caps lock
  23. return (host_keyboard_led_state().caps_lock);
  24. }
  25. bool is_num_lock_enabled(void) { // Num lock
  26. return (host_keyboard_led_state().num_lock);
  27. }
  28. bool is_adjust_layer_sticky(layer_state_t state) { // Adjust layer sticky
  29. // Checks if the state is equal to just the adjust layer being active.
  30. // Doing it this way can leverage SPLIT_LAYER_STATE_ENABLE
  31. return (state == (1UL << _ADJUST)) ? true : false;
  32. }
  33. // Indicator color based on the RGB Matrix mode
  34. RGB indicator_color(void) {
  35. RGB rgb;
  36. // Normally the indicator color is white to stand out in the RGB rainbow.
  37. // When using the custom RGB mode that already is white, the indicator color
  38. // is green to stand out.
  39. if (rgb_matrix_config.mode == RGB_MATRIX_CUSTOM_WHITE_UNDERGLOW_CYCLE) {
  40. // Green
  41. rgb.r = 0;
  42. rgb.g = 255;
  43. rgb.b = 0;
  44. } else {
  45. // White
  46. rgb.r = 255;
  47. rgb.g = 255;
  48. rgb.b = 255;
  49. }
  50. return rgb;
  51. }
  52. const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
  53. // clang-format off
  54. /*
  55. Traditional QWERTY
  56. ` 1 2 3 4 5 6 7 8 9 0 -
  57. Tab Q W E R T Y U I O P Ent
  58. Esc A S D F G H J K L ; '
  59. Sft Z X C V B Mut N M , . / Sft
  60. CtrWinAltDelSpc SftBspAltWinCtr
  61. For macOS - GUI (cmd) and Alt (opt) swapped
  62. */
  63. [_QWERTY] = LAYOUT_split4x6_r1(
  64. KC_GRAVE, KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_MINUS,
  65. KC_TAB, KC_Q, KC_W, KC_E, KC_R, KC_T, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_ENTER,
  66. KC_ESC, KC_A, KC_S, KC_D, KC_F, KC_G, KC_H, KC_J, KC_K, KC_L, KC_SEMICOLON, KC_QUOTE,
  67. KC_LSFT, KC_Z, KC_X, KC_C, KC_V, KC_B, KC_MUTE, KC_MPLY, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, KC_RSFT,
  68. KC_LCTL, KC_LGUI, KC_LALT, LT(_EXT, KC_DEL), KC_SPC, KC_RSFT, LT(_SYMB, KC_BSPC), KC_RALT, KC_RGUI, KC_RCTL
  69. ),
  70. /*
  71. Colemak-DH
  72. ` 1 2 3 4 5 6 7 8 9 0 -
  73. Tab Q W F P B J L U Y ; Ent
  74. Esc A R S T G M N E I O '
  75. Sft Z X C D V Mut K H , . / Sft
  76. CtrWinAltDelSpc SftBspAltWinCtr
  77. For macOS - GUI/Win (cmd) and Alt (opt) swapped
  78. */
  79. [_COLEMAK_DH] = LAYOUT_split4x6_r1(
  80. KC_GRAVE, KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_MINUS,
  81. KC_TAB, KC_Q, KC_W, KC_F, KC_P, KC_B, KC_J, KC_L, KC_U, KC_Y, KC_SEMICOLON, KC_ENTER,
  82. KC_ESC, KC_A, KC_R, KC_S, KC_T, KC_G, KC_M, KC_N, KC_E, KC_I, KC_O, KC_QUOTE,
  83. KC_LSFT, KC_Z, KC_X, KC_C, KC_D, KC_V, KC_MUTE, KC_MPLY, KC_K, KC_H, KC_COMM, KC_DOT, KC_SLSH, KC_RSFT,
  84. KC_LCTL, KC_LGUI, KC_LALT, LT(_EXT, KC_DEL), KC_SPC, KC_RSFT, LT(_SYMB, KC_BSPC), KC_RALT, KC_RGUI, KC_RCTL
  85. ),
  86. /*
  87. Symbols/Numpad Layer
  88. F1 F2 F3 F4 F5 F6 F7 F8 F9 F10F11F12
  89. Tab ! @ # $ % * 7 8 9 + Ent
  90. \ _ [ { ( ^ = 4 5 6 0 NUM
  91. ___ | ] } ) & ___ ___ / 1 2 3 - ___
  92. _______________ _______________
  93. */
  94. [_SYMB] = LAYOUT_split4x6_r1(
  95. KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12,
  96. KC_TAB, KC_EXLM, KC_AT, KC_HASH, KC_DLR, KC_PERC, KC_PAST, KC_P7, KC_P8, KC_P9, KC_PPLS, KC_ENTER,
  97. KC_BSLS, KC_UNDS, KC_LBRC, KC_LCBR, KC_LPRN, KC_CIRC, KC_PEQL, KC_P4, KC_P5, KC_P6, KC_P0, KC_NUM,
  98. _______, KC_PIPE, KC_RBRC, KC_RCBR, KC_RPRN, KC_AMPR, _______, _______, KC_PSLS, KC_P1, KC_P2, KC_P3, KC_PMNS, _______,
  99. _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
  100. ),
  101. /*
  102. Extension/Function Layer
  103. F1 F2 F3 F4 F5 F6 F7 F8 F9 F10 F11 F12
  104. PGUPEnd Home BRIU F7 F8 F9 F10____
  105. Cps PGDN BRID F4 F5 F6 F11____
  106. ____ ____ ____ F1 F2 F3 F12____
  107. _______________ _______________
  108. */
  109. [_EXT] = LAYOUT_split4x6_r1(
  110. KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12,
  111. S(KC_TAB), KC_PGUP, KC_END, KC_UP, KC_HOME, _______, KC_BRIU, KC_F7, KC_F8, KC_F9, KC_F10, _______,
  112. KC_CAPS, KC_PGDN, KC_LEFT, KC_DOWN, KC_RIGHT, _______, KC_BRID, KC_F4, KC_F5, KC_F6, KC_F11, _______,
  113. _______, _______, _______, _______, _______, _______, _______, _______, _______, KC_F1, KC_F2, KC_F3, KC_F12, _______,
  114. _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
  115. ),
  116. /*
  117. Adjust Layer, Keyboard Settings
  118. SpdUHueUSatUValURnxtStck EClrRbt DBUGBOOT
  119. SpdDHueDSatDValDRprvRTgl LOUT OS
  120. RTgl
  121. _______________ _______________
  122. */
  123. [_ADJUST] = LAYOUT_split4x6_r1(
  124. KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO,
  125. RGB_SPI, RGB_HUI, RGB_SAI, RGB_VAI, RGB_MOD, KC_ADJST, KC_NO, EE_CLR, QK_RBT, DB_TOGG, QK_BOOT, KC_NO,
  126. RGB_SPD, RGB_HUD, RGB_SAD, RGB_VAD, RGB_RMOD, RGB_TOG, KC_NO, KC_TOGGLE_BASE, KC_OS, KC_NO, KC_NO, KC_NO,
  127. KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, RGB_TOG, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO,
  128. _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
  129. )
  130. // clang-format on
  131. };
  132. /*
  133. --- Rotary Encoder Mappings ---
  134. Encoder mappings go from leftmost encoder to rightmost encoder on the physical board.
  135. index 0 is the the optional leftmost encoder on the left half, index 1 is the right encoder
  136. on the left half (by the thumb keys), index 2 is the left encoder on the right half (by the
  137. thumb keys), and index 3 is the optional rightmost encoder on the right half.
  138. If you are only using the 2 required encoders by the thumb keys, you only need to worry about
  139. index 1 and index 2.
  140. Note that the key to be sent for counter-clockwise rotation (CCW) goes first and then the key for
  141. clockwise (CW) within ENCODER_CCW_CW.
  142. */
  143. #ifdef ENCODER_MAP_ENABLE
  144. // clang-format off
  145. // Base layer encoder mappings:
  146. const uint16_t PROGMEM encoder_map[][NUM_ENCODERS][NUM_DIRECTIONS] = {
  147. // Base layers
  148. // index 0: mouse wheel up (CCW)/down (CW) index 1: volume up/down index 2: media prev/next index 3: mouse wheel left/right
  149. [_QWERTY] = { ENCODER_CCW_CW(KC_MS_WH_UP, KC_MS_WH_DOWN), ENCODER_CCW_CW(KC_VOLD, KC_VOLU), ENCODER_CCW_CW(KC_MPRV, KC_MNXT), ENCODER_CCW_CW(KC_WH_L, KC_WH_R) },
  150. [_COLEMAK_DH] = { ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS) },
  151. // Passes through to base layer
  152. [_SYMB] = { ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS) },
  153. // undo/redo previous word/next word
  154. [_EXT] = { ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_TRNS, KC_TRNS), ENCODER_CCW_CW(KC_UNDO, KC_REDO), ENCODER_CCW_CW(KC_WPRV, KC_WNXT) },
  155. // RGB Speed down/up RGB previous mode/next mode RGB brightness down/up
  156. [_ADJUST] = { ENCODER_CCW_CW(RGB_SPD, RGB_SPI), ENCODER_CCW_CW(RGB_RMOD, RGB_MOD), ENCODER_CCW_CW(RGB_VAD, RGB_VAI), ENCODER_CCW_CW(KC_TRNS, KC_TRNS) },
  157. // clang-format on
  158. };
  159. #endif
  160. // Called whenever a layer is changed
  161. layer_state_t layer_state_set_user(layer_state_t state) {
  162. // Make sure the adjust layer isn't sticky
  163. if (is_adjust_layer_sticky(state)) return state;
  164. // When both the symbol and extension layer keys are held, the Adjust layer is active.
  165. return update_tri_layer_state(state, _SYMB, _EXT, _ADJUST);
  166. }
  167. bool process_record_user(uint16_t keycode, keyrecord_t *record) {
  168. switch (keycode) {
  169. // Toggle base layer
  170. case KC_TOGGLE_BASE:
  171. if (record->event.pressed) {
  172. // Toggle swapping base layers between Colemak-DH and QWERTY.
  173. // When base layer is QWERTY, swap to Colemak-DH and vice-versa
  174. if (get_highest_layer(default_layer_state) == _QWERTY) {
  175. default_layer_set(1UL << _COLEMAK_DH);
  176. } else {
  177. default_layer_set(1UL << _QWERTY);
  178. }
  179. }
  180. return false;
  181. // Override undo in favor of the more modern undo action
  182. case KC_UNDO:
  183. if (record->event.pressed) {
  184. // Use the correct modifier for macOS or Windows
  185. uint16_t mod = keymap_config.swap_lalt_lgui ? KC_LGUI : KC_LCTL;
  186. // Send Ctrl+Z/Cmd+Z
  187. register_code(mod);
  188. tap_code_delay(KC_Z, 10);
  189. unregister_code(mod);
  190. }
  191. return false;
  192. // Redo action
  193. case KC_REDO:
  194. if (record->event.pressed) {
  195. // Whether or not macOS mapping is enabled
  196. if (keymap_config.swap_lalt_lgui) {
  197. // macOS - Send Cmd+Shift+Z
  198. register_code(KC_LGUI);
  199. register_code(KC_LSFT);
  200. tap_code_delay(KC_Z, 10);
  201. unregister_code(KC_LSFT);
  202. unregister_code(KC_LGUI);
  203. } else {
  204. // Windows - Send Ctrl+Y
  205. register_code(KC_LCTL);
  206. tap_code_delay(KC_Y, 10);
  207. unregister_code(KC_LCTL);
  208. }
  209. }
  210. return false;
  211. // Next word
  212. case KC_WNXT:
  213. if (record->event.pressed) {
  214. // Use the correct modifier for macOS or Windows
  215. uint16_t mod = keymap_config.swap_lalt_lgui ? KC_LALT : KC_LCTL;
  216. // Send Ctrl+Right/Option+Right
  217. register_code(mod);
  218. tap_code_delay(KC_RGHT, 10);
  219. unregister_code(mod);
  220. }
  221. return false;
  222. // Previous word
  223. case KC_WPRV:
  224. if (record->event.pressed) {
  225. // Use the correct modifier for macOS or Windows
  226. uint16_t mod = keymap_config.swap_lalt_lgui ? KC_LALT : KC_LCTL;
  227. // Send Ctrl+Left/Option+Left
  228. register_code(mod);
  229. tap_code_delay(KC_LEFT, 10);
  230. unregister_code(mod);
  231. }
  232. return false;
  233. // Stick / Unstick the adjust layer
  234. case KC_ADJST:
  235. if (record->event.pressed) {
  236. // If currently not sticky, we want only the adjust layer to be active to make it stick.
  237. // Otherwise we want the default layer, un-stick.
  238. is_adjust_layer_sticky(layer_state) ? layer_state_set(default_layer_state) : layer_move(_ADJUST);
  239. dprintf("Adjust layer is now %s\n", is_adjust_layer_sticky(layer_state) ? "stuck" : "un-stuck");
  240. }
  241. return false;
  242. default:
  243. return true;
  244. }
  245. }
  246. /* Indicators (Caps Lock / Num Lock / Adjust Layer Sticky) */
  247. bool rgb_matrix_indicators_user(void) {
  248. layer_state_t curr_layer_state = layer_state;
  249. layer_state_t layer = get_highest_layer(curr_layer_state);
  250. RGB rgb = indicator_color();
  251. /* Only show the indicator on their respective layers */
  252. // Caps Lock is only on the extension layer
  253. if (is_caps_lock_enabled() && layer == _EXT) {
  254. rgb_matrix_set_color(CAPS_LOCK_LED_INDEX, rgb.r, rgb.g, rgb.b);
  255. }
  256. // Num Lock is only on the symbol layer
  257. if (is_num_lock_enabled() && layer == _SYMB) {
  258. rgb_matrix_set_color(NUM_LOCK_LED_INDEX, rgb.r, rgb.g, rgb.b);
  259. }
  260. // If the adjust layer is stuck/sticky, light it up. Don't need to care about
  261. // checking the layer since it can only be active on the adjust layer anyway
  262. if (is_adjust_layer_sticky(curr_layer_state)) rgb_matrix_set_color(ADJST_LED_INDEX, rgb.r, rgb.g, rgb.b);
  263. return false;
  264. }