You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

122 lines
3.7 KiB

  1. // Copyright 2023 John Barbero Unenge (@jbarberu)
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include "quantum.h"
  4. // oled keylog rendering has been kindly borrowed from crkbd <3
  5. char key_name = ' ';
  6. uint16_t last_keycode;
  7. uint8_t last_row;
  8. uint8_t last_col;
  9. static const char PROGMEM code_to_name[60] = {' ', ' ', ' ', ' ', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', 'R', 'E', 'B', 'T', '_', '-', '=', '[', ']', '\\', '#', ';', '\'', '`', ',', '.', '/', ' ', ' ', ' '};
  10. static void set_keylog(uint16_t keycode, keyrecord_t *record) {
  11. last_row = record->event.key.row;
  12. last_col = record->event.key.col;
  13. key_name = ' ';
  14. last_keycode = keycode;
  15. if (IS_QK_MOD_TAP(keycode)) {
  16. if (record->tap.count) {
  17. keycode = QK_MOD_TAP_GET_TAP_KEYCODE(keycode);
  18. } else {
  19. keycode = 0xE0 + biton(QK_MOD_TAP_GET_MODS(keycode) & 0xF) + biton(QK_MOD_TAP_GET_MODS(keycode) & 0x10);
  20. }
  21. } else if (IS_QK_LAYER_TAP(keycode) && record->tap.count) {
  22. keycode = QK_LAYER_TAP_GET_TAP_KEYCODE(keycode);
  23. } else if (IS_QK_MODS(keycode)) {
  24. keycode = QK_MODS_GET_BASIC_KEYCODE(keycode);
  25. } else if (IS_QK_ONE_SHOT_MOD(keycode)) {
  26. keycode = 0xE0 + biton(QK_ONE_SHOT_MOD_GET_MODS(keycode) & 0xF) + biton(QK_ONE_SHOT_MOD_GET_MODS(keycode) & 0x10);
  27. }
  28. if (keycode > ARRAY_SIZE(code_to_name)) {
  29. return;
  30. }
  31. // update keylog
  32. key_name = pgm_read_byte(&code_to_name[keycode]);
  33. }
  34. static const char *depad_str(const char *depad_str, char depad_char) {
  35. while (*depad_str == depad_char) {
  36. ++depad_str;
  37. }
  38. return depad_str;
  39. }
  40. static void oled_render_keylog(void) {
  41. oled_write_char('0' + last_row, false);
  42. oled_write("x", false);
  43. oled_write_char('0' + last_col, false);
  44. oled_write(", k", false);
  45. const char *last_keycode_str = get_u16_str(last_keycode, ' ');
  46. oled_write(depad_str(last_keycode_str, ' '), false);
  47. oled_write(":", false);
  48. oled_write_char(key_name, false);
  49. }
  50. __attribute__((weak)) const char * get_layer_name_user(uint8_t layer) {
  51. return get_u8_str(layer, ' ');
  52. }
  53. static void oled_render_layer(void) {
  54. oled_write("Layer: ", false);
  55. oled_write_ln(get_layer_name_user(get_highest_layer(layer_state)), false);
  56. }
  57. bool oled_task_kb(void) {
  58. if (!oled_task_user()) {
  59. return false;
  60. }
  61. oled_render_layer();
  62. oled_render_keylog();
  63. oled_advance_page(true);
  64. return false;
  65. }
  66. static void setupForFlashing(void) {
  67. oled_clear();
  68. oled_write(" ", false);
  69. oled_write(" In flash mode... ", false);
  70. oled_write(" ", false);
  71. oled_write(" ", false);
  72. // Force data to be rendered
  73. oled_render_dirty(true);
  74. // Set alternating backlight colors
  75. const uint8_t max = 20;
  76. rgblight_mode_noeeprom(RGBLIGHT_MODE_STATIC_LIGHT);
  77. for (size_t i = 0; i < RGBLIGHT_LED_COUNT; ++i) {
  78. rgb_led_t *led_ = (rgb_led_t *)&led[i];
  79. switch (i % 2) {
  80. case 0:
  81. setrgb(max, 0, max, led_);
  82. break;
  83. case 1:
  84. setrgb(0, max, max, led_);
  85. break;
  86. }
  87. }
  88. rgblight_set();
  89. }
  90. bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
  91. if (record->event.pressed) {
  92. set_keylog(keycode, record);
  93. }
  94. if (keycode == QK_BOOT) {
  95. setupForFlashing();
  96. }
  97. return process_record_user(keycode, record);
  98. }
  99. void keyboard_post_init_kb(void) {
  100. rgblight_enable_noeeprom();
  101. rgblight_sethsv_noeeprom(HSV_MAGENTA);
  102. rgblight_mode_noeeprom(RGBLIGHT_MODE_RAINBOW_SWIRL);
  103. keyboard_post_init_user();
  104. }