diff options
Diffstat (limited to 'quantum/action_macro.h')
| -rw-r--r-- | quantum/action_macro.h | 123 |
1 files changed, 123 insertions, 0 deletions
diff --git a/quantum/action_macro.h b/quantum/action_macro.h new file mode 100644 index 000000000..685e2c6ff --- /dev/null +++ b/quantum/action_macro.h | |||
| @@ -0,0 +1,123 @@ | |||
| 1 | /* | ||
| 2 | Copyright 2013 Jun Wako <wakojun@gmail.com> | ||
| 3 | |||
| 4 | This program is free software: you can redistribute it and/or modify | ||
| 5 | it under the terms of the GNU General Public License as published by | ||
| 6 | the Free Software Foundation, either version 2 of the License, or | ||
| 7 | (at your option) any later version. | ||
| 8 | |||
| 9 | This program is distributed in the hope that it will be useful, | ||
| 10 | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
| 12 | GNU General Public License for more details. | ||
| 13 | |||
| 14 | You should have received a copy of the GNU General Public License | ||
| 15 | along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
| 16 | */ | ||
| 17 | |||
| 18 | #pragma once | ||
| 19 | |||
| 20 | #include <stdint.h> | ||
| 21 | #include "progmem.h" | ||
| 22 | |||
| 23 | typedef uint8_t macro_t; | ||
| 24 | |||
| 25 | #define MACRO_NONE (macro_t *)0 | ||
| 26 | #define MACRO(...) \ | ||
| 27 | ({ \ | ||
| 28 | static const macro_t __m[] PROGMEM = {__VA_ARGS__}; \ | ||
| 29 | &__m[0]; \ | ||
| 30 | }) | ||
| 31 | #define MACRO_GET(p) pgm_read_byte(p) | ||
| 32 | |||
| 33 | // Sends press when the macro key is pressed, release when release, or tap_macro when the key has been tapped | ||
| 34 | #define MACRO_TAP_HOLD(record, press, release, tap_macro) (((record)->event.pressed) ? (((record)->tap.count <= 0 || (record)->tap.interrupted) ? (press) : MACRO_NONE) : (((record)->tap.count > 0 && !((record)->tap.interrupted)) ? (tap_macro) : (release))) | ||
| 35 | |||
| 36 | // Holds down the modifier mod when the macro key is held, or sends macro instead when tapped | ||
| 37 | #define MACRO_TAP_HOLD_MOD(record, macro, mod) MACRO_TAP_HOLD(record, (MACRO(D(mod), END)), MACRO(U(mod), END), macro) | ||
| 38 | |||
| 39 | // Holds down the modifier mod when the macro key is held, or pressed a shifted key when tapped (eg: shift+3 for #) | ||
| 40 | #define MACRO_TAP_SHFT_KEY_HOLD_MOD(record, key, mod) MACRO_TAP_HOLD_MOD(record, (MACRO(I(10), D(LSFT), T(key), U(LSFT), END)), mod) | ||
| 41 | |||
| 42 | // Momentary switch layer when held, sends macro if tapped | ||
| 43 | #define MACRO_TAP_HOLD_LAYER(record, macro, layer) \ | ||
| 44 | (((record)->event.pressed) ? (((record)->tap.count <= 0 || (record)->tap.interrupted) ? ({ \ | ||
| 45 | layer_on((layer)); \ | ||
| 46 | MACRO_NONE; \ | ||
| 47 | }) \ | ||
| 48 | : MACRO_NONE) \ | ||
| 49 | : (((record)->tap.count > 0 && !((record)->tap.interrupted)) ? (macro) : ({ \ | ||
| 50 | layer_off((layer)); \ | ||
| 51 | MACRO_NONE; \ | ||
| 52 | }))) | ||
| 53 | |||
| 54 | // Momentary switch layer when held, presses a shifted key when tapped (eg: shift+3 for #) | ||
| 55 | #define MACRO_TAP_SHFT_KEY_HOLD_LAYER(record, key, layer) MACRO_TAP_HOLD_LAYER(record, MACRO(I(10), D(LSFT), T(key), U(LSFT), END), layer) | ||
| 56 | |||
| 57 | #ifndef NO_ACTION_MACRO | ||
| 58 | void action_macro_play(const macro_t *macro_p); | ||
| 59 | #else | ||
| 60 | # define action_macro_play(macro) | ||
| 61 | #endif | ||
| 62 | |||
| 63 | /* Macro commands | ||
| 64 | * code(0x04-73) // key down(1byte) | ||
| 65 | * code(0x04-73) | 0x80 // key up(1byte) | ||
| 66 | * { KEY_DOWN, code(0x04-0xff) } // key down(2bytes) | ||
| 67 | * { KEY_UP, code(0x04-0xff) } // key up(2bytes) | ||
| 68 | * WAIT // wait milli-seconds | ||
| 69 | * INTERVAL // set interval between macro commands | ||
| 70 | * END // stop macro execution | ||
| 71 | * | ||
| 72 | * Ideas(Not implemented): | ||
| 73 | * modifiers | ||
| 74 | * system usage | ||
| 75 | * consumer usage | ||
| 76 | * unicode usage | ||
| 77 | * function call | ||
| 78 | * conditionals | ||
| 79 | * loop | ||
| 80 | */ | ||
| 81 | enum macro_command_id { | ||
| 82 | /* 0x00 - 0x03 */ | ||
| 83 | END = 0x00, | ||
| 84 | KEY_DOWN, | ||
| 85 | KEY_UP, | ||
| 86 | |||
| 87 | /* 0x04 - 0x73 (reserved for keycode down) */ | ||
| 88 | |||
| 89 | /* 0x74 - 0x83 */ | ||
| 90 | WAIT = 0x74, | ||
| 91 | INTERVAL, | ||
| 92 | |||
| 93 | /* 0x84 - 0xf3 (reserved for keycode up) */ | ||
| 94 | |||
| 95 | /* 0xf4 - 0xff */ | ||
| 96 | }; | ||
| 97 | |||
| 98 | /* TODO: keycode:0x04-0x73 can be handled by 1byte command else 2bytes are needed | ||
| 99 | * if keycode between 0x04 and 0x73 | ||
| 100 | * keycode / (keycode|0x80) | ||
| 101 | * else | ||
| 102 | * {KEY_DOWN, keycode} / {KEY_UP, keycode} | ||
| 103 | */ | ||
| 104 | #define DOWN(key) KEY_DOWN, (key) | ||
| 105 | #define UP(key) KEY_UP, (key) | ||
| 106 | #define TYPE(key) DOWN(key), UP(key) | ||
| 107 | #define WAIT(ms) WAIT, (ms) | ||
| 108 | #define INTERVAL(ms) INTERVAL, (ms) | ||
| 109 | |||
| 110 | /* key down */ | ||
| 111 | #define D(key) DOWN(KC_##key) | ||
| 112 | /* key up */ | ||
| 113 | #define U(key) UP(KC_##key) | ||
| 114 | /* key type */ | ||
| 115 | #define T(key) TYPE(KC_##key) | ||
| 116 | /* wait */ | ||
| 117 | #define W(ms) WAIT(ms) | ||
| 118 | /* interval */ | ||
| 119 | #define I(ms) INTERVAL(ms) | ||
| 120 | |||
| 121 | /* for backward comaptibility */ | ||
| 122 | #define MD(key) DOWN(KC_##key) | ||
| 123 | #define MU(key) UP(KC_##key) | ||
