qmk-keychron-q3-colemak-dh/keyboards/xiudi/xd75/keymaps/odyssey/keymap.c
2023-09-04 01:19:59 +01:00

238 lines
7.3 KiB
C

#include QMK_KEYBOARD_H
// Layer shorthand
#define _QW 0
#define _CM 1
#define _FN 2
typedef enum {
NONE_NONE,
SINGLE_TAP,
SINGLE_HOLD,
DOUBLE_SINGLE_TAP
} td_state_t;
static td_state_t td_state = NONE_NONE;
int cur_dance (tap_dance_state_t *state);
void altlp_finished (tap_dance_state_t *state, void *user_data);
void altlp_reset (tap_dance_state_t *state, void *user_data);
// Four differend underglow states for 2 language layouts x 2 states of colemak layer
enum layer_states {
S_ENGLISH,
S_COLEMAK,
S_CRUSSIAN,
S_QRUSSIAN
} l_state = S_ENGLISH;
// CLMK to toggle colemak (active only with english qwerty, saves intent in russian but does nothing untill switched back to english)
// RNBW - rainbow underglow
// PING - ping pong underglow
// SP_UP/DW - underglow speed controls
enum custom_keycodes {
STUB = SAFE_RANGE,
CLMK,
RNBW,
PING,
SP_UP,
SP_DW
};
// Escape if taped, FN layer mod if held
enum tapdance_codes {
LESC
};
bool sstate = false;
int rgb = 0;
int speed = 300;
void set_colors(int r, int g, int b) {
for(int i = 0; i<6; i++) {
sethsv(r, g, b, (rgb_led_t *)&led[i]);
}
rgblight_set();
}
void update_led_state_c(void) {
if (rgb>0) return;
if (td_state == SINGLE_HOLD) set_colors(HSV_BLUE);
else {
if (l_state == S_ENGLISH) set_colors(HSV_RED);
if (l_state == S_QRUSSIAN) set_colors(HSV_GREEN);
if (l_state == S_COLEMAK) set_colors(HSV_YELLOW);
if (l_state == S_CRUSSIAN) set_colors(85, 100, 100);
}
}
int cur_dance (tap_dance_state_t *state) {
if (state->count == 1) {
if (state->interrupted && state->pressed) { return SINGLE_HOLD; }
else if (!state->pressed) { return SINGLE_TAP; }
else { return SINGLE_HOLD; }
}
if (state->count == 2) { return DOUBLE_SINGLE_TAP; }
else { return 3; }
}
// Tapdance code stolen long time ago from one of the forum answers i found related to my problem, sadly can't provide link for the credits
void lesc_finished (tap_dance_state_t *state, void *user_data) {
td_state = cur_dance(state);
switch (td_state) {
case SINGLE_TAP:
register_code16(KC_ESC);
break;
case SINGLE_HOLD:
layer_on(_FN);
update_led_state_c();
sstate = false;
break;
case DOUBLE_SINGLE_TAP:
break;
case NONE_NONE:
break;
}
}
void lesc_reset (tap_dance_state_t *state, void *user_data) {
switch (td_state) {
case SINGLE_TAP:
unregister_code16(KC_ESC);
break;
case SINGLE_HOLD:
layer_off(_FN);
if (sstate == false) {
register_code16(KC_ESC);
unregister_code16(KC_ESC);
}
break;
case DOUBLE_SINGLE_TAP:
break;
case NONE_NONE:
break;
}
td_state = NONE_NONE;
update_led_state_c();
}
tap_dance_action_t tap_dance_actions[] = {
[LESC] = ACTION_TAP_DANCE_FN_ADVANCED(NULL, lesc_finished, lesc_reset)
};
const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
[_QW] = LAYOUT_ortho_5x15( /* QWERTY */
KC_GRV, KC_1, KC_2, KC_3, KC_4, KC_5, KC_MINS, RNBW, KC_EQL, KC_6, KC_7, KC_8, KC_9, KC_0, KC_BSPC,
KC_TAB, KC_Q, KC_W, KC_E, KC_R, KC_T, KC_LBRC, KC_BSLS, KC_RBRC, KC_Y, KC_U, KC_I, KC_O, KC_P, KC_QUOT,
KC_CAPS, KC_A, KC_S, KC_D, KC_F, KC_G, CLMK, KC_DEL, SP_UP, KC_H, KC_J, KC_K, KC_L, KC_SCLN, KC_ENT,
KC_LSFT, KC_Z, KC_X, KC_C, KC_V, KC_B, _______, KC_UP, SP_DW, KC_N, KC_M, KC_COMM, KC_DOT, KC_SLSH, KC_RSFT,
KC_LCTL, LSFT(KC_9), KC_LALT, KC_LGUI, TD(LESC),KC_SPC, KC_LEFT, KC_DOWN, KC_RGHT, KC_ENT, TD(LESC),KC_RGUI, KC_RALT, RSFT(KC_0),KC_RCTL
),
[_CM] = LAYOUT_ortho_5x15( /* COLEMAK */
_______, _______, _______, _______, _______, _______, _______, PING, _______, _______, _______, _______, _______, _______, _______,
_______, KC_Q, KC_W, KC_F, KC_P, KC_Z, _______, _______, _______, KC_J, KC_L, KC_U, KC_Y, KC_SCLN, KC_SLSH,
_______, KC_A, KC_R, KC_S, KC_T, KC_G, _______, _______, _______, KC_M, KC_N, KC_E, KC_I, KC_O, _______,
_______, KC_X, KC_V, KC_C, KC_D, KC_B, _______, _______, _______, KC_K, KC_H, KC_DOT, KC_COMM, KC_QUOT, _______,
_______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
),
[_FN] = LAYOUT_ortho_5x15( /* FUNCTION */
KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_P7, KC_P8, KC_P9, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12,
QK_BOOT, KC_BTN1, KC_MS_U, KC_BTN2, KC_BTN3, KC_WH_U, KC_P4, KC_P5, KC_P6, _______, _______, _______, RSFT(KC_MINS), LSFT(KC_MINS), _______,
_______, KC_MS_L, KC_MS_D, KC_MS_R, KC_MINS, KC_WH_D, KC_P1, KC_P2, KC_P3, _______, KC_PPLS, KC_ACL0, KC_ACL1, KC_ACL2, _______,
_______, KC_ACL0, KC_ACL1, KC_ACL2, KC_WH_L, KC_WH_R, KC_NUM, KC_PGUP, KC_P0, _______, _______, _______, _______, _______, _______,
RGB_TOG, _______, _______, _______, KC_BSPC, KC_BSPC, KC_HOME, KC_PGDN, KC_END, KC_BSPC, KC_BSPC, _______, _______, _______, _______
)
};
bool process_record_user(uint16_t keycode, keyrecord_t *record) {
if(keycode != TD(LESC))
sstate = true;
switch (keycode) {
case SP_UP:
if (speed>25)
speed -= 25;
break;
case SP_DW:
if (speed < 1000)
speed += 25;
break;
case RNBW:
if(record->event.pressed) {
rgb = rgb == 0 ? 1 : 0;
update_led_state_c();
}
break;
case PING:
if(record->event.pressed) {
rgb = rgb == 0 ? 2 : 0;
update_led_state_c();
}
break;
case CLMK:
if(record->event.pressed) {
if (l_state == S_COLEMAK) {
layer_off(_CM);
l_state = S_ENGLISH;
}
else if (l_state == S_ENGLISH) {
layer_on(_CM);
l_state = S_COLEMAK;
}
else if (l_state == S_CRUSSIAN) { l_state = S_QRUSSIAN; }
else if (l_state == S_QRUSSIAN) { l_state = S_CRUSSIAN; }
}
update_led_state_c();
break;
case KC_CAPS:
if(record->event.pressed) {
if (l_state == S_ENGLISH) l_state = S_QRUSSIAN;
else if (l_state == S_QRUSSIAN) l_state = S_ENGLISH;
else if (l_state == S_COLEMAK) {
layer_off(_CM);
l_state = S_CRUSSIAN;
}
else if (l_state == S_CRUSSIAN) {
layer_on(_CM);
l_state = S_COLEMAK;
}
update_led_state_c();
}
break;
}
return true;
}
// Rainbow globals
int t = 0;
int rc = 0;
int col = 0;
bool right = false;
void matrix_scan_user(void) {
if (rgb==0) return; // Do nothing each tick right away if neither of animations enabled
if (rgb==1) {
if (rc == 0) {
rc = speed;
for(int i = 0; i<6; i++) {
sethsv(42*((t+i)%6), 255, 255, (rgb_led_t *)&led[i]);
}
rgblight_set();
t++; t = t % 6;
}
} else if (rgb==2) {
if (rc == 0) {
rc = speed;
col = (col + 1) % 36;
for (int i = 0; i<6; i++) {
if (i==t)
sethsv(42*(((col-1)/6)%6), 255, 255, (rgb_led_t *)&led[(right ? t : 5-t)]);
else
sethsv(0, 0, 0, (rgb_led_t *)&led[right ? i : 5-i]);
}
rgblight_set();
t++; t = t % 6;
if (t == 0) right = !right;
}
}
rc--;
}