200 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Copyright 2022 Jouke Witteveen
 | |
|  *
 | |
|  * This program is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU General Public License as published by
 | |
|  * the Free Software Foundation, either version 2 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include "quantum.h"
 | |
| #include "examples.h"
 | |
| 
 | |
| // Example code from the tap dance documentation, adapted for testing
 | |
| 
 | |
| // clang-format off
 | |
| 
 | |
| // Example 1
 | |
| 
 | |
| void dance_egg(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     if (state->count >= 100) {
 | |
|         // SEND_STRING("Safety dance!");
 | |
|         tap_code(KC_C);
 | |
|         reset_tap_dance(state);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| // Example 2
 | |
| 
 | |
| void dance_flsh_each(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     switch (state->count) {
 | |
|         case 1:
 | |
|             register_code(KC_3);
 | |
|             break;
 | |
|         case 2:
 | |
|             register_code(KC_2);
 | |
|             break;
 | |
|         case 3:
 | |
|             register_code(KC_1);
 | |
|             break;
 | |
|         case 4:
 | |
|             unregister_code(KC_3);
 | |
|             // wait_ms(50);
 | |
|             unregister_code(KC_2);
 | |
|             // wait_ms(50);
 | |
|             unregister_code(KC_1);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void dance_flsh_finished(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     if (state->count >= 4) {
 | |
|         // reset_keyboard();
 | |
|         tap_code(KC_R);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void dance_flsh_reset(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     unregister_code(KC_1);
 | |
|     // wait_ms(50);
 | |
|     unregister_code(KC_2);
 | |
|     // wait_ms(50);
 | |
|     unregister_code(KC_3);
 | |
| }
 | |
| 
 | |
| 
 | |
| // Example 3
 | |
| 
 | |
| typedef struct {
 | |
|     uint16_t tap;
 | |
|     uint16_t hold;
 | |
|     uint16_t held;
 | |
| } tap_dance_tap_hold_t;
 | |
| 
 | |
| bool process_record_user(uint16_t keycode, keyrecord_t *record) {
 | |
|     qk_tap_dance_action_t *action;
 | |
| 
 | |
|     switch (keycode) {
 | |
|         case TD(CT_CLN):
 | |
|             action = &tap_dance_actions[TD_INDEX(keycode)];
 | |
|             if (!record->event.pressed && action->state.count && !action->state.finished) {
 | |
|                 tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)action->user_data;
 | |
|                 tap_code16(tap_hold->tap);
 | |
|             }
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void tap_dance_tap_hold_finished(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)user_data;
 | |
| 
 | |
|     if (state->pressed) {
 | |
|         if (state->count == 1
 | |
| #ifndef PERMISSIVE_HOLD
 | |
|             && !state->interrupted
 | |
| #endif
 | |
|         ) {
 | |
|             register_code16(tap_hold->hold);
 | |
|             tap_hold->held = tap_hold->hold;
 | |
|         } else {
 | |
|             register_code16(tap_hold->tap);
 | |
|             tap_hold->held = tap_hold->tap;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void tap_dance_tap_hold_reset(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     tap_dance_tap_hold_t *tap_hold = (tap_dance_tap_hold_t *)user_data;
 | |
| 
 | |
|     if (tap_hold->held) {
 | |
|         unregister_code16(tap_hold->held);
 | |
|         tap_hold->held = 0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| #define ACTION_TAP_DANCE_TAP_HOLD(tap, hold) \
 | |
|     { .fn = {NULL, tap_dance_tap_hold_finished, tap_dance_tap_hold_reset}, .user_data = (void *)&((tap_dance_tap_hold_t){tap, hold, 0}), }
 | |
| 
 | |
| 
 | |
| // Example 4
 | |
| 
 | |
| typedef enum {
 | |
|     TD_NONE,
 | |
|     TD_UNKNOWN,
 | |
|     TD_SINGLE_TAP,
 | |
|     TD_SINGLE_HOLD,
 | |
|     TD_DOUBLE_TAP,
 | |
|     TD_DOUBLE_HOLD,
 | |
|     TD_DOUBLE_SINGLE_TAP,
 | |
|     TD_TRIPLE_TAP,
 | |
|     TD_TRIPLE_HOLD
 | |
| } td_state_t;
 | |
| 
 | |
| typedef struct {
 | |
|     bool is_press_action;
 | |
|     td_state_t state;
 | |
| } td_tap_t;
 | |
| 
 | |
| td_state_t cur_dance(qk_tap_dance_state_t *state) {
 | |
|     if (state->count == 1) {
 | |
|         if (state->interrupted || !state->pressed) return TD_SINGLE_TAP;
 | |
|         else return TD_SINGLE_HOLD;
 | |
|     } else if (state->count == 2) {
 | |
|         if (state->interrupted) return TD_DOUBLE_SINGLE_TAP;
 | |
|         else if (state->pressed) return TD_DOUBLE_HOLD;
 | |
|         else return TD_DOUBLE_TAP;
 | |
|     }
 | |
| 
 | |
|     if (state->count == 3) {
 | |
|         if (state->interrupted || !state->pressed) return TD_TRIPLE_TAP;
 | |
|         else return TD_TRIPLE_HOLD;
 | |
|     } else return TD_UNKNOWN;
 | |
| }
 | |
| 
 | |
| static td_tap_t xtap_state = {
 | |
|     .is_press_action = true,
 | |
|     .state = TD_NONE
 | |
| };
 | |
| 
 | |
| void x_finished(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     xtap_state.state = cur_dance(state);
 | |
|     switch (xtap_state.state) {
 | |
|         case TD_SINGLE_TAP: register_code(KC_X); break;
 | |
|         case TD_SINGLE_HOLD: register_code(KC_LCTL); break;
 | |
|         case TD_DOUBLE_TAP: register_code(KC_ESC); break;
 | |
|         case TD_DOUBLE_HOLD: register_code(KC_LALT); break;
 | |
|         case TD_DOUBLE_SINGLE_TAP: tap_code(KC_X); register_code(KC_X);
 | |
|         default: break;  // Not present in documentation
 | |
|     }
 | |
| }
 | |
| 
 | |
| void x_reset(qk_tap_dance_state_t *state, void *user_data) {
 | |
|     switch (xtap_state.state) {
 | |
|         case TD_SINGLE_TAP: unregister_code(KC_X); break;
 | |
|         case TD_SINGLE_HOLD: unregister_code(KC_LCTL); break;
 | |
|         case TD_DOUBLE_TAP: unregister_code(KC_ESC); break;
 | |
|         case TD_DOUBLE_HOLD: unregister_code(KC_LALT);
 | |
|         case TD_DOUBLE_SINGLE_TAP: unregister_code(KC_X);
 | |
|         default: break;  // Not present in documentation
 | |
|     }
 | |
|     xtap_state.state = TD_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| qk_tap_dance_action_t tap_dance_actions[] = {
 | |
|     [TD_ESC_CAPS] = ACTION_TAP_DANCE_DOUBLE(KC_ESC, KC_CAPS),
 | |
|     [CT_EGG]      = ACTION_TAP_DANCE_FN(dance_egg),
 | |
|     [CT_FLSH]     = ACTION_TAP_DANCE_FN_ADVANCED(dance_flsh_each, dance_flsh_finished, dance_flsh_reset),
 | |
|     [CT_CLN]      = ACTION_TAP_DANCE_TAP_HOLD(KC_COLN, KC_SCLN),
 | |
|     [X_CTL]       = ACTION_TAP_DANCE_FN_ADVANCED(NULL, x_finished, x_reset)
 | |
| };
 | |
| 
 | |
| // clang-format on
 |