768 lines
35 KiB
C
768 lines
35 KiB
C
#include QMK_KEYBOARD_H
|
|
|
|
#include <math.h> // sqrtf, powf
|
|
|
|
#ifdef CONSOLE_ENABLE
|
|
|
|
#include <print.h>
|
|
|
|
#endif
|
|
|
|
enum ctrl_keycodes {
|
|
L_BRI = SAFE_RANGE, //LED Brightness Increase //Working
|
|
L_BRD, //LED Brightness Decrease //Working
|
|
L_PTN, //LED Pattern Select Next //Working
|
|
L_PTP, //LED Pattern Select Previous //Working
|
|
L_PSI, //LED Pattern Speed Increase //Working
|
|
L_PSD, //LED Pattern Speed Decrease //Working
|
|
L_T_MD, //LED Toggle Mode //Working
|
|
L_T_ONF, //LED Toggle On / Off //Broken
|
|
L_ON, //LED On //Broken
|
|
L_OFF, //LED Off //Broken
|
|
L_T_BR, //LED Toggle Breath Effect //Working
|
|
L_T_PTD, //LED Toggle Scrolling Pattern Direction //Working
|
|
U_T_AGCR, //USB Toggle Automatic GCR control //Working
|
|
DBG_TOG, //DEBUG Toggle On / Off //
|
|
DBG_MTRX, //DEBUG Toggle Matrix Prints //
|
|
DBG_KBD, //DEBUG Toggle Keyboard Prints //
|
|
DBG_MOU, //DEBUG Toggle Mouse Prints //
|
|
MD_BOOT, //Restart into bootloader after hold timeout //Working
|
|
|
|
|
|
L_SP_PR, //LED Splash Pattern Select Previous
|
|
L_SP_NE, //LED Splash Pattern Select Next
|
|
|
|
L_SP_WD, //LED Splash Widen Wavefront width
|
|
L_SP_NW, //LED Splash Narrow Wavefront width
|
|
|
|
L_SP_FA, //LED Splash wave travel speed faster (shorter period)
|
|
L_SP_SL, //LED Splash wave travel speed slower (longer period)
|
|
|
|
L_CP_PR, //LED Color Pattern Select Previous
|
|
L_CP_NX, //LEB Color Pattern Select Next
|
|
|
|
S_RESET // reset all parameters
|
|
};
|
|
|
|
keymap_config_t keymap_config;
|
|
|
|
const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
|
|
[0] = LAYOUT(
|
|
KC_ESC, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12, KC_PSCR, KC_SLCK, KC_PAUS, \
|
|
KC_GRV, KC_1, KC_2, KC_3, KC_4, KC_5, KC_6, KC_7, KC_8, KC_9, KC_0, KC_MINS, KC_EQL, KC_BSPC, KC_INS, KC_HOME, KC_PGUP, \
|
|
KC_TAB, KC_Q, KC_W, KC_E, KC_R, KC_T, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_LBRC, KC_RBRC, KC_BSLS, KC_DEL, KC_END, KC_PGDN, \
|
|
KC_CAPS, KC_A, KC_S, KC_D, KC_F, KC_G, KC_H, KC_J, KC_K, KC_L, KC_SCLN, KC_QUOT, KC_ENT, \
|
|
KC_LSFT, KC_Z, KC_X, KC_C, KC_V, KC_B, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, KC_RSFT, KC_UP, \
|
|
KC_LCTL, KC_LGUI, KC_LALT, KC_SPC, KC_RALT, MO(1), KC_APP, KC_RCTL, KC_LEFT, KC_DOWN, KC_RGHT \
|
|
),
|
|
[1] = LAYOUT(
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, KC_MUTE, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, KC_MPLY, KC_MSTP, KC_VOLU, \
|
|
L_T_BR, L_PSD, L_BRI, L_PSI, _______, _______, _______, _______, U_T_AGCR,_______, MO(2), _______, _______, _______, KC_MPRV, KC_MNXT, KC_VOLD, \
|
|
L_T_PTD, L_PTP, L_BRD, L_PTN, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, L_T_MD, L_T_ONF, _______, _______, MD_BOOT, NK_TOGG, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______ \
|
|
),
|
|
[2] = LAYOUT(
|
|
S_RESET, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
S_RESET, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
L_CP_NX, L_SP_SL, L_SP_WD, L_SP_FA, _______, _______, L_CP_NX, L_SP_SL, L_SP_WD, L_SP_FA, _______, _______, _______, _______, _______, _______, _______, \
|
|
L_CP_PR, L_SP_PR, L_SP_NW, L_SP_NE, _______, _______, L_CP_PR, L_SP_PR, L_SP_NW, L_SP_NE, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______ \
|
|
)
|
|
/*
|
|
[X] = LAYOUT(
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, NK_TOGG, _______, _______, _______, _______, _______, _______, \
|
|
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______ \
|
|
),
|
|
*/
|
|
};
|
|
|
|
#define DISTANCE_NORAMLIZING_PARAMETER 3
|
|
struct {
|
|
uint8_t PATTERN_INDEX;
|
|
float WAVE_WIDTH;
|
|
float WAVE_SPEED;
|
|
int COLOR_PATTERN_INDEX;
|
|
float TRAVEL_DISTANCE;
|
|
} USER_CONFIG = {
|
|
.PATTERN_INDEX = 1,
|
|
.WAVE_WIDTH = 10, // width of the wave in keycaps
|
|
.WAVE_SPEED = 15, // travel how many keycaps per second
|
|
.COLOR_PATTERN_INDEX = 0,
|
|
.TRAVEL_DISTANCE = 25,
|
|
};
|
|
|
|
|
|
#define COLOR_PATTERN_RGB_COUNT 18
|
|
static uint8_t COLOR_PATTERNS[][COLOR_PATTERN_RGB_COUNT][3] = {
|
|
{ // default rainbow color
|
|
{255, 0, 0}, {255, 0, 0}, {255, 127, 0},
|
|
{255, 127, 0}, {255, 255, 0}, {255, 255, 0},
|
|
{120, 255, 0}, {120, 255, 0}, { 0, 255, 0},
|
|
{ 0, 255, 0}, { 0, 255, 120}, { 0, 255, 120},
|
|
{ 0, 0, 255}, { 0, 0, 255}, { 75, 0, 130},
|
|
{ 75, 0, 130}, { 43, 0, 130}, { 43, 0, 130},
|
|
}, { // light rainbow color
|
|
{248, 12, 18}, {238, 17, 0}, {255, 51, 17},
|
|
{255, 68, 32}, {255, 102, 68}, {255, 153, 51},
|
|
{254, 174, 45}, {204, 187, 51}, {208, 195, 16},
|
|
{170, 204, 34}, {105, 208, 37}, { 34, 204, 170},
|
|
{ 18, 189, 185}, { 17, 170, 187}, { 68, 68, 221},
|
|
{ 51, 17, 187}, { 59, 12, 189}, { 68, 34, 153},
|
|
}, { // white flat
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
{255, 255, 255}, {255, 255, 255}, {255, 255, 255},
|
|
}, { // white fade, cos curve
|
|
{255, 255, 255}, {255, 255, 255}, {252, 252, 252},
|
|
{247, 247, 247}, {240, 240, 240}, {232, 232, 232},
|
|
{221, 221, 221}, {209, 209, 209}, {196, 196, 196},
|
|
{181, 181, 181}, {164, 164, 164}, {147, 147, 147},
|
|
{128, 128, 128}, {108, 108, 108}, { 88, 88, 88},
|
|
{ 66, 66, 66}, { 45, 45, 45}, { 23, 23, 23},
|
|
},
|
|
};
|
|
static const uint8_t COLOR_PATTERNS_COUNT = (
|
|
sizeof(COLOR_PATTERNS) / sizeof(COLOR_PATTERNS[0]));
|
|
|
|
/**
|
|
* trimed down version of `ISSI3733_LED_MAP`:
|
|
*
|
|
* `ISSI3733_LED_MAP` is defined in keyboards/massdrop/ctrl/config_led.h is not directly usable,
|
|
* the numbers inside this map could probably be related to the PCB layout instead of
|
|
* the actual physical layout,
|
|
*
|
|
* this `ISSI3733_LED_MAP` is used somewhere in protocol/ but is not globally accessible
|
|
* so one is created here
|
|
*
|
|
* x and y are coordinates of the physical layout
|
|
* KC_ESC is (0, 0), gap between function keys and number rows is 1.5
|
|
* +y is downwards
|
|
* 1 unit is width/height of 1 standard keycap
|
|
*/
|
|
#define MAX_LED_ID ISSI3733_LED_COUNT
|
|
typedef struct led_info_s {
|
|
uint16_t id;
|
|
uint16_t scan;
|
|
float x;
|
|
float y;
|
|
uint8_t distance_to[MAX_LED_ID + 1];
|
|
} led_info_t;
|
|
led_info_t led_info[MAX_LED_ID + 1] = {
|
|
{ .id = 0 },
|
|
{ .id = 1, .x = 0.0, .y = 0.0, .scan = 41 }, // ESC
|
|
{ .id = 2, .x = 2.0, .y = 0.0, .scan = 58 }, // F1
|
|
{ .id = 3, .x = 3.0, .y = 0.0, .scan = 59 }, // F2
|
|
{ .id = 4, .x = 3.5, .y = 0.0, .scan = 60 }, // F3
|
|
{ .id = 5, .x = 5.0, .y = 0.0, .scan = 61 }, // F4
|
|
{ .id = 6, .x = 6.5, .y = 0.0, .scan = 62 }, // F5
|
|
{ .id = 7, .x = 7.5, .y = 0.0, .scan = 63 }, // F6
|
|
{ .id = 8, .x = 8.5, .y = 0.0, .scan = 64 }, // F7
|
|
{ .id = 9, .x = 9.5, .y = 0.0, .scan = 65 }, // F8
|
|
{ .id = 10, .x = 11, .y = 0.0, .scan = 66 }, // F9
|
|
{ .id = 11, .x = 12, .y = 0.0, .scan = 67 }, // F10
|
|
{ .id = 12, .x = 13, .y = 0.0, .scan = 68 }, // F11
|
|
{ .id = 13, .x = 14, .y = 0.0, .scan = 69 }, // F12
|
|
{ .id = 14, .x = 15.5, .y = 0.0, .scan = 70 }, // Print
|
|
{ .id = 15, .x = 16.5, .y = 0.0, .scan = 71 }, // Scoll Lock
|
|
{ .id = 16, .x = 17.5, .y = 0.0, .scan = 72 }, // Pause
|
|
{ .id = 17, .x = 0.0, .y = 1.5, .scan = 53 }, // `
|
|
{ .id = 18, .x = 1.0, .y = 1.5, .scan = 30 }, // 1
|
|
{ .id = 19, .x = 2.0, .y = 1.5, .scan = 31 }, // 2
|
|
{ .id = 20, .x = 3.0, .y = 1.5, .scan = 32 }, // 3
|
|
{ .id = 21, .x = 3.5, .y = 1.5, .scan = 33 }, // 4
|
|
{ .id = 22, .x = 5.0, .y = 1.5, .scan = 34 }, // 5
|
|
{ .id = 23, .x = 6.0, .y = 1.5, .scan = 35 }, // 6
|
|
{ .id = 24, .x = 7.0, .y = 1.5, .scan = 36 }, // 7
|
|
{ .id = 25, .x = 8.0, .y = 1.5, .scan = 37 }, // 8
|
|
{ .id = 26, .x = 9.0, .y = 1.5, .scan = 38 }, // 9
|
|
{ .id = 27, .x = 10.0, .y = 1.5, .scan = 39 }, // 0
|
|
{ .id = 28, .x = 11.0, .y = 1.5, .scan = 45 }, // -
|
|
{ .id = 29, .x = 12.0, .y = 1.5, .scan = 46 }, // =
|
|
{ .id = 30, .x = 13.5, .y = 1.5, .scan = 42 }, // Backspace
|
|
{ .id = 31, .x = 15.5, .y = 1.5, .scan = 73 }, // Insert
|
|
{ .id = 32, .x = 16.6, .y = 1.5, .scan = 74 }, // Home
|
|
{ .id = 33, .x = 17.5, .y = 1.5, .scan = 75 }, // Page Up
|
|
{ .id = 34, .x = 0.2, .y = 2.5, .scan = 43 }, // Tab
|
|
{ .id = 35, .x = 1.5, .y = 2.5, .scan = 20 }, // Q
|
|
{ .id = 36, .x = 2.5, .y = 2.5, .scan = 26 }, // W
|
|
{ .id = 37, .x = 3.5, .y = 2.5, .scan = 8 }, // E
|
|
{ .id = 38, .x = 4.5, .y = 2.5, .scan = 21 }, // R
|
|
{ .id = 39, .x = 5.5, .y = 2.5, .scan = 23 }, // T
|
|
{ .id = 40, .x = 6.5, .y = 2.5, .scan = 28 }, // Y
|
|
{ .id = 41, .x = 7.5, .y = 2.5, .scan = 24 }, // U
|
|
{ .id = 42, .x = 8.5, .y = 2.5, .scan = 12 }, // I
|
|
{ .id = 43, .x = 9.5, .y = 2.5, .scan = 18 }, // O
|
|
{ .id = 44, .x = 10.5, .y = 2.5, .scan = 19 }, // P
|
|
{ .id = 45, .x = 11.5, .y = 2.5, .scan = 47 }, // [
|
|
{ .id = 46, .x = 12.5, .y = 2.5, .scan = 48 }, // ]
|
|
{ .id = 47, .x = 13.75, .y = 2.5, .scan = 49 }, /* \ */
|
|
{ .id = 48, .x = 15.5, .y = 2.5, .scan = 76 }, // Delete
|
|
{ .id = 49, .x = 16.5, .y = 2.5, .scan = 77 }, // End
|
|
{ .id = 50, .x = 17.5, .y = 2.5, .scan = 78 }, // Page Down
|
|
{ .id = 51, .x = 0.4, .y = 3.5, .scan = 57 }, // Caps Lock
|
|
{ .id = 52, .x = 2.5, .y = 3.5, .scan = 4 }, // A
|
|
{ .id = 53, .x = 3.5, .y = 3.5, .scan = 22 }, // S
|
|
{ .id = 54, .x = 4.5, .y = 3.5, .scan = 7 }, // D
|
|
{ .id = 55, .x = 5.5, .y = 3.5, .scan = 9 }, // F
|
|
{ .id = 56, .x = 6.5, .y = 3.5, .scan = 10 }, // G
|
|
{ .id = 57, .x = 7.5, .y = 3.5, .scan = 11 }, // H
|
|
{ .id = 58, .x = 8.5, .y = 3.5, .scan = 13 }, // J
|
|
{ .id = 59, .x = 9.5, .y = 3.5, .scan = 14 }, // K
|
|
{ .id = 60, .x = 10.5, .y = 3.5, .scan = 15 }, // L
|
|
{ .id = 61, .x = 11.5, .y = 3.5, .scan = 51 }, // ;
|
|
{ .id = 62, .x = 12.5, .y = 3.5, .scan = 52 }, // '
|
|
{ .id = 63, .x = 13.5, .y = 3.5, .scan = 40 }, // Enter
|
|
{ .id = 64, .x = 0.5, .y = 4.5, .scan = 225 }, // LSHIFT
|
|
{ .id = 65, .x = 2.25, .y = 4.5, .scan = 29 }, // Z
|
|
{ .id = 66, .x = 3.25, .y = 4.5, .scan = 27 }, // X
|
|
{ .id = 67, .x = 4.25, .y = 4.5, .scan = 6 }, // C
|
|
{ .id = 68, .x = 5.25, .y = 4.5, .scan = 25 }, // V
|
|
{ .id = 69, .x = 6.25, .y = 4.5, .scan = 5 }, // B
|
|
{ .id = 70, .x = 7.25, .y = 4.5, .scan = 17 }, // N
|
|
{ .id = 71, .x = 8.25, .y = 4.5, .scan = 16 }, // M
|
|
{ .id = 72, .x = 9.25, .y = 4.5, .scan = 54 }, // COMMA
|
|
{ .id = 73, .x = 10.25, .y = 4.5, .scan = 55 }, // DOT
|
|
{ .id = 74, .x = 11.25, .y = 4.5, .scan = 56 }, // SLASH
|
|
{ .id = 75, .x = 13.2, .y = 4.5, .scan = 229 }, // RSHIFT
|
|
{ .id = 76, .x = 16.5, .y = 4.5, .scan = 82 }, // UP
|
|
{ .id = 77, .x = 0.1, .y = 5.5, .scan = 224 }, // LCTRL
|
|
{ .id = 78, .x = 1.25, .y = 5.5, .scan = 227 }, // WIN
|
|
{ .id = 79, .x = 2.5, .y = 5.5, .scan = 226 }, // LALT
|
|
{ .id = 80, .x = 6.25, .y = 5.5, .scan = 44 }, // SPACE
|
|
|
|
#define MAX_CACHED_SCAN_CODE 231
|
|
{ .id = 81, .x = 10.25, .y = 5.5, .scan = 230 }, // RALT
|
|
|
|
#define FN_KEY_LED_ID 82
|
|
#define FN_KEY_SCAN_CODE 20737
|
|
{ .id = 82, .x = 11.5, .y = 5.5, .scan = 20737 }, // FN
|
|
{ .id = 83, .x = 12.7, .y = 5.5, .scan = 101 }, // APP
|
|
{ .id = 84, .x = 13.75, .y = 5.5, .scan = 228 }, // RCTRL
|
|
{ .id = 85, .x = 15.5, .y = 5.5, .scan = 80 }, // LEFT
|
|
{ .id = 86, .x = 16.5, .y = 5.5, .scan = 81 }, // DOWN
|
|
{ .id = 87, .x = 17.5, .y = 5.5, .scan = 79 }, // RIGHT
|
|
|
|
#define MAX_LED_ID_WITH_SCANCODE 87
|
|
|
|
{ .id = 88, .x = 18.5, .y = 6.5, .scan = 255 },
|
|
{ .id = 89, .x = 16.917, .y = 6.5, .scan = 255 },
|
|
{ .id = 90, .x = 15.333, .y = 6.5, .scan = 255 },
|
|
{ .id = 91, .x = 13.75, .y = 6.5, .scan = 255 },
|
|
{ .id = 92, .x = 12.167, .y = 6.5, .scan = 255 },
|
|
{ .id = 93, .x = 10.583, .y = 6.5, .scan = 255 },
|
|
{ .id = 94, .x = 9, .y = 6.5, .scan = 255 },
|
|
{ .id = 95, .x = 7.417, .y = 6.5, .scan = 255 },
|
|
{ .id = 96, .x = 5.833, .y = 6.5, .scan = 255 },
|
|
{ .id = 97, .x = 4.25, .y = 6.5, .scan = 255 },
|
|
{ .id = 98, .x = 2.667, .y = 6.5, .scan = 255 },
|
|
{ .id = 99, .x = 1.083, .y = 6.5, .scan = 255 },
|
|
{ .id = 100, .x = -0.5, .y = 6.5, .scan = 255 },
|
|
{ .id = 101, .x = -0.5, .y = 4.75, .scan = 255 },
|
|
{ .id = 102, .x = -0.5, .y = 3, .scan = 255 },
|
|
{ .id = 103, .x = -0.5, .y = 1.25, .scan = 255 },
|
|
{ .id = 104, .x = -0.5, .y = -0.5, .scan = 255 },
|
|
{ .id = 105, .x = 1.083, .y = -0.5, .scan = 255 },
|
|
{ .id = 106, .x = 2.667, .y = -0.5, .scan = 255 },
|
|
{ .id = 107, .x = 4.25, .y = -0.5, .scan = 255 },
|
|
{ .id = 108, .x = 5.833, .y = -0.5, .scan = 255 },
|
|
{ .id = 109, .x = 7.417, .y = -0.5, .scan = 255 },
|
|
{ .id = 110, .x = 9, .y = -0.5, .scan = 255 },
|
|
{ .id = 111, .x = 10.583, .y = -0.5, .scan = 255 },
|
|
{ .id = 112, .x = 12.167, .y = -0.5, .scan = 255 },
|
|
{ .id = 113, .x = 13.75, .y = -0.5, .scan = 255 },
|
|
{ .id = 114, .x = 15.333, .y = -0.5, .scan = 255 },
|
|
{ .id = 115, .x = 16.917, .y = -0.5, .scan = 255 },
|
|
{ .id = 116, .x = 18.5, .y = 1.25, .scan = 255 },
|
|
{ .id = 117, .x = 18.5, .y = 3, .scan = 255 },
|
|
{ .id = 118, .x = 18.5, .y = 4.75, .scan = 255 },
|
|
{ .id = 119, .x = 18.5, .y = 6.5, .scan = 255 },
|
|
};
|
|
|
|
/**
|
|
* there are a few variables are used here
|
|
* keycode, scancode, led id
|
|
*
|
|
* scancode relates to actual physical key press
|
|
*
|
|
* keycode is software key press, or scancode with modifiers (shift, ctrl, alt, etc.),
|
|
* keycode with the value less than 255 are usually the same with scan code (I hope so)
|
|
*
|
|
* the led pattern are running based on led id, because led on the keyboard
|
|
* are not limited to keys only
|
|
*/
|
|
led_info_t* get_led_info_by_scancode(uint16_t scancode){
|
|
static bool init = false;
|
|
static led_info_t* scancode_to_led_info[MAX_CACHED_SCAN_CODE + 1];
|
|
if(!init){
|
|
for(int i = 1; i <= MAX_LED_ID_WITH_SCANCODE; ++i){
|
|
uint16_t scan = led_info[i].scan;
|
|
if(scan <= MAX_CACHED_SCAN_CODE){
|
|
scancode_to_led_info[scan] = (led_info + i);
|
|
}
|
|
}
|
|
init = true;
|
|
}
|
|
|
|
if(scancode <= MAX_CACHED_SCAN_CODE){
|
|
return scancode_to_led_info[scancode];
|
|
} else if(scancode == FN_KEY_SCAN_CODE){ // FN
|
|
return (led_info + FN_KEY_LED_ID);
|
|
}
|
|
return led_info;
|
|
}
|
|
|
|
|
|
|
|
void init_led_info(void){
|
|
for(int i = 1; i <= MAX_LED_ID; ++i){
|
|
led_info_t *entry1 = led_info + i;
|
|
for(int j = i; j <= MAX_LED_ID; ++j){
|
|
led_info_t *entry2 = led_info + j;
|
|
/**
|
|
* distance is tripled because
|
|
* convertion from float to int reduces accuracy
|
|
*
|
|
*/
|
|
uint8_t distance = (uint8_t)sqrtf(
|
|
powf(entry1->x - entry2->x, 2.0) +
|
|
powf(entry1->y - entry2->y, 2.0)) *
|
|
DISTANCE_NORAMLIZING_PARAMETER;
|
|
entry1->distance_to[j] = distance;
|
|
entry2->distance_to[i] = distance;
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
// Runs just one time when the keyboard initializes.
|
|
void matrix_init_user(void) {
|
|
init_led_info();
|
|
};
|
|
|
|
typedef struct keystroke_s {
|
|
uint16_t scancode;
|
|
uint32_t timer;
|
|
bool active;
|
|
} keystroke_t;
|
|
|
|
#define MAX_ACTIVE_KEYSTORKES 10
|
|
keystroke_t ACTIVE_KEYSTROKES[MAX_ACTIVE_KEYSTORKES];
|
|
|
|
void reset_led_for_instruction(int led_instruction_index){
|
|
led_instructions[led_instruction_index].id0 = 0;
|
|
led_instructions[led_instruction_index].id1 = 0;
|
|
led_instructions[led_instruction_index].id2 = 0;
|
|
led_instructions[led_instruction_index].id3 = 0;
|
|
};
|
|
void add_led_to_instruction(int led_instruction_index, int led_id){
|
|
if(32 >= led_id && led_id >= 1){
|
|
led_instructions[led_instruction_index].id0 += ( 1 << (led_id - 1) );
|
|
} else if(64 >= led_id){
|
|
led_instructions[led_instruction_index].id1 += ( 1 << (led_id - 33) );
|
|
} else if(96 >= led_id){
|
|
led_instructions[led_instruction_index].id2 += ( 1 << (led_id - 65) );
|
|
} else if(128 >= led_id){
|
|
led_instructions[led_instruction_index].id3 += ( 1 << (led_id - 97) );
|
|
}
|
|
};
|
|
|
|
|
|
void wave_effect(void);
|
|
void set_wave_color(int);
|
|
// Runs constantly in the background, in a loop.
|
|
void matrix_scan_user(void) {
|
|
wave_effect();
|
|
set_wave_color(USER_CONFIG.PATTERN_INDEX);
|
|
};
|
|
|
|
|
|
#define MODS_SHIFT (get_mods() & MOD_BIT(KC_LSHIFT) || get_mods() & MOD_BIT(KC_RSHIFT))
|
|
#define MODS_CTRL (get_mods() & MOD_BIT(KC_LCTL) || get_mods() & MOD_BIT(KC_RCTRL))
|
|
#define MODS_ALT (get_mods() & MOD_BIT(KC_LALT) || get_mods() & MOD_BIT(KC_RALT))
|
|
|
|
void register_keystroke(uint16_t keycode){
|
|
if(get_led_info_by_scancode(keycode)->id){
|
|
uint32_t oldest_keystroke_lifespan = 0;
|
|
int8_t oldest_keystroke_index = -1;
|
|
bool registered = false;
|
|
|
|
keystroke_t *keystroke = ACTIVE_KEYSTROKES;
|
|
for(int i = 0; i < MAX_ACTIVE_KEYSTORKES; ++i){
|
|
if(!keystroke->active){
|
|
keystroke->scancode = keycode;
|
|
keystroke->timer = timer_read32();
|
|
keystroke->active = true;
|
|
registered = true;
|
|
break;
|
|
}
|
|
|
|
uint32_t lifespan = timer_elapsed32(keystroke->timer);
|
|
if(lifespan > oldest_keystroke_lifespan){
|
|
oldest_keystroke_index = i;
|
|
oldest_keystroke_lifespan = lifespan;
|
|
}
|
|
|
|
++keystroke;
|
|
}
|
|
|
|
// override the oldest keystroke
|
|
if(!registered){
|
|
keystroke = ACTIVE_KEYSTROKES + oldest_keystroke_index;
|
|
keystroke->scancode = keycode;
|
|
keystroke->timer = timer_read32();
|
|
keystroke->active = true; // presumably active already
|
|
}
|
|
}
|
|
}
|
|
|
|
bool process_record_user(uint16_t keycode, keyrecord_t *record) {
|
|
static uint32_t key_timer;
|
|
|
|
switch (keycode) {
|
|
case L_BRI:
|
|
if (record->event.pressed) {
|
|
if (LED_GCR_STEP > LED_GCR_MAX - gcr_desired) gcr_desired = LED_GCR_MAX;
|
|
else gcr_desired += LED_GCR_STEP;
|
|
if (led_animation_breathing) gcr_breathe = gcr_desired;
|
|
}
|
|
return false;
|
|
case L_BRD:
|
|
if (record->event.pressed) {
|
|
if (LED_GCR_STEP > gcr_desired) gcr_desired = 0;
|
|
else gcr_desired -= LED_GCR_STEP;
|
|
if (led_animation_breathing) gcr_breathe = gcr_desired;
|
|
}
|
|
return false;
|
|
case L_PTN:
|
|
if (record->event.pressed) {
|
|
if (led_animation_id == led_setups_count - 1) led_animation_id = 0;
|
|
else led_animation_id++;
|
|
}
|
|
return false;
|
|
case L_PTP:
|
|
if (record->event.pressed) {
|
|
if (led_animation_id == 0) led_animation_id = led_setups_count - 1;
|
|
else led_animation_id--;
|
|
}
|
|
return false;
|
|
case L_PSI:
|
|
if (record->event.pressed) {
|
|
led_animation_speed += ANIMATION_SPEED_STEP;
|
|
}
|
|
return false;
|
|
case L_PSD:
|
|
if (record->event.pressed) {
|
|
led_animation_speed -= ANIMATION_SPEED_STEP;
|
|
if (led_animation_speed < 0) led_animation_speed = 0;
|
|
}
|
|
return false;
|
|
case L_T_MD:
|
|
if (record->event.pressed) {
|
|
led_lighting_mode++;
|
|
if (led_lighting_mode > LED_MODE_MAX_INDEX) led_lighting_mode = LED_MODE_NORMAL;
|
|
}
|
|
return false;
|
|
case L_T_ONF:
|
|
if (record->event.pressed) {
|
|
led_enabled = !led_enabled;
|
|
I2C3733_Control_Set(led_enabled);
|
|
}
|
|
return false;
|
|
case L_ON:
|
|
if (record->event.pressed) {
|
|
led_enabled = 1;
|
|
I2C3733_Control_Set(led_enabled);
|
|
}
|
|
return false;
|
|
case L_OFF:
|
|
if (record->event.pressed) {
|
|
led_enabled = 0;
|
|
I2C3733_Control_Set(led_enabled);
|
|
}
|
|
return false;
|
|
case L_T_BR:
|
|
if (record->event.pressed) {
|
|
led_animation_breathing = !led_animation_breathing;
|
|
if (led_animation_breathing) {
|
|
gcr_breathe = gcr_desired;
|
|
led_animation_breathe_cur = BREATHE_MIN_STEP;
|
|
breathe_dir = 1;
|
|
}
|
|
}
|
|
return false;
|
|
case L_T_PTD:
|
|
if (record->event.pressed) {
|
|
led_animation_direction = !led_animation_direction;
|
|
}
|
|
return false;
|
|
case U_T_AGCR:
|
|
if (record->event.pressed && MODS_SHIFT && MODS_CTRL) {
|
|
TOGGLE_FLAG_AND_PRINT(usb_gcr_auto, "USB GCR auto mode");
|
|
}
|
|
return false;
|
|
case DBG_TOG:
|
|
if (record->event.pressed) {
|
|
TOGGLE_FLAG_AND_PRINT(debug_enable, "Debug mode");
|
|
}
|
|
return false;
|
|
case DBG_MTRX:
|
|
if (record->event.pressed) {
|
|
TOGGLE_FLAG_AND_PRINT(debug_matrix, "Debug matrix");
|
|
}
|
|
return false;
|
|
case DBG_KBD:
|
|
if (record->event.pressed) {
|
|
TOGGLE_FLAG_AND_PRINT(debug_keyboard, "Debug keyboard");
|
|
}
|
|
return false;
|
|
case DBG_MOU:
|
|
if (record->event.pressed) {
|
|
TOGGLE_FLAG_AND_PRINT(debug_mouse, "Debug mouse");
|
|
}
|
|
return false;
|
|
case MD_BOOT:
|
|
if (record->event.pressed) {
|
|
key_timer = timer_read32();
|
|
} else {
|
|
if (timer_elapsed32(key_timer) >= 500) {
|
|
reset_keyboard();
|
|
}
|
|
}
|
|
return false;
|
|
|
|
|
|
|
|
case S_RESET:
|
|
// reset all parameters
|
|
|
|
USER_CONFIG.PATTERN_INDEX = 1;
|
|
USER_CONFIG.WAVE_WIDTH = 10;
|
|
USER_CONFIG.WAVE_SPEED = 15;
|
|
USER_CONFIG.COLOR_PATTERN_INDEX = 0;
|
|
USER_CONFIG.TRAVEL_DISTANCE = 25;
|
|
|
|
return false;
|
|
case L_SP_PR: // previous dripple pattern
|
|
case L_SP_NE: // next dripple pattern
|
|
if (record->event.pressed) {
|
|
#define PATTERN_COUNT 7
|
|
uint8_t incre = keycode == L_SP_PR ? PATTERN_COUNT-1 : 1;
|
|
USER_CONFIG.PATTERN_INDEX += incre;
|
|
USER_CONFIG.PATTERN_INDEX %= PATTERN_COUNT;
|
|
|
|
if(USER_CONFIG.PATTERN_INDEX <= 4){
|
|
USER_CONFIG.TRAVEL_DISTANCE = 25;
|
|
USER_CONFIG.COLOR_PATTERN_INDEX = 0;
|
|
USER_CONFIG.WAVE_SPEED = 10;
|
|
}
|
|
|
|
switch(USER_CONFIG.PATTERN_INDEX){
|
|
case 0: // None
|
|
break;
|
|
case 1: // background off, wave on
|
|
USER_CONFIG.WAVE_WIDTH = 2;
|
|
break;
|
|
case 2: // background on, wave off
|
|
USER_CONFIG.WAVE_WIDTH = 5;
|
|
break;
|
|
case 3: // background off, rainbow wave
|
|
USER_CONFIG.WAVE_WIDTH = 10;
|
|
break;
|
|
case 4: // background on, rainbow wave
|
|
USER_CONFIG.WAVE_WIDTH = 10;
|
|
break;
|
|
case 5:
|
|
USER_CONFIG.WAVE_WIDTH = 10;
|
|
|
|
USER_CONFIG.COLOR_PATTERN_INDEX = 2;
|
|
USER_CONFIG.TRAVEL_DISTANCE = 0;
|
|
USER_CONFIG.WAVE_SPEED = 10;
|
|
break;
|
|
case 6:
|
|
USER_CONFIG.WAVE_WIDTH = 10;
|
|
|
|
USER_CONFIG.COLOR_PATTERN_INDEX = 3;
|
|
USER_CONFIG.TRAVEL_DISTANCE = 2;
|
|
USER_CONFIG.WAVE_SPEED = 10;
|
|
break;
|
|
}
|
|
|
|
// remove effect after changing pattern
|
|
for(int i = 0; i < MAX_ACTIVE_KEYSTORKES; ++i){
|
|
ACTIVE_KEYSTROKES[i].active = 0;
|
|
}
|
|
}
|
|
return false;
|
|
case L_SP_WD:
|
|
case L_SP_NW:
|
|
if(record->event.pressed){
|
|
short incre = keycode == L_SP_WD ? 1 : -1;
|
|
USER_CONFIG.WAVE_WIDTH += incre;
|
|
if(USER_CONFIG.WAVE_WIDTH < 1){
|
|
USER_CONFIG.WAVE_WIDTH = 1;
|
|
}
|
|
}
|
|
return false;
|
|
case L_SP_FA:
|
|
case L_SP_SL:
|
|
if(record->event.pressed){
|
|
short incre = keycode == L_SP_FA ? -1 : 1;
|
|
|
|
USER_CONFIG.WAVE_SPEED += incre;
|
|
if(USER_CONFIG.WAVE_SPEED > 50){
|
|
USER_CONFIG.WAVE_SPEED = 50;
|
|
} else if(USER_CONFIG.WAVE_SPEED < 1){
|
|
USER_CONFIG.WAVE_SPEED = 1;
|
|
}
|
|
}
|
|
return false;
|
|
// these are the keys not in range 0x04 - 0x52
|
|
case L_CP_PR:
|
|
case L_CP_NX:
|
|
if(record->event.pressed){
|
|
uint8_t incre = keycode == L_CP_PR ? COLOR_PATTERNS_COUNT - 1 : 1;
|
|
USER_CONFIG.COLOR_PATTERN_INDEX += incre;
|
|
USER_CONFIG.COLOR_PATTERN_INDEX %= COLOR_PATTERNS_COUNT;
|
|
set_wave_color(USER_CONFIG.COLOR_PATTERN_INDEX);
|
|
}
|
|
return false;
|
|
|
|
default:
|
|
|
|
|
|
if(record->event.pressed){
|
|
register_keystroke(keycode);
|
|
|
|
#ifdef CONSOLE_ENABLE
|
|
led_info_t *entry = get_led_info_by_scancode(keycode);
|
|
uprintf(("KL: kc: %u, led id: %u, x: %f, y: %f, "
|
|
"col: %u, row: %u, pressed: %u, time: %u\n"),
|
|
keycode, entry->id, entry->x, entry->y,
|
|
record->event.key.col, record->event.key.row,
|
|
record->event.pressed, record->event.time);
|
|
#endif
|
|
}
|
|
return true; //Process all other keycodes normally
|
|
}
|
|
}
|
|
|
|
led_instruction_t led_instructions[] = {
|
|
//LEDs are normally inactive, no processing is performed on them
|
|
//Flags are used in matching criteria for an LED to be active and indicate how to color it
|
|
//Flags can be found in tmk_core/protocol/arm_atsam/led_matrix.h (prefixed with LED_FLAG_)
|
|
//LED IDs can be found in config_led.h in the keyboard's directory
|
|
//Examples are below
|
|
|
|
//All LEDs use the user's selected pattern (this is the factory default)
|
|
{ .flags = LED_FLAG_USE_ROTATE_PATTERN },
|
|
|
|
//Specific LEDs use the user's selected pattern while all others are off
|
|
// { .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_ROTATE_PATTERN, .id0 = 0xFFFFFFFF, .id1 = 0xAAAAAAAA, .id2 = 0x55555555, .id3 = 0x11111111 },
|
|
|
|
//Specific LEDs use specified RGB values while all others are off
|
|
// { .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0xFF, .id1 = 0x00FF, .id2 = 0x0000FF00, .id3 = 0xFF000000, .r = 75, .g = 150, .b = 225 },
|
|
|
|
//All LEDs use the user's selected pattern
|
|
//On layer 1, all key LEDs (except the top row which keeps active pattern) are red while all edge LEDs are green
|
|
//When layer 1 is active, key LEDs use red (id0 32 - 17: 1111 1111 1111 1111 0000 0000 0000 0000 = 0xFFFF0000) (except top row 16 - 1)
|
|
//When layer 1 is active, key LEDs use red (id1 64 - 33: 1111 1111 1111 1111 1111 1111 1111 1111 = 0xFFFFFFFF)
|
|
//When layer 1 is active, key LEDs use red (id2 87 - 65: 0000 0000 0111 1111 1111 1111 1111 1111 = 0x007FFFFF)
|
|
//When layer 1 is active, edge LEDs use green (id2 95 - 88: 1111 1111 1000 0000 0000 0000 0000 0000 = 0xFF800000)
|
|
//When layer 1 is active, edge LEDs use green (id3 119 - 96: 0000 0000 1111 1111 1111 1111 1111 1111 = 0x00FFFFFF)
|
|
// { .flags = LED_FLAG_USE_ROTATE_PATTERN },
|
|
|
|
#define WAVE_LED_INSTRUCTION_START 1
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0, .id1 = 0, .id2 = 0, .g = 255 },
|
|
#define WAVE_LED_INSTRUCTION_END 18
|
|
|
|
//All key LEDs use red while edge LEDs use the active pattern
|
|
//All key LEDs use red (id0 32 - 1: 1111 1111 1111 1111 1111 1111 1111 1111 = 0xFFFFFFFF)
|
|
//All key LEDs use red (id1 64 - 33: 1111 1111 1111 1111 1111 1111 1111 1111 = 0xFFFFFFFF)
|
|
//All key LEDs use red (id2 87 - 65: 0000 0000 0111 1111 1111 1111 1111 1111 = 0x007FFFFF)
|
|
//Edge uses active pattern (id2 95 - 88: 1111 1111 1000 0000 0000 0000 0000 0000 = 0xFF800000)
|
|
//Edge uses active pattern (id3 119 - 96: 0000 0000 1111 1111 1111 1111 1111 1111 = 0x00FFFFFF)
|
|
// { .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB, .id0 = 0xFFFFFFFF, .id1 = 0xFFFFFFFF, .id2 = 0x007FFFFF, .r = 255 },
|
|
// { .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_ROTATE_PATTERN , .id2 = 0xFF800000, .id3 = 0x00FFFFFF },
|
|
|
|
{ .flags = LED_FLAG_MATCH_ID | LED_FLAG_USE_RGB | LED_FLAG_MATCH_LAYER,
|
|
.id1 = 0b00001111001111000000011110011110,
|
|
.r = 0, .g = 255, .b = 60, .layer = 2 },
|
|
|
|
//end must be set to 1 to indicate end of instruction set
|
|
{ .end = 1 }
|
|
};
|
|
|
|
|
|
void set_wave_color(int color_pattern_index){
|
|
for(int i = WAVE_LED_INSTRUCTION_START; i < WAVE_LED_INSTRUCTION_END; ++i){
|
|
for(int j = 0; j < COLOR_PATTERN_RGB_COUNT; ++j){
|
|
led_instructions[i].r = COLOR_PATTERNS[color_pattern_index][i][0];
|
|
led_instructions[i].g = COLOR_PATTERNS[color_pattern_index][i][1];
|
|
led_instructions[i].b = COLOR_PATTERNS[color_pattern_index][i][2];
|
|
}
|
|
}
|
|
};
|
|
|
|
void wave_effect(void){
|
|
for(int i = WAVE_LED_INSTRUCTION_START; i < WAVE_LED_INSTRUCTION_END; ++i){
|
|
reset_led_for_instruction(i);
|
|
}
|
|
int wave_led_instruction_span = WAVE_LED_INSTRUCTION_END - WAVE_LED_INSTRUCTION_START;
|
|
|
|
|
|
keystroke_t *keystroke = ACTIVE_KEYSTROKES;
|
|
for(int i = 0; i < MAX_ACTIVE_KEYSTORKES; ++i, ++keystroke){
|
|
if(!keystroke->active) continue;
|
|
bool active = false;
|
|
|
|
uint16_t keystroke_led_id = get_led_info_by_scancode(keystroke->scancode)->id;
|
|
|
|
float elapsed_s = timer_elapsed32(keystroke->timer) / 1000.0f;
|
|
float travel = elapsed_s * USER_CONFIG.WAVE_SPEED;
|
|
|
|
for(uint16_t id = 1; id <= MAX_LED_ID; ++id){
|
|
float normalized_distance =
|
|
led_info[id].distance_to[keystroke_led_id] /
|
|
(float)DISTANCE_NORAMLIZING_PARAMETER;
|
|
|
|
if(travel >= normalized_distance && travel - normalized_distance >= 0 &&
|
|
normalized_distance >= travel - USER_CONFIG.WAVE_WIDTH){
|
|
int portion = (travel - normalized_distance) *
|
|
wave_led_instruction_span / USER_CONFIG.WAVE_WIDTH;
|
|
add_led_to_instruction(portion, id);
|
|
|
|
active = true;
|
|
}
|
|
}
|
|
|
|
keystroke->active = active;
|
|
}
|
|
};
|