diff options
| author | Tim Schumacher <twschum@gmail.com> | 2019-11-21 06:07:33 -0500 |
|---|---|---|
| committer | Drashna Jaelre <drashna@live.com> | 2019-11-21 03:07:33 -0800 |
| commit | 84065e1d7458ad6b0a04a82fed4dfc69a4ded2b2 (patch) | |
| tree | c3985a31591fb3f2c3252caf49d7751cd5d8cb10 /users | |
| parent | 90bb7db48e658624c6c4609c832ac1451e4db98b (diff) | |
| download | qmk_firmware-84065e1d7458ad6b0a04a82fed4dfc69a4ded2b2.tar.gz qmk_firmware-84065e1d7458ad6b0a04a82fed4dfc69a4ded2b2.zip | |
[Keymap] Add DZ60, Levinson keymaps, twschum user files (#7358)
* Adds layout for DZ60rev2 plate B, options 4 and 10
* Use KC_TRANS for layer keys
* Format layer0 with visual key size layout
* Add RGB controls; add start description; add KC_LOCK
* Update comments on rules.mk for DZ60
* Gets keymap compiling
* Fix wiring for shift on row 3
* Hold a with other key to send Ctrl-A before other key
* Adds compile-time defined mac-compatability media keycodes
* Adds logic in place for capturing taps (w/ timeout)
* Add send_keys(...) which can send up to 64 nested keycodes
* Implement send_keys callable with n repeats
* Tweaks some of the keymap
* Add reset keycode; add disable to custom shortcuts
* Adds a special "off" layer where bonus mod goes to layer 0, every other key KC_NO
* Adjust timeout; mousekey settings
* Changes layout of Home,End,PgUp/Dn on the dedicated arrow keys L1/L2
* PoC on rgb lighted layer indication
* Refactors color table defines
* Adds logic for controlling layer lights
* Only change state on one side of the event lol
* Switch back to Tapping Toggle for layer 4
* Add custom config file for keyboard; TT and mousekey settings
* Code cleanup; starting to refactor special ctrl tapping keys functions
* Move defines and reak out functions
* Remove debugging light
* Adjust keymaps; add enter
* Adds a couple custom macros
* Add simplified version of keymap to help debug issues
* Adds basi numpad configuration for levinson keyboard half
* Use ANSI ctrl key as layer 1 for better one hand (`)
* Adapt to new 8bit hue from #5547; layer enum use everywhere
* Move custom code out to users/ space
* Flip col pins, move key assignments to "left hand"
* Update readme
* Implement placeholder macros
* Notes
* Reduce tapping time for SFT_T(/)
* Adds vim features; refactors things
* Adds fork of the vim features written by xton
* Use correct backspace keycode
* Add VIM_ESC
* Add "OFF" to the RGB/HSV definitions
* Clean up rules, use new "OFF" definition
* Add windows KC_CALCULATOR key to numpad
* Reformat layers with better guide; change layer names
* Add sleep key
* Change timeout delay
* Add a "code paste" which surrounds a ^V with ```
* Try removing shift tap on rshift /
* Update macros
* Update the "code paste" macro
* Update keymap with reset, calc, equals
* Update keyboards/dz60/keymaps/twschum_b_4_10_simple/keymap.c
Co-Authored-By: Drashna Jaelre <drashna@live.com>
* Update keyboards/dz60/keymaps/twschum_b_4_10/keymap.c
Co-Authored-By: Drashna Jaelre <drashna@live.com>
* Update users/twschum/twschum.c
Co-Authored-By: Drashna Jaelre <drashna@live.com>
* Update users/twschum/twschum.c
Co-Authored-By: Drashna Jaelre <drashna@live.com>
* Revert quantum/rgblight_list.h
* Better name for split space layout; rename KEYMAP->LAYOUT
* Rename KEYMAP->LAYOUT
* Use simpler :flash command
* Clean up layout files
* Use qmk's MOD_MASK_CTRL
* Sync lufa submodule
* Cleanup from noroadsleft's PR comments
* Modernize vim layer code
Diffstat (limited to 'users')
| -rw-r--r-- | users/twschum/config.h | 0 | ||||
| -rw-r--r-- | users/twschum/readme.md | 14 | ||||
| -rw-r--r-- | users/twschum/rules.mk | 5 | ||||
| -rw-r--r-- | users/twschum/twschum.c | 257 | ||||
| -rw-r--r-- | users/twschum/twschum.h | 131 | ||||
| -rw-r--r-- | users/twschum/xtonhasvim.c | 593 | ||||
| -rw-r--r-- | users/twschum/xtonhasvim.h | 62 |
7 files changed, 1062 insertions, 0 deletions
diff --git a/users/twschum/config.h b/users/twschum/config.h new file mode 100644 index 000000000..e69de29bb --- /dev/null +++ b/users/twschum/config.h | |||
diff --git a/users/twschum/readme.md b/users/twschum/readme.md new file mode 100644 index 000000000..b354e4b79 --- /dev/null +++ b/users/twschum/readme.md | |||
| @@ -0,0 +1,14 @@ | |||
| 1 | Copyright 2019 Tim Schumacher <twschum@gmail.com> @twschum | ||
| 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/>. | ||
diff --git a/users/twschum/rules.mk b/users/twschum/rules.mk new file mode 100644 index 000000000..9878e6f69 --- /dev/null +++ b/users/twschum/rules.mk | |||
| @@ -0,0 +1,5 @@ | |||
| 1 | SRC += twschum.c | ||
| 2 | SRC += xtonhasvim.c | ||
| 3 | ifeq ($(strip $(FLASH_BOOTLOADER)), yes) | ||
| 4 | OPT_DEFS += -DFLASH_BOOTLOADER | ||
| 5 | endif | ||
diff --git a/users/twschum/twschum.c b/users/twschum/twschum.c new file mode 100644 index 000000000..2d34f9571 --- /dev/null +++ b/users/twschum/twschum.c | |||
| @@ -0,0 +1,257 @@ | |||
| 1 | #include "twschum.h" | ||
| 2 | |||
| 3 | #ifdef TWSCHUM_TAPPING_CTRL_PREFIX | ||
| 4 | // state for the great state machine of custom actions! | ||
| 5 | #define TIMEOUT_DELAY 200 // ms | ||
| 6 | static uint16_t idle_timer; | ||
| 7 | static bool timeout_is_active = false; | ||
| 8 | |||
| 9 | static bool ctrl_shortcuts_enabled_g = false; | ||
| 10 | //static bool B_down = 0; // TODO just use top bit from count | ||
| 11 | //static int8_t B_count = 0; | ||
| 12 | |||
| 13 | #define N_TAPPING_CTRL_KEYS 2 | ||
| 14 | static struct Tapping_ctrl_key_t special_keys_g[N_TAPPING_CTRL_KEYS] = { | ||
| 15 | {false, 0, KC_B}, {false, 0, KC_A} | ||
| 16 | }; | ||
| 17 | |||
| 18 | static inline void start_idle_timer(void) { | ||
| 19 | idle_timer = timer_read(); | ||
| 20 | timeout_is_active = true; | ||
| 21 | } | ||
| 22 | static inline void clear_state_after_idle_timeout(void) { | ||
| 23 | idle_timer = 0; | ||
| 24 | timeout_is_active = false; | ||
| 25 | |||
| 26 | // send timed out plain keys from tapping ctrl mod | ||
| 27 | for (int i = 0; i < N_TAPPING_CTRL_KEYS; ++i) { | ||
| 28 | struct Tapping_ctrl_key_t* key = special_keys_g + i; | ||
| 29 | repeat_send_keys(key->count, key->keycode); | ||
| 30 | key->count = 0; | ||
| 31 | } | ||
| 32 | } | ||
| 33 | |||
| 34 | inline void matrix_scan_user(void) { | ||
| 35 | if (timeout_is_active && timer_elapsed(idle_timer) > TIMEOUT_DELAY) { | ||
| 36 | clear_state_after_idle_timeout(); | ||
| 37 | } | ||
| 38 | } | ||
| 39 | |||
| 40 | static inline bool tap_ctrl_event(struct Tapping_ctrl_key_t* key, keyrecord_t* record) { | ||
| 41 | if (!ctrl_shortcuts_enabled_g) { | ||
| 42 | // normal operation, just send the plain keycode | ||
| 43 | if (record->event.pressed) { | ||
| 44 | register_code(key->keycode); | ||
| 45 | } | ||
| 46 | else { | ||
| 47 | unregister_code(key->keycode); | ||
| 48 | } | ||
| 49 | return false; | ||
| 50 | } | ||
| 51 | key->down = record->event.pressed; | ||
| 52 | // increment count and reset timer when key pressed | ||
| 53 | // start the timeout when released | ||
| 54 | if (key->down) { | ||
| 55 | ++(key->count); | ||
| 56 | timeout_is_active = false; | ||
| 57 | idle_timer = 0; | ||
| 58 | } | ||
| 59 | else { | ||
| 60 | if (key->count) { | ||
| 61 | start_idle_timer(); | ||
| 62 | } | ||
| 63 | } | ||
| 64 | return false; | ||
| 65 | } | ||
| 66 | |||
| 67 | static inline bool tap_ctrl_other_pressed(void) { | ||
| 68 | for (int i = 0; i < N_TAPPING_CTRL_KEYS; ++i) { | ||
| 69 | struct Tapping_ctrl_key_t* key = special_keys_g + i; | ||
| 70 | if (key->count) { | ||
| 71 | if (key->down) { | ||
| 72 | // another key has been pressed while the leader key is down, | ||
| 73 | // so send number of ctrl-KEY combos before the other key | ||
| 74 | repeat_send_keys(key->count, KC_LCTL, key->keycode); | ||
| 75 | key->count = 0; | ||
| 76 | } | ||
| 77 | else { | ||
| 78 | // another key pressed after leader key released, | ||
| 79 | // need to send the plain keycode plus potential mods | ||
| 80 | if (get_mods() & MOD_MASK_CTRL) { | ||
| 81 | // make sure to send a shift if prssed | ||
| 82 | repeat_send_keys(key->count, KC_RSHIFT, key->keycode); | ||
| 83 | } | ||
| 84 | else { | ||
| 85 | repeat_send_keys(key->count, key->keycode); | ||
| 86 | } | ||
| 87 | key->count = 0; | ||
| 88 | } | ||
| 89 | return true; // will send the other keycode | ||
| 90 | } | ||
| 91 | } | ||
| 92 | return true; // safe default | ||
| 93 | } | ||
| 94 | #endif /* TWSCHUM_TAPPING_CTRL_PREFIX */ | ||
| 95 | |||
| 96 | |||
| 97 | /* Use RGB underglow to indicate layer | ||
| 98 | * https://docs.qmk.fm/reference/customizing-functionality | ||
| 99 | */ | ||
| 100 | // add to quantum/rgblight_list.h | ||
| 101 | #ifdef RGBLIGHT_ENABLE | ||
| 102 | static bool rgb_layers_enabled = true; | ||
| 103 | static bool rgb_L0_enabled = false; | ||
| 104 | |||
| 105 | layer_state_t layer_state_set_user(layer_state_t state) { | ||
| 106 | if (!rgb_layers_enabled) { | ||
| 107 | return state; | ||
| 108 | } | ||
| 109 | switch (get_highest_layer(state)) { | ||
| 110 | case _Base: | ||
| 111 | if (rgb_L0_enabled) { | ||
| 112 | rgblight_sethsv_noeeprom(_Base_HSV_ON); | ||
| 113 | } | ||
| 114 | else { | ||
| 115 | rgblight_sethsv_noeeprom(_Base_HSV_OFF); | ||
| 116 | } | ||
| 117 | break; | ||
| 118 | case _Vim: | ||
| 119 | rgblight_sethsv_noeeprom(_Vim_HSV); | ||
| 120 | break; | ||
| 121 | case _Fn: | ||
| 122 | rgblight_sethsv_noeeprom(_Fn_HSV); | ||
| 123 | break; | ||
| 124 | case _Nav: | ||
| 125 | rgblight_sethsv_noeeprom(_Nav_HSV); | ||
| 126 | break; | ||
| 127 | case _Num: | ||
| 128 | rgblight_sethsv_noeeprom(_Num_HSV); | ||
| 129 | break; | ||
| 130 | case _Cfg: | ||
| 131 | rgblight_sethsv_noeeprom(_Cfg_HSV); | ||
| 132 | break; | ||
| 133 | case _None: | ||
| 134 | rgblight_sethsv_noeeprom(_None_HSV); | ||
| 135 | break; | ||
| 136 | } | ||
| 137 | return state; | ||
| 138 | } | ||
| 139 | #endif /* RGBLIGHT_ENABLE */ | ||
| 140 | |||
| 141 | /* process_record_vimlayer: handles the VIM_ keycodes from xtonhasvim's vim | ||
| 142 | * emulation layer | ||
| 143 | * add process_record_keymap to allow specific keymap to still add keys | ||
| 144 | * Makes the callstack look like: | ||
| 145 | * process_record_ | ||
| 146 | * _quantum | ||
| 147 | * _kb | ||
| 148 | * _user | ||
| 149 | * _keymap | ||
| 150 | * _vimlayer | ||
| 151 | */ | ||
| 152 | __attribute__ ((weak)) | ||
| 153 | bool process_record_keymap(uint16_t keycode, keyrecord_t *record) { | ||
| 154 | return true; | ||
| 155 | } | ||
| 156 | |||
| 157 | /* Return True to continue processing keycode, false to stop further processing | ||
| 158 | * process_record_keymap to be call by process_record_user in the vim addon */ | ||
| 159 | bool process_record_user(uint16_t keycode, keyrecord_t *record) { | ||
| 160 | |||
| 161 | /* keymap gets first whack, then vimlayer */ | ||
| 162 | if(!process_record_keymap(keycode, record)) return false; | ||
| 163 | if(!process_record_vimlayer(keycode, record)) return false; | ||
| 164 | |||
| 165 | switch (keycode) { | ||
| 166 | /* KC_MAKE is a keycode to be used with any keymap | ||
| 167 | * Outputs `make <keyboard>:<keymap>` | ||
| 168 | * Holding shift will add the appropriate flashing command (:dfu, | ||
| 169 | * :teensy, :avrdude, :dfu-util) for a majority of keyboards. | ||
| 170 | * Holding control will add some commands that will speed up compiling | ||
| 171 | * time by processing multiple files at once | ||
| 172 | * For the boards that lack a shift key, or that you want to always | ||
| 173 | * attempt the flashing part, you can add FLASH_BOOTLOADER = yes to the | ||
| 174 | * rules.mk of that keymap. | ||
| 175 | */ | ||
| 176 | case KC_MAKE: // Compiles the firmware, and adds the flash command based on keyboard bootloader | ||
| 177 | if (!record->event.pressed) { | ||
| 178 | uint8_t temp_mod = get_mods(); | ||
| 179 | uint8_t temp_osm = get_oneshot_mods(); | ||
| 180 | clear_mods(); clear_oneshot_mods(); | ||
| 181 | SEND_STRING("make " QMK_KEYBOARD ":" QMK_KEYMAP); | ||
| 182 | #ifndef FLASH_BOOTLOADER | ||
| 183 | if ( (temp_mod | temp_osm) & MOD_MASK_SHIFT ) { | ||
| 184 | SEND_STRING(":flash"); | ||
| 185 | } | ||
| 186 | #endif | ||
| 187 | if ( (temp_mod | temp_osm) & MOD_MASK_CTRL) { | ||
| 188 | SEND_STRING(" -j8 --output-sync"); | ||
| 189 | } | ||
| 190 | SEND_STRING(SS_TAP(X_ENTER)); | ||
| 191 | set_mods(temp_mod); | ||
| 192 | } | ||
| 193 | break; | ||
| 194 | |||
| 195 | #ifdef RGBLIGHT_ENABLE | ||
| 196 | case TG_LAYER_RGB: | ||
| 197 | if (record->event.pressed) { | ||
| 198 | rgb_layers_enabled = !rgb_layers_enabled; | ||
| 199 | } | ||
| 200 | return false; | ||
| 201 | case TG_L0_RGB: | ||
| 202 | if (record->event.pressed) { | ||
| 203 | rgb_L0_enabled = !rgb_L0_enabled; | ||
| 204 | } | ||
| 205 | return false; | ||
| 206 | #endif | ||
| 207 | |||
| 208 | case SALT_CMD: | ||
| 209 | if (!record->event.pressed) { | ||
| 210 | SEND_STRING(SALT_CMD_MACRO); | ||
| 211 | } | ||
| 212 | return false; | ||
| 213 | case LESS_PD: | ||
| 214 | if (!record->event.pressed) { | ||
| 215 | SEND_STRING(LESS_PD_MACRO); | ||
| 216 | } | ||
| 217 | return false; | ||
| 218 | case CODE_PASTE: | ||
| 219 | if (!record->event.pressed) { | ||
| 220 | SEND_STRING(CODE_PASTE_MACRO); | ||
| 221 | } | ||
| 222 | return false; | ||
| 223 | |||
| 224 | #ifdef TWSCHUM_TAPPING_CTRL_PREFIX | ||
| 225 | case EN_CTRL_SHORTCUTS: | ||
| 226 | if (record->event.pressed) { | ||
| 227 | ctrl_shortcuts_enabled_g = !ctrl_shortcuts_enabled_g; | ||
| 228 | start_idle_timer(); // need to clear out state in some cases | ||
| 229 | } | ||
| 230 | return false; | ||
| 231 | case CTRL_A: | ||
| 232 | return tap_ctrl_event(&special_keys_g[1], record); | ||
| 233 | case CTRL_B: | ||
| 234 | return tap_ctrl_event(&special_keys_g[0], record); | ||
| 235 | default: | ||
| 236 | if (record->event.pressed) { | ||
| 237 | return tap_ctrl_other_pressed(); | ||
| 238 | } | ||
| 239 | #endif | ||
| 240 | } | ||
| 241 | return true; | ||
| 242 | } | ||
| 243 | |||
| 244 | #ifdef RGBLIGHT_ENABLE | ||
| 245 | void matrix_init_user(void) { | ||
| 246 | // called once on board init | ||
| 247 | rgblight_enable(); | ||
| 248 | } | ||
| 249 | #endif | ||
| 250 | |||
| 251 | void suspend_power_down_user(void) { | ||
| 252 | // TODO shut off backlighting | ||
| 253 | } | ||
| 254 | |||
| 255 | void suspend_wakeup_init_user(void) { | ||
| 256 | // TODO turn on backlighting | ||
| 257 | } | ||
diff --git a/users/twschum/twschum.h b/users/twschum/twschum.h new file mode 100644 index 000000000..e8c9aeffc --- /dev/null +++ b/users/twschum/twschum.h | |||
| @@ -0,0 +1,131 @@ | |||
| 1 | #pragma once | ||
| 2 | #include <stdarg.h> | ||
| 3 | #include "quantum.h" | ||
| 4 | #include "xtonhasvim.h" | ||
| 5 | |||
| 6 | /************************** | ||
| 7 | * QMK Features Used | ||
| 8 | ************************** | ||
| 9 | * RGBLIGHT_ENABLE | ||
| 10 | * - Adds layer indication via RGB underglow | ||
| 11 | * - see the `layer_definitions` enum and following _*_HSV #defines | ||
| 12 | * | ||
| 13 | * | ||
| 14 | * | ||
| 15 | ************************** | ||
| 16 | * Custom Feature Flags | ||
| 17 | ************************** | ||
| 18 | * | ||
| 19 | * TWSCHUM_TAPPING_CTRL_PREFIX | ||
| 20 | * - Adds feature that makes sending nested sequences of C-a, C-b[, C-b, ...] | ||
| 21 | * as simple as C-a b [b ...] | ||
| 22 | * - Not necessarily super useful outside specialized nested tmux sessions, | ||
| 23 | * but it was a fun state-machine to build | ||
| 24 | * | ||
| 25 | * TWSCHUM_VIM_LAYER | ||
| 26 | * - Fork of xtonhasvim, adding vim-emulation | ||
| 27 | * | ||
| 28 | * TWSCHUM_IS_MAC | ||
| 29 | * - Flag for handling media keys and other settings between OSX and Win/Unix | ||
| 30 | * without having to include bootmagic | ||
| 31 | * | ||
| 32 | ************************** | ||
| 33 | * Features Wishlist | ||
| 34 | ************************** | ||
| 35 | * use VIM_Q as macro recorder! | ||
| 36 | * Dynamic macros | ||
| 37 | * Leader functions | ||
| 38 | * Uniicode leader commands??? (symbolic unicode) | ||
| 39 | * Mac mode vs not: -probably bootmagic or use default with dynamic swap out here | ||
| 40 | * KC_MFFD(KC_MEDIA_FAST_FORWARD) and KC_MRWD(KC_MEDIA_REWIND) instead of KC_MNXT and KC_MPRV | ||
| 41 | */ | ||
| 42 | |||
| 43 | /* Each layer gets a color, overwritable per keyboard */ | ||
| 44 | enum layers_definitions { | ||
| 45 | _Base, | ||
| 46 | _Vim, | ||
| 47 | _Fn, | ||
| 48 | _Nav, | ||
| 49 | _Num, | ||
| 50 | _Cfg, | ||
| 51 | _None, | ||
| 52 | }; | ||
| 53 | #ifdef RGBLIGHT_ENABLE | ||
| 54 | #define _Base_HSV_ON HSV_WHITE | ||
| 55 | #define _Base_HSV_OFF 0, 0, 0 | ||
| 56 | #define _Vim_HSV HSV_ORANGE | ||
| 57 | #define _Fn_HSV HSV_GREEN | ||
| 58 | #define _Nav_HSV HSV_AZURE | ||
| 59 | #define _Num_HSV HSV_GOLD | ||
| 60 | #define _Cfg_HSV HSV_RED | ||
| 61 | #define _None_HSV HSV_WHITE | ||
| 62 | #endif | ||
| 63 | |||
| 64 | enum extra_keycodes { | ||
| 65 | TWSCHUM_START = VIM_SAFE_RANGE, | ||
| 66 | KC_MAKE, // types the make command for this keyboard | ||
| 67 | #ifdef TWSCHUM_TAPPING_CTRL_PREFIX | ||
| 68 | CTRL_A, | ||
| 69 | CTRL_B, | ||
| 70 | EN_CTRL_SHORTCUTS, | ||
| 71 | #endif | ||
| 72 | #ifdef RGBLIGHT_ENABLE | ||
| 73 | TG_LAYER_RGB, // Toggle between standard RGB underglow, and RGB underglow to do layer indication | ||
| 74 | TG_L0_RGB, // Toggle color on or off of layer0 | ||
| 75 | #endif | ||
| 76 | SALT_CMD, // macro | ||
| 77 | LESS_PD, // macro | ||
| 78 | CODE_PASTE, // macro | ||
| 79 | KEYMAP_SAFE_RANGE, // range to start for the keymap | ||
| 80 | }; | ||
| 81 | #define SALT_CMD_MACRO "sudo salt \\* cmd.run ''"SS_TAP(X_LEFT) | ||
| 82 | #define LESS_PD_MACRO "sudo less /pipedream/cache/" | ||
| 83 | // TODO mac vs linux | ||
| 84 | #define CODE_PASTE_MACRO SS_LSFT("\n")"```"SS_LSFT("\n")SS_LALT("v")SS_LSFT("\n")"```" | ||
| 85 | |||
| 86 | |||
| 87 | /* PP_NARG macro returns the number of arguments passed to it. | ||
| 88 | * https://groups.google.com/forum/#!topic/comp.std.c/d-6Mj5Lko_s | ||
| 89 | */ | ||
| 90 | #define PP_NARG(...) PP_NARG_(__VA_ARGS__,PP_RSEQ_N()) | ||
| 91 | #define PP_NARG_(...) PP_ARG_N(__VA_ARGS__) | ||
| 92 | #define PP_MAX_ARGS 64 | ||
| 93 | #define PP_ARG_N( \ | ||
| 94 | _1, _2, _3, _4, _5, _6, _7, _8, _9,_10, \ | ||
| 95 | _11,_12,_13,_14,_15,_16,_17,_18,_19,_20, \ | ||
| 96 | _21,_22,_23,_24,_25,_26,_27,_28,_29,_30, \ | ||
| 97 | _31,_32,_33,_34,_35,_36,_37,_38,_39,_40, \ | ||
| 98 | _41,_42,_43,_44,_45,_46,_47,_48,_49,_50, \ | ||
| 99 | _51,_52,_53,_54,_55,_56,_57,_58,_59,_60, \ | ||
| 100 | _61,_62,_63,N,...) N | ||
| 101 | #define PP_RSEQ_N() 63,62,61,60, \ | ||
| 102 | 59,58,57,56,55,54,53,52,51,50, \ | ||
| 103 | 49,48,47,46,45,44,43,42,41,40, \ | ||
| 104 | 39,38,37,36,35,34,33,32,31,30, \ | ||
| 105 | 29,28,27,26,25,24,23,22,21,20, \ | ||
| 106 | 19,18,17,16,15,14,13,12,11,10, \ | ||
| 107 | 9,8,7,6,5,4,3,2,1,0 | ||
| 108 | |||
| 109 | #define send_keys(...) send_n_keys(PP_NARG(__VA_ARGS__), __VA_ARGS__) | ||
| 110 | static inline void send_n_keys(int n, ...) { | ||
| 111 | uint8_t i = 0; | ||
| 112 | uint16_t keycodes[PP_MAX_ARGS]; | ||
| 113 | va_list keys; | ||
| 114 | va_start(keys, n); | ||
| 115 | for (; i < n; ++i) { | ||
| 116 | keycodes[i] = (uint16_t)va_arg(keys, int); // cast suppresses warning | ||
| 117 | register_code(keycodes[i]); | ||
| 118 | } | ||
| 119 | for (; n > 0; --n) { | ||
| 120 | unregister_code(keycodes[n-1]); | ||
| 121 | } | ||
| 122 | va_end(keys); | ||
| 123 | } | ||
| 124 | #define repeat_send_keys(n, ...) {for (int i=0; i < n; ++i) {send_keys(__VA_ARGS__);}} | ||
| 125 | |||
| 126 | /* State functions for nested c-a & c-b leader keystrokes */ | ||
| 127 | struct Tapping_ctrl_key_t { | ||
| 128 | bool down; | ||
| 129 | int8_t count; | ||
| 130 | const uint16_t keycode; | ||
| 131 | }; | ||
diff --git a/users/twschum/xtonhasvim.c b/users/twschum/xtonhasvim.c new file mode 100644 index 000000000..a1adf39f0 --- /dev/null +++ b/users/twschum/xtonhasvim.c | |||
| @@ -0,0 +1,593 @@ | |||
| 1 | /* Copyright 2015-2017 Christon DeWan * | ||
| 2 | * This program is free software: you can redistribute it and/or modify | ||
| 3 | * it under the terms of the GNU General Public License as published by | ||
| 4 | * the Free Software Foundation, either version 2 of the License, or | ||
| 5 | * (at your option) any later version. | ||
| 6 | * | ||
| 7 | * This program is distributed in the hope that it will be useful, | ||
| 8 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 9 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
| 10 | * GNU General Public License for more details. | ||
| 11 | * | ||
| 12 | * You should have received a copy of the GNU General Public License | ||
| 13 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
| 14 | */ | ||
| 15 | |||
| 16 | #include "xtonhasvim.h" | ||
| 17 | |||
| 18 | |||
| 19 | uint16_t vstate = VIM_START; | ||
| 20 | static bool yank_was_lines = false; | ||
| 21 | static bool SHIFTED = false; | ||
| 22 | static uint32_t mod_override_layer_state = 0; | ||
| 23 | static uint16_t mod_override_triggering_key = 0; | ||
| 24 | |||
| 25 | static void edit(void) { vstate = VIM_START; layer_clear(); } | ||
| 26 | #define EDIT edit() | ||
| 27 | |||
| 28 | |||
| 29 | static void simple_movement(uint16_t keycode) { | ||
| 30 | switch(keycode) { | ||
| 31 | case VIM_B: | ||
| 32 | register_code(KC_LALT); | ||
| 33 | tap_code16(LSFT(KC_LEFT)); // select to start of this word | ||
| 34 | unregister_code(KC_LALT); | ||
| 35 | break; | ||
| 36 | case VIM_E: | ||
| 37 | register_code(KC_LALT); | ||
| 38 | tap_code16(LSFT(KC_RIGHT)); // select to end of this word | ||
| 39 | unregister_code(KC_LALT); | ||
| 40 | break; | ||
| 41 | case VIM_H: | ||
| 42 | tap_code16(LSFT(KC_LEFT)); | ||
| 43 | break; | ||
| 44 | case VIM_J: | ||
| 45 | tap_code16(LGUI(KC_LEFT)); | ||
| 46 | tap_code16(LSFT(KC_DOWN)); | ||
| 47 | tap_code16(LSFT(KC_DOWN)); | ||
| 48 | break; | ||
| 49 | case VIM_K: | ||
| 50 | tap_code16(LGUI(KC_LEFT)); | ||
| 51 | tap_code(KC_DOWN); | ||
| 52 | tap_code16(LSFT(KC_UP)); | ||
| 53 | tap_code16(LSFT(KC_UP)); | ||
| 54 | break; | ||
| 55 | case VIM_L: | ||
| 56 | tap_code16(LSFT(KC_RIGHT)); | ||
| 57 | break; | ||
| 58 | case VIM_W: | ||
| 59 | register_code(KC_LALT); | ||
| 60 | tap_code16(LSFT(KC_RIGHT)); // select to end of this word | ||
| 61 | tap_code16(LSFT(KC_RIGHT)); // select to end of next word | ||
| 62 | tap_code16(LSFT(KC_LEFT)); // select to start of next word | ||
| 63 | unregister_code(KC_LALT); | ||
| 64 | break; | ||
| 65 | } | ||
| 66 | } | ||
| 67 | |||
| 68 | static void comma_period(uint16_t keycode) { | ||
| 69 | switch (keycode) { | ||
| 70 | case VIM_COMMA: | ||
| 71 | if (SHIFTED) { | ||
| 72 | // indent | ||
| 73 | tap_code16(LGUI(KC_LBRACKET)); | ||
| 74 | } else { | ||
| 75 | // toggle comment | ||
| 76 | tap_code16(LGUI(KC_SLASH)); | ||
| 77 | } | ||
| 78 | break; | ||
| 79 | case VIM_PERIOD: | ||
| 80 | if (SHIFTED) { | ||
| 81 | // outdent | ||
| 82 | tap_code16(LGUI(KC_RBRACKET)); | ||
| 83 | } | ||
| 84 | break; | ||
| 85 | } | ||
| 86 | } | ||
| 87 | |||
| 88 | |||
| 89 | bool process_record_vimlayer(uint16_t keycode, keyrecord_t *record) { | ||
| 90 | |||
| 91 | /****** mod passthru *****/ | ||
| 92 | if(record->event.pressed && layer_state_is(vim_cmd_layer()) && (IS_MOD(keycode) || keycode == LSFT(KC_LALT))) { | ||
| 93 | mod_override_layer_state = layer_state; | ||
| 94 | mod_override_triggering_key = keycode; | ||
| 95 | // TODO: change this to track key location instead | ||
| 96 | layer_clear(); | ||
| 97 | return true; // let the event fall through... | ||
| 98 | } | ||
| 99 | if(mod_override_layer_state && !record->event.pressed && keycode == mod_override_triggering_key) { | ||
| 100 | layer_state_set(mod_override_layer_state); | ||
| 101 | mod_override_layer_state = 0; | ||
| 102 | mod_override_triggering_key = 0; | ||
| 103 | return true; | ||
| 104 | } | ||
| 105 | |||
| 106 | if (VIM_START <= keycode && keycode <= VIM_ESC) { | ||
| 107 | if(keycode == VIM_SHIFT) { | ||
| 108 | SHIFTED = record->event.pressed; | ||
| 109 | return false; | ||
| 110 | } | ||
| 111 | |||
| 112 | if (record->event.pressed) { | ||
| 113 | if(keycode == VIM_START) { | ||
| 114 | // entry from anywhere | ||
| 115 | layer_on(vim_cmd_layer()); | ||
| 116 | vstate = VIM_START; | ||
| 117 | |||
| 118 | // reset state | ||
| 119 | yank_was_lines = false; | ||
| 120 | SHIFTED = false; | ||
| 121 | mod_override_layer_state = 0; | ||
| 122 | mod_override_triggering_key = 0; | ||
| 123 | |||
| 124 | return false; | ||
| 125 | } | ||
| 126 | switch(vstate) { | ||
| 127 | case VIM_START: | ||
| 128 | switch(keycode){ | ||
| 129 | /***************************** | ||
| 130 | * ground state | ||
| 131 | *****************************/ | ||
| 132 | case VIM_A: | ||
| 133 | if(SHIFTED) { | ||
| 134 | // tap_code16(LGUI(KC_RIGHT)); | ||
| 135 | tap_code16(LCTL(KC_E)); | ||
| 136 | } else { | ||
| 137 | tap_code(KC_RIGHT); | ||
| 138 | } | ||
| 139 | EDIT; | ||
| 140 | break; | ||
| 141 | case VIM_B: | ||
| 142 | register_code(KC_LALT); | ||
| 143 | register_code(KC_LEFT); | ||
| 144 | break; | ||
| 145 | case VIM_C: | ||
| 146 | if(SHIFTED) { | ||
| 147 | register_code(KC_LSHIFT); | ||
| 148 | tap_code16(LGUI(KC_RIGHT)); | ||
| 149 | unregister_code(KC_LSHIFT); | ||
| 150 | tap_code16(LGUI(KC_X)); | ||
| 151 | yank_was_lines = false; | ||
| 152 | EDIT; | ||
| 153 | } else { | ||
| 154 | vstate = VIM_C; | ||
| 155 | } | ||
| 156 | break; | ||
| 157 | case VIM_D: | ||
| 158 | if(SHIFTED) { | ||
| 159 | tap_code16(LCTL(KC_K)); | ||
| 160 | } else { | ||
| 161 | vstate = VIM_D; | ||
| 162 | } | ||
| 163 | break; | ||
| 164 | case VIM_E: | ||
| 165 | register_code(KC_LALT); | ||
| 166 | register_code(KC_RIGHT); | ||
| 167 | break; | ||
| 168 | case VIM_G: | ||
| 169 | if(SHIFTED) { | ||
| 170 | tap_code(KC_END); | ||
| 171 | } else { | ||
| 172 | vstate = VIM_G; | ||
| 173 | } | ||
| 174 | break; | ||
| 175 | case VIM_H: | ||
| 176 | register_code(KC_LEFT); | ||
| 177 | break; | ||
| 178 | case VIM_I: | ||
| 179 | if(SHIFTED){ | ||
| 180 | tap_code16(LCTL(KC_A)); | ||
| 181 | } | ||
| 182 | EDIT; | ||
| 183 | break; | ||
| 184 | case VIM_J: | ||
| 185 | if(SHIFTED) { | ||
| 186 | tap_code16(LGUI(KC_RIGHT)); | ||
| 187 | tap_code(KC_DEL); | ||
| 188 | } else { | ||
| 189 | register_code(KC_DOWN); | ||
| 190 | } | ||
| 191 | break; | ||
| 192 | case VIM_K: | ||
| 193 | register_code(KC_UP); | ||
| 194 | break; | ||
| 195 | case VIM_L: | ||
| 196 | register_code(KC_RIGHT); | ||
| 197 | break; | ||
| 198 | case VIM_O: | ||
| 199 | if(SHIFTED) { | ||
| 200 | tap_code16(LGUI(KC_LEFT)); | ||
| 201 | tap_code(KC_ENTER); | ||
| 202 | tap_code(KC_UP); | ||
| 203 | EDIT; | ||
| 204 | } else { | ||
| 205 | tap_code16(LGUI(KC_RIGHT)); | ||
| 206 | tap_code(KC_ENTER); | ||
| 207 | EDIT; | ||
| 208 | } | ||
| 209 | break; | ||
| 210 | case VIM_P: | ||
| 211 | if(SHIFTED) { | ||
| 212 | tap_code16(LGUI(KC_LEFT)); | ||
| 213 | tap_code16(LGUI(KC_V)); | ||
| 214 | } else { | ||
| 215 | if(yank_was_lines) { | ||
| 216 | tap_code16(LGUI(KC_RIGHT)); | ||
| 217 | tap_code(KC_RIGHT); | ||
| 218 | tap_code16(LGUI(KC_V)); | ||
| 219 | } else { | ||
| 220 | tap_code16(LGUI(KC_V)); | ||
| 221 | } | ||
| 222 | } | ||
| 223 | break; | ||
| 224 | case VIM_S: | ||
| 225 | // s for substitute? | ||
| 226 | if(SHIFTED) { | ||
| 227 | tap_code16(LGUI(KC_LEFT)); | ||
| 228 | register_code(KC_LSHIFT); | ||
| 229 | tap_code16(LGUI(KC_RIGHT)); | ||
| 230 | unregister_code(KC_LSHIFT); | ||
| 231 | tap_code16(LGUI(KC_X)); | ||
| 232 | yank_was_lines = false; | ||
| 233 | EDIT; | ||
| 234 | } else { | ||
| 235 | tap_code16(LSFT(KC_RIGHT)); | ||
| 236 | tap_code16(LGUI(KC_X)); | ||
| 237 | yank_was_lines = false; | ||
| 238 | EDIT; | ||
| 239 | } | ||
| 240 | break; | ||
| 241 | case VIM_U: | ||
| 242 | if(SHIFTED) { | ||
| 243 | register_code(KC_LSFT); | ||
| 244 | tap_code16(LGUI(KC_Z)); | ||
| 245 | unregister_code(KC_LSHIFT); | ||
| 246 | } else { | ||
| 247 | tap_code16(LGUI(KC_Z)); | ||
| 248 | } | ||
| 249 | break; | ||
| 250 | case VIM_V: | ||
| 251 | if(SHIFTED) { | ||
| 252 | tap_code16(LGUI(KC_LEFT)); | ||
| 253 | tap_code16(LSFT(KC_DOWN)); | ||
| 254 | vstate = VIM_VS; | ||
| 255 | } else { | ||
| 256 | vstate = VIM_V; | ||
| 257 | } | ||
| 258 | break; | ||
| 259 | case VIM_W: | ||
| 260 | register_code(KC_LALT); | ||
| 261 | tap_code(KC_RIGHT); | ||
| 262 | tap_code(KC_RIGHT); | ||
| 263 | tap_code(KC_LEFT); | ||
| 264 | unregister_code(KC_LALT); | ||
| 265 | break; | ||
| 266 | case VIM_X: | ||
| 267 | // tap_code16(LSFT(KC_RIGHT)); | ||
| 268 | // tap_code16(LGUI(KC_X)); | ||
| 269 | register_code(KC_DEL); | ||
| 270 | break; | ||
| 271 | case VIM_Y: | ||
| 272 | if(SHIFTED) { | ||
| 273 | tap_code16(LGUI(KC_LEFT)); | ||
| 274 | tap_code16(LSFT(KC_DOWN)); | ||
| 275 | tap_code16(LGUI(KC_C)); | ||
| 276 | tap_code(KC_RIGHT); | ||
| 277 | yank_was_lines = true; | ||
| 278 | } else { | ||
| 279 | vstate = VIM_Y; | ||
| 280 | } | ||
| 281 | break; | ||
| 282 | case VIM_COMMA: | ||
| 283 | case VIM_PERIOD: | ||
| 284 | comma_period(keycode); | ||
| 285 | break; | ||
| 286 | } | ||
| 287 | break; | ||
| 288 | case VIM_C: | ||
| 289 | /***************************** | ||
| 290 | * c- ...for change. I never use this... | ||
| 291 | *****************************/ | ||
| 292 | switch(keycode) { | ||
| 293 | case VIM_B: | ||
| 294 | case VIM_E: | ||
| 295 | case VIM_H: | ||
| 296 | case VIM_J: | ||
| 297 | case VIM_K: | ||
| 298 | case VIM_L: | ||
| 299 | case VIM_W: | ||
| 300 | simple_movement(keycode); | ||
| 301 | tap_code16(LGUI(KC_X)); | ||
| 302 | yank_was_lines = false; | ||
| 303 | EDIT; | ||
| 304 | break; | ||
| 305 | |||
| 306 | case VIM_C: | ||
| 307 | tap_code16(LGUI(KC_LEFT)); | ||
| 308 | register_code(KC_LSHIFT); | ||
| 309 | tap_code16(LGUI(KC_RIGHT)); | ||
| 310 | unregister_code(KC_LSHIFT); | ||
| 311 | tap_code16(LGUI(KC_X)); | ||
| 312 | yank_was_lines = false; | ||
| 313 | EDIT; | ||
| 314 | break; | ||
| 315 | case VIM_I: | ||
| 316 | vstate = VIM_CI; | ||
| 317 | break; | ||
| 318 | default: | ||
| 319 | vstate = VIM_START; | ||
| 320 | break; | ||
| 321 | } | ||
| 322 | break; | ||
| 323 | case VIM_CI: | ||
| 324 | /***************************** | ||
| 325 | * ci- ...change inner word | ||
| 326 | *****************************/ | ||
| 327 | switch(keycode) { | ||
| 328 | case VIM_W: | ||
| 329 | tap_code16(LALT(KC_LEFT)); | ||
| 330 | register_code(KC_LSHIFT); | ||
| 331 | tap_code16(LALT(KC_RIGHT)); | ||
| 332 | unregister_code(KC_LSHIFT); | ||
| 333 | tap_code16(LGUI(KC_X)); | ||
| 334 | yank_was_lines = false; | ||
| 335 | EDIT; | ||
| 336 | default: | ||
| 337 | vstate = VIM_START; | ||
| 338 | break; | ||
| 339 | } | ||
| 340 | break; | ||
| 341 | case VIM_D: | ||
| 342 | /***************************** | ||
| 343 | * d- ...delete stuff | ||
| 344 | *****************************/ | ||
| 345 | switch(keycode) { | ||
| 346 | case VIM_B: | ||
| 347 | case VIM_E: | ||
| 348 | case VIM_H: | ||
| 349 | case VIM_J: | ||
| 350 | case VIM_K: | ||
| 351 | case VIM_L: | ||
| 352 | case VIM_W: | ||
| 353 | simple_movement(keycode); | ||
| 354 | tap_code16(LGUI(KC_X)); | ||
| 355 | yank_was_lines = false; | ||
| 356 | vstate = VIM_START; | ||
| 357 | break; | ||
| 358 | case VIM_D: | ||
| 359 | tap_code16(LGUI(KC_LEFT)); | ||
| 360 | tap_code16(LSFT(KC_DOWN)); | ||
| 361 | tap_code16(LGUI(KC_X)); | ||
| 362 | yank_was_lines = true; | ||
| 363 | vstate = VIM_START; | ||
| 364 | break; | ||
| 365 | case VIM_I: | ||
| 366 | vstate = VIM_DI; | ||
| 367 | break; | ||
| 368 | default: | ||
| 369 | vstate = VIM_START; | ||
| 370 | break; | ||
| 371 | } | ||
| 372 | break; | ||
| 373 | case VIM_DI: | ||
| 374 | /***************************** | ||
| 375 | * ci- ...delete a word... FROM THE INSIDE! | ||
| 376 | *****************************/ | ||
| 377 | switch(keycode) { | ||
| 378 | case VIM_W: | ||
| 379 | tap_code16(LALT(KC_LEFT)); | ||
| 380 | register_code(KC_LSHIFT); | ||
| 381 | tap_code16(LALT(KC_RIGHT)); | ||
| 382 | unregister_code(KC_LSHIFT); | ||
| 383 | tap_code16(LGUI(KC_X)); | ||
| 384 | yank_was_lines = false; | ||
| 385 | vstate = VIM_START; | ||
| 386 | default: | ||
| 387 | vstate = VIM_START; | ||
| 388 | break; | ||
| 389 | } | ||
| 390 | break; | ||
| 391 | case VIM_V: | ||
| 392 | /***************************** | ||
| 393 | * visual! | ||
| 394 | *****************************/ | ||
| 395 | switch(keycode) { | ||
| 396 | case VIM_D: | ||
| 397 | case VIM_X: | ||
| 398 | tap_code16(LGUI(KC_X)); | ||
| 399 | yank_was_lines = false; | ||
| 400 | vstate = VIM_START; | ||
| 401 | break; | ||
| 402 | case VIM_B: | ||
| 403 | register_code(KC_LALT); | ||
| 404 | register_code(KC_LSHIFT); | ||
| 405 | register_code(KC_LEFT); | ||
| 406 | // leave open for key repeat | ||
| 407 | break; | ||
| 408 | case VIM_E: | ||
| 409 | register_code(KC_LALT); | ||
| 410 | register_code(KC_LSHIFT); | ||
| 411 | register_code(KC_RIGHT); | ||
| 412 | // leave open for key repeat | ||
| 413 | break; | ||
| 414 | case VIM_H: | ||
| 415 | register_code(KC_LSHIFT); | ||
| 416 | register_code(KC_LEFT); | ||
| 417 | break; | ||
| 418 | case VIM_I: | ||
| 419 | vstate = VIM_VI; | ||
| 420 | break; | ||
| 421 | case VIM_J: | ||
| 422 | register_code(KC_LSHIFT); | ||
| 423 | register_code(KC_DOWN); | ||
| 424 | break; | ||
| 425 | case VIM_K: | ||
| 426 | register_code(KC_LSHIFT); | ||
| 427 | register_code(KC_UP); | ||
| 428 | break; | ||
| 429 | case VIM_L: | ||
| 430 | register_code(KC_LSHIFT); | ||
| 431 | register_code(KC_RIGHT); | ||
| 432 | break; | ||
| 433 | case VIM_W: | ||
| 434 | register_code(KC_LALT); | ||
| 435 | tap_code16(LSFT(KC_RIGHT)); // select to end of this word | ||
| 436 | tap_code16(LSFT(KC_RIGHT)); // select to end of next word | ||
| 437 | tap_code16(LSFT(KC_LEFT)); // select to start of next word | ||
| 438 | unregister_code(KC_LALT); | ||
| 439 | break; | ||
| 440 | case VIM_P: | ||
| 441 | tap_code16(LGUI(KC_V)); | ||
| 442 | vstate = VIM_START; | ||
| 443 | break; | ||
| 444 | case VIM_Y: | ||
| 445 | tap_code16(LGUI(KC_C)); | ||
| 446 | tap_code(KC_RIGHT); | ||
| 447 | yank_was_lines = false; | ||
| 448 | vstate = VIM_START; | ||
| 449 | break; | ||
| 450 | case VIM_V: | ||
| 451 | case VIM_ESC: | ||
| 452 | tap_code(KC_RIGHT); | ||
| 453 | vstate = VIM_START; | ||
| 454 | break; | ||
| 455 | case VIM_COMMA: | ||
| 456 | case VIM_PERIOD: | ||
| 457 | comma_period(keycode); | ||
| 458 | break; | ||
| 459 | default: | ||
| 460 | // do nothing | ||
| 461 | break; | ||
| 462 | } | ||
| 463 | break; | ||
| 464 | case VIM_VI: | ||
| 465 | /***************************** | ||
| 466 | * vi- ...select a word... FROM THE INSIDE! | ||
| 467 | *****************************/ | ||
| 468 | switch(keycode) { | ||
| 469 | case VIM_W: | ||
| 470 | tap_code16(LALT(KC_LEFT)); | ||
| 471 | register_code(KC_LSHIFT); | ||
| 472 | tap_code16(LALT(KC_RIGHT)); | ||
| 473 | unregister_code(KC_LSHIFT); | ||
| 474 | vstate = VIM_V; | ||
| 475 | default: | ||
| 476 | // ignore | ||
| 477 | vstate = VIM_V; | ||
| 478 | break; | ||
| 479 | } | ||
| 480 | break; | ||
| 481 | case VIM_VS: | ||
| 482 | /***************************** | ||
| 483 | * visual line | ||
| 484 | *****************************/ | ||
| 485 | switch(keycode) { | ||
| 486 | case VIM_D: | ||
| 487 | case VIM_X: | ||
| 488 | tap_code16(LGUI(KC_X)); | ||
| 489 | yank_was_lines = true; | ||
| 490 | vstate = VIM_START; | ||
| 491 | break; | ||
| 492 | case VIM_J: | ||
| 493 | register_code(KC_LSHIFT); | ||
| 494 | register_code(KC_DOWN); | ||
| 495 | break; | ||
| 496 | case VIM_K: | ||
| 497 | register_code(KC_LSHIFT); | ||
| 498 | register_code(KC_UP); | ||
| 499 | break; | ||
| 500 | case VIM_Y: | ||
| 501 | tap_code16(LGUI(KC_C)); | ||
| 502 | yank_was_lines = true; | ||
| 503 | tap_code(KC_RIGHT); | ||
| 504 | vstate = VIM_START; | ||
| 505 | break; | ||
| 506 | case VIM_P: | ||
| 507 | tap_code16(LGUI(KC_V)); | ||
| 508 | vstate = VIM_START; | ||
| 509 | break; | ||
| 510 | case VIM_V: | ||
| 511 | case VIM_ESC: | ||
| 512 | tap_code(KC_RIGHT); | ||
| 513 | vstate = VIM_START; | ||
| 514 | break; | ||
| 515 | case VIM_COMMA: | ||
| 516 | case VIM_PERIOD: | ||
| 517 | comma_period(keycode); | ||
| 518 | break; | ||
| 519 | default: | ||
| 520 | // do nothing | ||
| 521 | break; | ||
| 522 | } | ||
| 523 | break; | ||
| 524 | case VIM_G: | ||
| 525 | /***************************** | ||
| 526 | * gg, and a grab-bag of other macros i find useful | ||
| 527 | *****************************/ | ||
| 528 | switch(keycode) { | ||
| 529 | case VIM_G: | ||
| 530 | tap_code(KC_HOME); | ||
| 531 | break; | ||
| 532 | // codes b | ||
| 533 | case VIM_H: | ||
| 534 | tap_code16(LCTL(KC_A)); | ||
| 535 | break; | ||
| 536 | case VIM_J: | ||
| 537 | register_code(KC_PGDN); | ||
| 538 | break; | ||
| 539 | case VIM_K: | ||
| 540 | register_code(KC_PGUP); | ||
| 541 | break; | ||
| 542 | case VIM_L: | ||
| 543 | tap_code16(LCTL(KC_E)); | ||
| 544 | break; | ||
| 545 | default: | ||
| 546 | // do nothing | ||
| 547 | break; | ||
| 548 | } | ||
| 549 | vstate = VIM_START; | ||
| 550 | break; | ||
| 551 | case VIM_Y: | ||
| 552 | /***************************** | ||
| 553 | * yoink! | ||
| 554 | *****************************/ | ||
| 555 | switch(keycode) { | ||
| 556 | case VIM_B: | ||
| 557 | case VIM_E: | ||
| 558 | case VIM_H: | ||
| 559 | case VIM_J: | ||
| 560 | case VIM_K: | ||
| 561 | case VIM_L: | ||
| 562 | case VIM_W: | ||
| 563 | simple_movement(keycode); | ||
| 564 | tap_code16(LGUI(KC_C)); | ||
| 565 | tap_code(KC_RIGHT); | ||
| 566 | yank_was_lines = false; | ||
| 567 | break; | ||
| 568 | case VIM_Y: | ||
| 569 | tap_code16(LGUI(KC_LEFT)); | ||
| 570 | tap_code16(LSFT(KC_DOWN)); | ||
| 571 | tap_code16(LGUI(KC_C)); | ||
| 572 | tap_code(KC_RIGHT); | ||
| 573 | yank_was_lines = true; | ||
| 574 | break; | ||
| 575 | default: | ||
| 576 | // NOTHING | ||
| 577 | break; | ||
| 578 | } | ||
| 579 | vstate = VIM_START; | ||
| 580 | break; | ||
| 581 | } | ||
| 582 | } else { | ||
| 583 | /************************ | ||
| 584 | * key unregister_code events | ||
| 585 | ************************/ | ||
| 586 | clear_keyboard(); | ||
| 587 | } | ||
| 588 | return false; | ||
| 589 | } else { | ||
| 590 | return true; | ||
| 591 | } | ||
| 592 | } | ||
| 593 | |||
diff --git a/users/twschum/xtonhasvim.h b/users/twschum/xtonhasvim.h new file mode 100644 index 000000000..fd9ebd4f0 --- /dev/null +++ b/users/twschum/xtonhasvim.h | |||
| @@ -0,0 +1,62 @@ | |||
| 1 | /* Copyright 2015-2017 Christon DeWan | ||
| 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 | |||
| 17 | #ifndef USERSPACE | ||
| 18 | #define USERSPACE | ||
| 19 | |||
| 20 | #include QMK_KEYBOARD_H | ||
| 21 | |||
| 22 | enum xtonhasvim_keycodes { | ||
| 23 | VIM_START = SAFE_RANGE, // bookend for vim states | ||
| 24 | VIM_A, | ||
| 25 | VIM_B, | ||
| 26 | VIM_C, | ||
| 27 | VIM_CI, | ||
| 28 | VIM_D, | ||
| 29 | VIM_DI, | ||
| 30 | VIM_E, | ||
| 31 | VIM_H, | ||
| 32 | VIM_G, | ||
| 33 | VIM_I, | ||
| 34 | VIM_J, | ||
| 35 | VIM_K, | ||
| 36 | VIM_L, | ||
| 37 | VIM_O, | ||
| 38 | VIM_P, | ||
| 39 | VIM_S, | ||
| 40 | VIM_U, | ||
| 41 | VIM_V, | ||
| 42 | VIM_VS, // visual-line | ||
| 43 | VIM_VI, | ||
| 44 | VIM_W, | ||
| 45 | VIM_X, | ||
| 46 | VIM_Y, | ||
| 47 | VIM_PERIOD, // to support indent/outdent | ||
| 48 | VIM_COMMA, // and toggle comments | ||
| 49 | VIM_SHIFT, // avoid side-effect of supporting real shift. | ||
| 50 | VIM_ESC, // bookend | ||
| 51 | VIM_SAFE_RANGE // start other keycodes here. | ||
| 52 | }; | ||
| 53 | |||
| 54 | bool process_record_vimlayer(uint16_t keycode, keyrecord_t *record); | ||
| 55 | |||
| 56 | // NOTE: Define this in keymap.c to return vim layer | ||
| 57 | extern uint8_t vim_cmd_layer(void); | ||
| 58 | |||
| 59 | extern uint16_t vstate; | ||
| 60 | |||
| 61 | |||
| 62 | #endif | ||
