logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

examples.c (6613B)


  1. /* Copyright 2022 Jouke Witteveen
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include "quantum.h"
  17. #include "examples.h"
  18. #include "keymap_introspection.h"
  19. // Example code from the tap dance documentation, adapted for testing
  20. // clang-format off
  21. // Example 1
  22. void dance_egg(tap_dance_state_t *state, void *user_data) {
  23. if (state->count >= 100) {
  24. // SEND_STRING("Safety dance!");
  25. tap_code(KC_C);
  26. reset_tap_dance(state);
  27. }
  28. }
  29. // Example 2
  30. void dance_flsh_each(tap_dance_state_t *state, void *user_data) {
  31. switch (state->count) {
  32. case 1:
  33. register_code(KC_3);
  34. break;
  35. case 2:
  36. register_code(KC_2);
  37. break;
  38. case 3:
  39. register_code(KC_1);
  40. break;
  41. case 4:
  42. unregister_code(KC_3);
  43. // wait_ms(50);
  44. unregister_code(KC_2);
  45. // wait_ms(50);
  46. unregister_code(KC_1);
  47. }
  48. }
  49. void dance_flsh_finished(tap_dance_state_t *state, void *user_data) {
  50. if (state->count >= 4) {
  51. // reset_keyboard();
  52. tap_code(KC_R);
  53. }
  54. }
  55. void dance_flsh_reset(tap_dance_state_t *state, void *user_data) {
  56. unregister_code(KC_1);
  57. // wait_ms(50);
  58. unregister_code(KC_2);
  59. // wait_ms(50);
  60. unregister_code(KC_3);
  61. }
  62. // Example 3
  63. typedef struct {
  64. uint16_t tap;
  65. uint16_t hold;
  66. uint16_t held;
  67. } tap_dance_tap_hold_t;
  68. bool process_record_user(uint16_t keycode, keyrecord_t *record) {
  69. tap_dance_action_t *action;
  70. switch (keycode) {
  71. case TD(CT_CLN):
  72. action = tap_dance_get(QK_TAP_DANCE_GET_INDEX(keycode));
  73. if (!record->event.pressed && action->state.count && !action->state.finished) {
  74. tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)action->user_data;
  75. tap_code16(tap_hold->tap);
  76. }
  77. }
  78. return true;
  79. }
  80. void tap_dance_tap_hold_finished(tap_dance_state_t *state, void *user_data) {
  81. tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)user_data;
  82. if (state->pressed) {
  83. if (state->count == 1
  84. #ifndef PERMISSIVE_HOLD
  85. && !state->interrupted
  86. #endif
  87. ) {
  88. register_code16(tap_hold->hold);
  89. tap_hold->held = tap_hold->hold;
  90. } else {
  91. register_code16(tap_hold->tap);
  92. tap_hold->held = tap_hold->tap;
  93. }
  94. }
  95. }
  96. void tap_dance_tap_hold_reset(tap_dance_state_t *state, void *user_data) {
  97. tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)user_data;
  98. if (tap_hold->held) {
  99. unregister_code16(tap_hold->held);
  100. tap_hold->held = 0;
  101. }
  102. }
  103. #define ACTION_TAP_DANCE_TAP_HOLD(tap, hold) \
  104. { .fn = {NULL, tap_dance_tap_hold_finished, tap_dance_tap_hold_reset}, .user_data = (void *)&((tap_dance_tap_hold_t){tap, hold, 0}), }
  105. // Example 4
  106. typedef enum {
  107. TD_NONE,
  108. TD_UNKNOWN,
  109. TD_SINGLE_TAP,
  110. TD_SINGLE_HOLD,
  111. TD_DOUBLE_TAP,
  112. TD_DOUBLE_HOLD,
  113. TD_DOUBLE_SINGLE_TAP,
  114. TD_TRIPLE_TAP,
  115. TD_TRIPLE_HOLD
  116. } td_state_t;
  117. typedef struct {
  118. bool is_press_action;
  119. td_state_t state;
  120. } td_tap_t;
  121. td_state_t cur_dance(tap_dance_state_t *state) {
  122. if (state->count == 1) {
  123. if (state->interrupted || !state->pressed) return TD_SINGLE_TAP;
  124. else return TD_SINGLE_HOLD;
  125. } else if (state->count == 2) {
  126. if (state->interrupted) return TD_DOUBLE_SINGLE_TAP;
  127. else if (state->pressed) return TD_DOUBLE_HOLD;
  128. else return TD_DOUBLE_TAP;
  129. }
  130. if (state->count == 3) {
  131. if (state->interrupted || !state->pressed) return TD_TRIPLE_TAP;
  132. else return TD_TRIPLE_HOLD;
  133. } else return TD_UNKNOWN;
  134. }
  135. static td_tap_t xtap_state = {
  136. .is_press_action = true,
  137. .state = TD_NONE
  138. };
  139. void x_finished(tap_dance_state_t *state, void *user_data) {
  140. xtap_state.state = cur_dance(state);
  141. switch (xtap_state.state) {
  142. case TD_SINGLE_TAP: register_code(KC_X); break;
  143. case TD_SINGLE_HOLD: register_code(KC_LCTL); break;
  144. case TD_DOUBLE_TAP: register_code(KC_ESC); break;
  145. case TD_DOUBLE_HOLD: register_code(KC_LALT); break;
  146. case TD_DOUBLE_SINGLE_TAP: tap_code(KC_X); register_code(KC_X);
  147. default: break; // Not present in documentation
  148. }
  149. }
  150. void x_reset(tap_dance_state_t *state, void *user_data) {
  151. switch (xtap_state.state) {
  152. case TD_SINGLE_TAP: unregister_code(KC_X); break;
  153. case TD_SINGLE_HOLD: unregister_code(KC_LCTL); break;
  154. case TD_DOUBLE_TAP: unregister_code(KC_ESC); break;
  155. case TD_DOUBLE_HOLD: unregister_code(KC_LALT);
  156. case TD_DOUBLE_SINGLE_TAP: unregister_code(KC_X);
  157. default: break; // Not present in documentation
  158. }
  159. xtap_state.state = TD_NONE;
  160. }
  161. static void release_press(tap_dance_state_t *state, void *user_data) {
  162. tap_code16(KC_P);
  163. }
  164. static void release_unpress(tap_dance_state_t *state, void *user_data) {
  165. tap_code16(KC_U);
  166. }
  167. static void release_unpress_mark_finished(tap_dance_state_t *state, void *user_data) {
  168. tap_code16(KC_U);
  169. state->finished = true;
  170. }
  171. static void release_finished(tap_dance_state_t *state, void *user_data) {
  172. tap_code16(KC_F);
  173. }
  174. static void release_reset(tap_dance_state_t *state, void *user_data) {
  175. tap_code16(KC_R);
  176. }
  177. tap_dance_action_t tap_dance_actions[] = {
  178. [TD_ESC_CAPS] = ACTION_TAP_DANCE_DOUBLE(KC_ESC, KC_CAPS),
  179. [CT_EGG] = ACTION_TAP_DANCE_FN(dance_egg),
  180. [CT_FLSH] = ACTION_TAP_DANCE_FN_ADVANCED(dance_flsh_each, dance_flsh_finished, dance_flsh_reset),
  181. [CT_CLN] = ACTION_TAP_DANCE_TAP_HOLD(KC_COLN, KC_SCLN),
  182. [X_CTL] = ACTION_TAP_DANCE_FN_ADVANCED(NULL, x_finished, x_reset),
  183. [TD_RELEASE] = ACTION_TAP_DANCE_FN_ADVANCED_WITH_RELEASE(release_press, release_unpress, release_finished, release_reset),
  184. [TD_RELEASE_AND_FINISH] = ACTION_TAP_DANCE_FN_ADVANCED_WITH_RELEASE(release_press, release_unpress_mark_finished, release_finished, release_reset),
  185. };
  186. // clang-format on