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.

301 lines
10 KiB

  1. /* Copyright 2020 Christopher Courtney, aka Drashna Jael're (@drashna) <drashna@live.com>
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include "drashna.h"
  17. extern uint8_t is_master;
  18. #ifndef KEYLOGGER_LENGTH
  19. // # ifdef OLED_DISPLAY_128X64
  20. # define KEYLOGGER_LENGTH ((int)(OLED_DISPLAY_HEIGHT / OLED_FONT_WIDTH))
  21. // # else
  22. // # define KEYLOGGER_LENGTH (uint8_t *(OLED_DISPLAY_WIDTH / OLED_FONT_HEIGHT))
  23. // # endif
  24. #endif
  25. static uint32_t oled_timer = 0;
  26. static char keylog_str[KEYLOGGER_LENGTH + 1] = {"\n"};
  27. static uint16_t log_timer = 0;
  28. // clang-format off
  29. static const char PROGMEM code_to_name[0xFF] = {
  30. // 0 1 2 3 4 5 6 7 8 9 A B c D E F
  31. ' ', ' ', ' ', ' ', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', // 0x
  32. 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '1', '2', // 1x
  33. '3', '4', '5', '6', '7', '8', '9', '0', 20, 19, 27, 26, 22, '-', '=', '[', // 2x
  34. ']','\\', '#', ';','\'', '`', ',', '.', '/', 128, ' ', ' ', ' ', ' ', ' ', ' ', // 3x
  35. ' ', ' ', ' ', ' ', ' ', ' ', 'P', 'S', ' ', ' ', ' ', ' ', 16, ' ', ' ', ' ', // 4x
  36. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // 5x
  37. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // 6x
  38. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // 7x
  39. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // 8x
  40. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // 9x
  41. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // Ax
  42. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // Bx
  43. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // Cx
  44. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // Dx
  45. 'C', 'S', 'A', 'C', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', // Ex
  46. ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ' // Fx
  47. };
  48. // clang-format on
  49. void add_keylog(uint16_t keycode) {
  50. if ((keycode >= QK_MOD_TAP && keycode <= QK_MOD_TAP_MAX) || (keycode >= QK_LAYER_TAP && keycode <= QK_LAYER_TAP_MAX) || (keycode >= QK_MODS && keycode <= QK_MODS_MAX)) {
  51. keycode = keycode & 0xFF;
  52. } else if (keycode > 0xFF) {
  53. keycode = 0;
  54. }
  55. for (uint8_t i = (KEYLOGGER_LENGTH - 1); i > 0; --i) {
  56. keylog_str[i] = keylog_str[i - 1];
  57. }
  58. if (keycode < (sizeof(code_to_name) / sizeof(char))) {
  59. keylog_str[0] = pgm_read_byte(&code_to_name[keycode]);
  60. }
  61. log_timer = timer_read();
  62. }
  63. bool process_record_user_oled(uint16_t keycode, keyrecord_t *record) {
  64. if (record->event.pressed) {
  65. #ifdef OLED_DRIVER_ENABLE
  66. oled_timer = timer_read32();
  67. add_keylog(keycode);
  68. #endif
  69. }
  70. return true;
  71. }
  72. void update_log(void) {
  73. if (timer_elapsed(log_timer) > 750) {
  74. // add_keylog(0);
  75. }
  76. }
  77. void render_keylogger_status(void) {
  78. oled_write_P(PSTR(OLED_RENDER_KEYLOGGER), false);
  79. oled_write(keylog_str, false);
  80. }
  81. void render_default_layer_state(void) {
  82. oled_write_P(PSTR(OLED_RENDER_LAYOUT_NAME), false);
  83. switch (get_highest_layer(default_layer_state)) {
  84. case _QWERTY:
  85. oled_write_P(PSTR(OLED_RENDER_LAYOUT_QWERTY), false);
  86. break;
  87. case _COLEMAK:
  88. oled_write_P(PSTR(OLED_RENDER_LAYOUT_COLEMAK), false);
  89. break;
  90. case _DVORAK:
  91. oled_write_P(PSTR(OLED_RENDER_LAYOUT_DVORAK), false);
  92. break;
  93. case _WORKMAN:
  94. oled_write_P(PSTR(OLED_RENDER_LAYOUT_WORKMAN), false);
  95. break;
  96. case _NORMAN:
  97. oled_write_P(PSTR(OLED_RENDER_LAYOUT_NORMAN), false);
  98. break;
  99. case _MALTRON:
  100. oled_write_P(PSTR(OLED_RENDER_LAYOUT_MALTRON), false);
  101. break;
  102. case _EUCALYN:
  103. oled_write_P(PSTR(OLED_RENDER_LAYOUT_EUCALYN), false);
  104. break;
  105. case _CARPLAX:
  106. oled_write_P(PSTR(OLED_RENDER_LAYOUT_CARPLAX), false);
  107. break;
  108. }
  109. #ifdef OLED_DISPLAY_128X64
  110. oled_advance_page(true);
  111. #endif
  112. }
  113. void render_layer_state(void) {
  114. oled_write_P(PSTR(OLED_RENDER_LAYER_NAME), false);
  115. #ifdef OLED_DISPLAY_128X64
  116. oled_write_P(PSTR(" "), false);
  117. #endif
  118. oled_write_P(PSTR(OLED_RENDER_LAYER_LOWER), layer_state_is(_LOWER));
  119. #ifdef OLED_DISPLAY_128X64
  120. oled_write_P(PSTR(" "), false);
  121. #endif
  122. oled_write_P(PSTR(OLED_RENDER_LAYER_RAISE), layer_state_is(_RAISE));
  123. #if _MODS
  124. # ifdef OLED_DISPLAY_128X64
  125. oled_write_P(PSTR(" "), false);
  126. # endif
  127. oled_write_P(PSTR(OLED_RENDER_LAYER_MODS), layer_state_is(_MODS));
  128. #endif
  129. #ifdef OLED_DISPLAY_128X64
  130. oled_advance_page(true);
  131. #endif
  132. }
  133. void render_keylock_status(uint8_t led_usb_state) {
  134. oled_write_P(PSTR(OLED_RENDER_LOCK_NAME), false);
  135. oled_write_P(PSTR(" "), false);
  136. oled_write_P(PSTR(OLED_RENDER_LOCK_NUML), led_usb_state & (1 << USB_LED_NUM_LOCK));
  137. #ifdef OLED_DISPLAY_128X64
  138. oled_write_P(PSTR(" "), false);
  139. #endif
  140. oled_write_P(PSTR(OLED_RENDER_LOCK_CAPS), led_usb_state & (1 << USB_LED_CAPS_LOCK));
  141. #ifdef OLED_DISPLAY_128X64
  142. oled_write_P(PSTR(" "), false);
  143. #endif
  144. oled_write_P(PSTR(OLED_RENDER_LOCK_SCLK), led_usb_state & (1 << USB_LED_SCROLL_LOCK));
  145. #ifndef OLED_DISPLAY_128X64
  146. oled_advance_page(true);
  147. #endif
  148. }
  149. void render_mod_status(uint8_t modifiers) {
  150. oled_write_P(PSTR(OLED_RENDER_MODS_NAME), false);
  151. oled_write_P(PSTR(OLED_RENDER_MODS_SFT), (modifiers & MOD_MASK_SHIFT));
  152. #ifdef OLED_DISPLAY_128X64
  153. oled_write_P(PSTR(" "), false);
  154. #endif
  155. oled_write_P(PSTR(OLED_RENDER_MODS_CTL), (modifiers & MOD_MASK_CTRL));
  156. #ifdef OLED_DISPLAY_128X64
  157. oled_write_P(PSTR(" "), false);
  158. #endif
  159. oled_write_P(PSTR(OLED_RENDER_MODS_ALT), (modifiers & MOD_MASK_ALT));
  160. #ifdef OLED_DISPLAY_128X64
  161. oled_write_P(PSTR(" "), false);
  162. #endif
  163. oled_write_P(PSTR(OLED_RENDER_MODS_GUI), (modifiers & MOD_MASK_GUI));
  164. }
  165. void render_bootmagic_status(void) {
  166. /* Show Ctrl-Gui Swap options */
  167. static const char PROGMEM logo[][2][3] = {
  168. {{0x97, 0x98, 0}, {0xb7, 0xb8, 0}},
  169. {{0x95, 0x96, 0}, {0xb5, 0xb6, 0}},
  170. };
  171. bool is_bootmagic_on;
  172. #ifdef OLED_DISPLAY_128X64
  173. is_bootmagic_on = !keymap_config.swap_lctl_lgui;
  174. #else
  175. is_bootmagic_on = keymap_config.swap_lctl_lgui;
  176. #endif
  177. oled_write_P(PSTR(OLED_RENDER_BOOTMAGIC_NAME), false);
  178. #ifdef OLED_DISPLAY_128X64
  179. if (keymap_config.swap_lctl_lgui)
  180. #else
  181. oled_write_P(PSTR(" "), false);
  182. #endif
  183. {
  184. oled_write_P(logo[1][0], is_bootmagic_on);
  185. #ifdef OLED_DISPLAY_128X64
  186. } else {
  187. #endif
  188. oled_write_P(logo[0][0], !is_bootmagic_on);
  189. }
  190. oled_write_P(PSTR(" "), false);
  191. #ifdef OLED_DISPLAY_128X64
  192. oled_write_P(PSTR(OLED_RENDER_BOOTMAGIC_NKRO), keymap_config.nkro);
  193. oled_write_P(PSTR(" "), false);
  194. oled_write_ln_P(PSTR(OLED_RENDER_BOOTMAGIC_NOGUI), !keymap_config.no_gui);
  195. oled_write_P(PSTR("Magic "), false);
  196. if (keymap_config.swap_lctl_lgui)
  197. #endif
  198. {
  199. oled_write_P(logo[1][1], is_bootmagic_on);
  200. #ifdef OLED_DISPLAY_128X64
  201. } else {
  202. #endif
  203. oled_write_P(logo[0][1], !is_bootmagic_on);
  204. }
  205. oled_write_P(PSTR(" "), false);
  206. #ifdef OLED_DISPLAY_128X64
  207. oled_write_P(PSTR(OLED_RENDER_BOOTMAGIC_GRV), keymap_config.swap_grave_esc);
  208. oled_write_P(PSTR(" "), false);
  209. oled_write_P(PSTR(OLED_RENDER_BOOTMAGIC_CAPS), keymap_config.swap_control_capslock);
  210. oled_advance_page(true);
  211. #else
  212. oled_write_P(PSTR(OLED_RENDER_BOOTMAGIC_NKRO), keymap_config.nkro);
  213. #endif
  214. }
  215. void render_user_status(void) {
  216. oled_write_P(PSTR(OLED_RENDER_USER_NAME), false);
  217. oled_write_P(PSTR(" "), false);
  218. oled_write_P(PSTR(OLED_RENDER_USER_ANIM), userspace_config.rgb_matrix_idle_anim);
  219. oled_write_P(PSTR(" "), false);
  220. oled_write_P(PSTR(OLED_RENDER_USER_LAYR), userspace_config.rgb_layer_change);
  221. oled_write_P(PSTR(" "), false);
  222. oled_write_P(PSTR(OLED_RENDER_USER_NUKE), userspace_config.nuke_switch);
  223. #ifdef OLED_DISPLAY_128X64
  224. oled_advance_page(true);
  225. #endif
  226. }
  227. __attribute__((weak)) void oled_driver_render_logo(void) {
  228. // clang-format off
  229. static const char PROGMEM qmk_logo[] = {
  230. 0x80,0x81,0x82,0x83,0x84,0x85,0x86,0x87,0x88,0x89,0x8a,0x8b,0x8c,0x8d,0x8e,0x8f,0x90,0x91,0x92,0x93,0x94,
  231. 0xa0,0xa1,0xa2,0xa3,0xa4,0xa5,0xa6,0xa7,0xa8,0xa9,0xaa,0xab,0xac,0xad,0xae,0xaf,0xb0,0xb1,0xb2,0xb3,0xb4,
  232. 0xc0,0xc1,0xc2,0xc3,0xc4,0xc5,0xc6,0xc7,0xc8,0xc9,0xca,0xcb,0xcc,0xcd,0xce,0xcf,0xd0,0xd1,0xd2,0xd3,0xd4,0};
  233. // clang-format on
  234. oled_write_P(qmk_logo, false);
  235. }
  236. void render_status_secondary(void) {
  237. #if !defined(SPLIT_TRANSPORT_MIRROR) || defined(OLED_DISPLAY_128X64)
  238. oled_driver_render_logo();
  239. #endif
  240. #ifdef SPLIT_TRANSPORT_MIRROR
  241. /* Show Keyboard Layout */
  242. render_default_layer_state();
  243. render_layer_state();
  244. render_mod_status(get_mods() | get_oneshot_mods());
  245. render_keylogger_status();
  246. #endif
  247. }
  248. // clang-format on
  249. void render_status_main(void) {
  250. /* Show Keyboard Layout */
  251. render_default_layer_state();
  252. render_keylock_status(host_keyboard_leds());
  253. render_bootmagic_status();
  254. render_user_status();
  255. render_keylogger_status();
  256. }
  257. void oled_task_user(void) {
  258. if (timer_elapsed32(oled_timer) > 30000) {
  259. oled_off();
  260. return;
  261. }
  262. #ifndef SPLIT_KEYBOARD
  263. else {
  264. oled_on();
  265. }
  266. #endif
  267. update_log();
  268. if (is_master) {
  269. render_status_main(); // Renders the current keyboard state (layer, lock, caps, scroll, etc)
  270. } else {
  271. render_status_secondary();
  272. }
  273. }