#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 }; 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/md_rgb_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; } };