145 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|   Copyright (c) 2020 Fred Silberberg
 | |
| 
 | |
|   Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|   of this software and associated documentation files (the "Software"), to deal
 | |
|   in the Software without restriction, including without limitation the rights
 | |
|   to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
|   copies of the Software, and to permit persons to whom the Software is
 | |
|   furnished to do so, subject to the following conditions:
 | |
| 
 | |
|   The above copyright notice and this permission notice shall be included in all
 | |
|   copies or substantial portions of the Software.
 | |
| 
 | |
|   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|   IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|   AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|   LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
|   OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | |
|   THE SOFTWARE.
 | |
| */
 | |
| 
 | |
| #include "333fred.h"
 | |
| #include "quantum.h"
 | |
| #include "action.h"
 | |
| 
 | |
| typedef enum {
 | |
|     SINGLE_TAP, SINGLE_HOLD, DOUBLE
 | |
| } tap_dance_state_enum;
 | |
| 
 | |
| static tap_dance_state_enum tap_dance_state;
 | |
| static bool tap_dance_active = false;
 | |
| 
 | |
| void tap_dance_sym_vim_finished(tap_dance_state_t *state, void *user_data) {
 | |
|     // Determine the current state
 | |
|     if (state->count == 1) {
 | |
|         if (state->interrupted || state->pressed == 0) tap_dance_state = SINGLE_TAP;
 | |
|         else tap_dance_state = SINGLE_HOLD;
 | |
|     } else {
 | |
|         // Handle any number of other taps as a VIM movement hold
 | |
|         tap_dance_state = DOUBLE;
 | |
|     }
 | |
| 
 | |
|     switch (tap_dance_state) {
 | |
|         case SINGLE_TAP:
 | |
|             if (tap_dance_active) {
 | |
|                 reset_oneshot_layer();
 | |
|                 tap_dance_active = false;
 | |
|             } else {
 | |
|                 set_oneshot_layer(SYMB, ONESHOT_START);
 | |
|                 tap_dance_active = true;
 | |
|             }
 | |
|             break;
 | |
|         case SINGLE_HOLD:
 | |
|             layer_on(SYMB);
 | |
|             break;
 | |
|         case DOUBLE:
 | |
|             layer_on(VIM);
 | |
|             break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void tap_dance_sym_vim_reset(tap_dance_state_t *state, void *user_data) {
 | |
|     switch(tap_dance_state) {
 | |
|         case SINGLE_TAP:
 | |
|             clear_oneshot_layer_state(ONESHOT_PRESSED);
 | |
|             break;
 | |
|         case SINGLE_HOLD:
 | |
|             layer_off(SYMB);
 | |
|             break;
 | |
|         case DOUBLE:
 | |
|             layer_off(VIM);
 | |
|             break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void tap_dance_copy_paste_finished(tap_dance_state_t *state, void *user_data) {
 | |
|     bool is_paste = state->count == 2;
 | |
|     // If either the one-shot shift is set, or if shift is being held, count as shift being held.
 | |
|     // We'll clear the one-shot shift if it was held
 | |
|     uint8_t one_shot_mods = get_oneshot_mods();
 | |
|     bool is_shift = false;
 | |
| 
 | |
|     if (get_mods() & MOD_MASK_SHIFT) {
 | |
|         is_shift = true;
 | |
|     } else if (one_shot_mods & MOD_MASK_SHIFT) {
 | |
|         set_oneshot_mods(one_shot_mods & ~MOD_MASK_SHIFT);
 | |
|         is_shift = true;
 | |
|     }
 | |
| 
 | |
|     if (is_paste) {
 | |
|         if (is_shift) {
 | |
|             SEND_STRING(SS_LSFT(SS_TAP(X_INSERT)));
 | |
|         } else {
 | |
|             SEND_STRING(SS_LCTL("v"));
 | |
|         }
 | |
|     } else {
 | |
|         if (is_shift) {
 | |
|             SEND_STRING(SS_LCTL(SS_TAP(X_INSERT)));
 | |
|         } else {
 | |
|             SEND_STRING(SS_LCTL("c"));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| tap_dance_action_t tap_dance_actions[] = {
 | |
|     [TD_SYM_VIM] = ACTION_TAP_DANCE_FN_ADVANCED(NULL, tap_dance_sym_vim_finished, tap_dance_sym_vim_reset),
 | |
|     [TD_COPY_PASTE] = ACTION_TAP_DANCE_FN_ADVANCED(NULL, tap_dance_copy_paste_finished, NULL)
 | |
| };
 | |
| 
 | |
| void tap_dance_process_keycode(uint16_t keycode) {
 | |
|     if (tap_dance_state == SINGLE_TAP && keycode != TD(TD_SYM_VIM)) {
 | |
|         tap_dance_active = false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| __attribute__ ((weak))
 | |
| void layer_state_set_rgb(layer_state_t state) {}
 | |
| 
 | |
| layer_state_t layer_state_set_user(layer_state_t state) {
 | |
|   layer_state_set_rgb(state);
 | |
|   return state;
 | |
| }
 | |
| 
 | |
| bool try_handle_macro(uint16_t keycode, keyrecord_t *record) {
 | |
|     switch (keycode)
 | |
|     {
 | |
|         case DLEFT:
 | |
|             if (record->event.pressed)
 | |
|                 SEND_STRING(SS_LGUI(SS_LALT(SS_TAP(X_LEFT))));
 | |
|             return true;
 | |
|         case DRIGHT:
 | |
|             if (record->event.pressed)
 | |
|                 SEND_STRING(SS_LGUI(SS_LALT(SS_TAP(X_RIGHT))));
 | |
|             return true;
 | |
|         case PSCREEN_APP:
 | |
|             if (record->event.pressed)
 | |
|                 SEND_STRING(SS_LALT(SS_TAP(X_PRINT_SCREEN)));
 | |
|             return true;
 | |
| 
 | |
|         default:
 | |
|             return false;
 | |
|     }
 | |
| }
 |