keymap.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. #include QMK_KEYBOARD_H
  2. #include "action_layer.h"
  3. // TODO: THOSE ARE IDEAS:
  4. // TODO:
  5. // TODO: - Proper support for "GUI+TAB" application switching, with the GUI holding and shift, etc.
  6. // TODO: - Maybe implement a process_record instead of a tap-dance, or complex tap dance?
  7. // TODO: - Media keys on Fn1 layer, *hjklm,.* or arrow cluster?
  8. // TODO: - What more than NumPad + RGB on the Fn3 toggled layer?
  9. // TODO: - Add an in-keymap way to toggle LGUI/F(0) key, HELD_ESC_IS_SHIFT and BSPC_BLOCKS_DEL.
  10. // TODO:
  11. // TODO: THOSE ARE BUGS TO FIX:
  12. // TODO: - None (found so far)
  13. // Adjusting process_record_user functionnalities, comment to disable.
  14. #define ISOLIKE_ALT_ENTER
  15. #define TWO_SFT_CAPS
  16. #define APP_IS_RALT_RCTRL
  17. #define HELD_ESC_IS_SHIFT
  18. #define BSPC_BLOCKS_DEL
  19. #define ALT_MINSEQL_IS_ALT_78
  20. // Cases where the GUI key will actually do what the GUI key normally does.
  21. #define AC_G_W LGUI(KC_W) // Ubuntu: Shows windows on current desktop
  22. #define AC_G_S LGUI(KC_S) // Ubuntu: Overview of all desktops
  23. #define AC_G_D LGUI(KC_D) // Windows: Show/Toggle desktop
  24. #define AC_G_L LGUI(KC_L) // Ubuntu/Windows: Lock session
  25. #define AC_G_T LGUI(KC_T) // Ubuntu: Shows Trash // elementary: Opens terminal
  26. #define AC_G_E LGUI(KC_E) // Windows: Opens file explorer
  27. #define AC_G_H LGUI(KC_H) // Windows: Show/Hide hidden files
  28. #define AC_G_SPC LGUI(KC_SPC) // elementary: Shows application launcher
  29. const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
  30. // 0: Base Layer
  31. LAYOUT_all(
  32. KC_GRAVE, KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_MINS, KC_EQL, KC_BSPC, KC_DEL, \
  33. KC_TAB, KC_Q, KC_W, KC_E, KC_R, KC_T, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_LBRC, KC_RBRC, KC_BSLS, \
  34. KC_ESC, KC_A, KC_S, KC_D, KC_F, KC_G, KC_H, KC_J, KC_K, KC_L, KC_SCLN, KC_QUOT, KC_NO, KC_ENT, \
  35. KC_LSFT, KC_LGUI, KC_Z, KC_X, KC_C, KC_V, KC_B, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, KC_RSFT, KC_UP, F(0), \
  36. KC_LCTL, TD(0), KC_LALT, KC_SPC , KC_RALT, KC_RCTRL, KC_LEFT, KC_DOWN, KC_RIGHT),
  37. // 1: Function 1 Layers
  38. LAYOUT_all(
  39. RESET, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12, KC_PAUS, KC_PSCR, \
  40. KC_TRNS, KC_BTN1, KC_MS_U, KC_BTN2, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, \
  41. KC_TRNS, KC_MS_L, KC_MS_D, KC_MS_R, KC_TRNS, KC_TRNS, KC_TRNS, KC_MPLY, KC_MPRV, KC_MNXT, KC_MSTP, KC_TRNS, KC_NO, KC_TRNS, \
  42. KC_TRNS, F(2), KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_MUTE, KC_VOLD, KC_VOLU, KC_TRNS, KC_TRNS, KC_PGUP, KC_TRNS, \
  43. KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_HOME, KC_PGDOWN, KC_END),
  44. // 2: GUI/Function 2 Layer
  45. LAYOUT_all(
  46. KC_ESC, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12, KC_PAUS, KC_PSCR, \
  47. KC_TRNS, KC_TRNS, AC_G_W, AC_G_E, KC_TRNS, AC_G_T, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, \
  48. KC_TRNS, KC_TRNS, AC_G_S, AC_G_D, KC_TRNS, KC_TRNS, AC_G_H, KC_TRNS, KC_TRNS, AC_G_L, KC_TRNS, KC_TRNS, KC_NO, KC_TRNS, \
  49. 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_PGUP, KC_TRNS, \
  50. KC_TRNS, KC_TRNS, KC_TRNS, AC_G_SPC, KC_TRNS, KC_TRNS, KC_HOME, KC_PGDOWN, KC_END),
  51. // 3: NumPad/Function 3 Toggle Layer
  52. LAYOUT_all(
  53. KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_NO, KC_7, KC_8, KC_9, KC_PSLS, KC_PMNS, KC_PPLS, KC_TRNS, KC_TRNS, \
  54. KC_TRNS, KC_NO, KC_NO, RGB_HUI, RGB_SAI, RGB_VAI, KC_NO, KC_4, KC_5, KC_6, KC_PAST, KC_NO, KC_NO, KC_NO, \
  55. KC_TRNS, KC_NO, KC_NO, RGB_HUD, RGB_SAD, RGB_VAD, KC_NO, KC_1, KC_2, KC_3, KC_PMNS, KC_PENT, KC_NO, KC_TRNS, \
  56. KC_NO, KC_TRNS, KC_NO, KC_NO, RGB_TOG, RGB_MOD, KC_NO, KC_NO, KC_0, KC_COMM, KC_DOT, KC_PPLS, KC_NO, KC_TRNS, KC_TRNS, \
  57. KC_NO, KC_TRNS, KC_NO, KC_TRNS, KC_PENT, KC_PENT, KC_TRNS, KC_TRNS, KC_TRNS) ,
  58. // TRaNSparent layer for reference
  59. /* LAYOUT_all(
  60. 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, \
  61. 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, \
  62. 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_NO, KC_TRNS, \
  63. 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, \
  64. / KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS, KC_TRNS), */
  65. };
  66. // Custom Actions
  67. const uint16_t PROGMEM fn_actions[] = {
  68. [0] = ACTION_LAYER_MOMENTARY(1), // to Fn1 layer
  69. [1] = ACTION_LAYER_MOMENTARY(2), // to GUI/Fn2 layer
  70. [2] = ACTION_LAYER_TOGGLE(3), // to Fn3/Num toggle layer
  71. };
  72. // Macros
  73. const macro_t *action_get_macro(keyrecord_t *record, uint8_t id, uint8_t opt) {
  74. // MACRODOWN only works in this function
  75. switch(id) {
  76. case 0:
  77. if (record->event.pressed) { register_code(KC_RSFT); }
  78. else { unregister_code(KC_RSFT); }
  79. break;
  80. }
  81. return MACRO_NONE;
  82. };
  83. // Loop
  84. void matrix_scan_user(void) {
  85. // Empty
  86. };
  87. // LGUI acts as F(1) if held or as a tapped LGUI if tapped. Adapted from https://goo.gl/WnqGNS
  88. // Commented lines are for the "sticky" layer on two presses. Left it here for reference.
  89. static const int GUILAY = 2; // GUI Layer is layer #2
  90. typedef struct {
  91. bool layer_toggle;
  92. bool sticky;
  93. } td_ta_state_t;
  94. static void ang_tap_dance_ta_finished (qk_tap_dance_state_t *state, void *user_data) {
  95. td_ta_state_t *td_ta = (td_ta_state_t *) user_data;
  96. // if (td_ta->sticky) {
  97. // td_ta->sticky = false;
  98. // td_ta->layer_toggle = false;
  99. // layer_off (GUILAY);
  100. // return;
  101. // }
  102. //
  103. if (state->count == 1 && !state->pressed) {
  104. register_code (KC_LGUI);
  105. // td_ta->sticky = false;
  106. td_ta->layer_toggle = false;
  107. } else {
  108. td_ta->layer_toggle = true;
  109. layer_on(GUILAY);
  110. // td_ta->sticky = (state->count == 2);
  111. }
  112. }
  113. // Added this one to make it more reactive on keyup
  114. static void ang_tap_dance_ta_each (qk_tap_dance_state_t *state, void *user_data) {
  115. td_ta_state_t *td_ta = (td_ta_state_t *) user_data;
  116. if (!td_ta->layer_toggle) { // Braces added for clarity
  117. unregister_code (KC_LGUI);
  118. }
  119. }
  120. static void ang_tap_dance_ta_reset (qk_tap_dance_state_t *state, void *user_data) {
  121. td_ta_state_t *td_ta = (td_ta_state_t *) user_data;
  122. if (!td_ta->layer_toggle) { // Braces added for clarity
  123. unregister_code (KC_LGUI);
  124. }
  125. // if (!td_ta->sticky)
  126. layer_off (GUILAY); // We don't verify it was swithed on, switching off regardless
  127. }
  128. qk_tap_dance_action_t tap_dance_actions[] = {
  129. [0] = {
  130. .fn = { ang_tap_dance_ta_each, ang_tap_dance_ta_finished, ang_tap_dance_ta_reset },
  131. .user_data = (void *)&((td_ta_state_t) { false, false })
  132. }
  133. };
  134. #if defined ISOLIKE_ALT_ENTER || defined TWO_SFT_CAPS || defined APP_IS_RALT_RCTRL || defined ALT_MINSEQL_IS_ALT_78
  135. // Function for the special modifiers actions below, makes it cleaner and yields smaller firmware.
  136. static bool special_mods(uint16_t keycode, keyrecord_t *record, uint16_t modifier) {
  137. if (record->event.pressed && (keyboard_report->mods & MOD_BIT(modifier))) {
  138. register_code(keycode);
  139. return false;
  140. } else {
  141. unregister_code(keycode);
  142. return true;
  143. }
  144. }
  145. #endif
  146. #ifdef BSPC_BLOCKS_DEL
  147. static bool del_blocked = false; // Static as to not be defined elsewhere
  148. #endif
  149. // This function is processed before the key events on each key press/release.
  150. bool process_record_user(uint16_t keycode, keyrecord_t *record) {
  151. switch (keycode){
  152. #ifdef ISOLIKE_ALT_ENTER
  153. // RALT + ENT is the same as RALT+BSLS.
  154. // For logical placement of *{* and *}* (the former being RALT+BSLS) with ANSI enter on
  155. // ISO-based Canadian Multilingual layout (or any other ISO layout on ANSI keyboards).
  156. case KC_ENT: return special_mods(KC_BSLS, record, KC_RALT); // RALT + ENT -> RALT + BSLS // See comment above
  157. #endif
  158. #ifdef TWO_SFT_CAPS
  159. case KC_LSFT: return special_mods(KC_CAPS, record, KC_RSFT); // Both shifts together -> Caps Lock // RSFT pressed first case
  160. case KC_RSFT: return special_mods(KC_CAPS, record, KC_LSFT); // Both shifts together -> Caps Lock // LSFT pressed first case
  161. #endif
  162. #ifdef APP_IS_RALT_RCTRL
  163. case KC_RCTRL: return special_mods(KC_APP, record, KC_RALT); // RALT + RCTRL -> APP // RCTRL pressed first case
  164. case KC_RALT: return special_mods(KC_APP, record, KC_RCTRL); // RALT + RCTRL -> APP // RALT pressed first case
  165. #endif
  166. #ifdef ALT_MINSEQL_IS_ALT_78
  167. case KC_MINS: return special_mods(KC_7, record, KC_RALT); // RALT + MINS -> RALT+7 // {} in CAN Mult. softawre layout
  168. case KC_EQL: return special_mods(KC_8, record, KC_RALT); // RALT + EQL -> RALT+8 // ½¬ normally... Less finger stretch.
  169. #endif
  170. #ifdef HELD_ESC_IS_SHIFT
  171. case KC_ESC: // Physically *ESC* is *CAPS* // Holding ESC -> SHIFT (0 delay) // Less awkward *<ESC>:wq* in vim
  172. if (record->event.pressed) {
  173. register_code(KC_ESC); // Tapping ESC
  174. unregister_code(KC_ESC);
  175. register_code(KC_LSFT); // Holding LSFT
  176. return false;
  177. } else {
  178. unregister_code(KC_LSFT); // Releasing LSFT
  179. return false;
  180. }
  181. #endif
  182. #ifdef BSPC_BLOCKS_DEL
  183. // If BSPC is held, we flag DEL as disabled. To avoids acidental presses of DEL with split backspace key.
  184. case KC_BSPC:
  185. del_blocked = record->event.pressed;
  186. return true;
  187. // We don't handle DEL if it is pressed and flagged as disabled
  188. case KC_DEL:
  189. if (del_blocked && record->event.pressed) {
  190. return false;
  191. } else {
  192. return true;
  193. }
  194. #endif
  195. default: return true; // Let QMK handle the rest as usual
  196. }
  197. }