/** @file oled.h * @brief mcrown oled service implementation. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * * @author Mario Corona (mariocc@comunidad.unam.mx) 2021 * */ #include QMK_KEYBOARD_H #include <stdio.h> #include <string.h> #include "mcrown.h" #include "oled.h" #define ASCII_TABLE_LENGTH (0x80) #define KEYLOG_STRING_STARTUP (KEYLOG_EOL_LEN+1) #define ALT_CODE (0x7E) #define SPECIAL_KEYS_SHIFT(kc) (0x18+(kc)) static void render_keylogger_status(void); static char keylog_str[KEYLOG_EOL_LEN] = {' '}; static uint16_t log_timer = 0; static uint8_t current_cursor_pos=0; static uint32_t cursor_oled_timer = 0; static uint32_t standby_oled_timer = 0; static char last_c=' '; /* Provides the ASCII value or the address of the character selected of the OLED font specified in glcfont.c */ static const char ascii_t[ASCII_TABLE_LENGTH] = { /* 0 1 2 3 4 5 6 7 8 9 A B C D E F */ /* | | | | | | | | | | | | | | | | */ 0x0F, 0x1A, 0x1B, 0x19, 0x18, 0x0E, ' ', ' ', 0x11, 0x1C, 0x97, ' ', ' ', ' ', ' ', ' ', /* 0 */ /* | | | | | | | | | | | | | | | | */ ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', 0x1D, ' ', ' ', ' ', ' ', /* 1 */ /* | | | | | | | | | | | | | | | | */ 0x16, '!', '"', '#', '$', '%', '&', '\'', '(', ')', '*', '+', ',', '-', '.', '/', /* 2 */ /* | | | | | | | | | | | | | | | | */ '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', ':', ';', '<', '=', '>', '?', /* 3 */ /* | | | | | | | | | | | | | | | | */ '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', /* 4 */ /* | | | | | | | | | | | | | | | | */ 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '[', '\\', ']', '^', '_', /* 5 */ /* | | | | | | | | | | | | | | | | */ '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', /* 6 */ /* | | | | | | | | | | | | | | | | */ 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '{', '|', '}', '~', 0x7F, /* 7 */ }; /* This table is to remap and get the corresponding ASCII value based on the KEYCODE (taken as the index of the array) of quatum_keycodes.h module */ static const unsigned char code_to_ascii[ASCII_TABLE_LENGTH] = { /* 0 1 2 3 4 5 6 7 8 9 A B C D E F */ /* | | | | | | | | | | | | | | | | */ 0x00, 0x00, 0x00, 0x00, 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', /* 0 */ /* | | | | | | | | | | | | | | | | */ 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '1', '2', /* 1 */ /* | | | | | | | | | | | | | | | | */ '3', '4', '5', '6', '7', '8', '9', '0', 0x0A, 0x1B, 0x08, 0x09, ' ', '-', '=', '[', /* 2 */ /* | | | | | | | | | | | | | | | | */ ']', '\\', 0x00, ';', '\'', '`', ',', '.', '/', 0x00, 0x00, 0x00, 0x00, 0x00, '!', '@', /* 3 */ /* | | | | | | | | | | | | | | | | */ '#', '$', '%', '^', '&', '*', '(', ')', 0x00, 0x00, 0x00, 0x00, 0x00, '_', '+', '{', /* 4 */ /* | | | | | | | | | | | | | | | | */ '}', '|', 0x00, 0x00, 0x00, '~', 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 5 */ /* | | | | | | | | | | | | | | | | */ 0x00, 0x00, 0x00, 0x00, 0x7F, 0x00, 0x00, 0x01, 0x02, 0x03, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, /* 6 */ /* | | | | | | | | | | | | | | | | */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, 0x00, /* 7 */ }; /** @brief maps the keycode to get the ascii value. * * If any argument is invalid, the function has no effect. * * @param keycode value of the pressed key. * @return ascii value of the pressed key or a special value for non-ascii keys. */ inline static char get_ascii(int16_t keycode){ uint8_t ascii_idx=0x00; if(keycode<KC_F1){ ascii_idx=code_to_ascii[(uint8_t)keycode]; }else if(keycode<KC_KP_ENTER){ ascii_idx=code_to_ascii[SPECIAL_KEYS_SHIFT(keycode)]; }else if(KC_LANG1==keycode){ ascii_idx=code_to_ascii[ALT_CODE]; }else if( QK_LSFT==(QK_LSFT&keycode) ){ ascii_idx=code_to_ascii[RM_LSFT(keycode)]; } return ascii_t[ascii_idx]; } /** @brief detect retuns the rotation of the display based on the keyboard side. * * If any argument is invalid, the function has no effect. * * @param oled_rotation_t rotation * @return rotation of the display. */ oled_rotation_t oled_init_user(oled_rotation_t rotation){ oled_rotation_t oled_rot=OLED_ROTATION_180; if(true==is_keyboard_master()){ #ifdef OLED_VERTICAL oled_rot=OLED_ROTATION_270; #else oled_rot=OLED_ROTATION_0; #endif } return oled_rot; } /** @brief renders the keylog string and display it. This function also toggles the cursor. * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ static void render_keylogger_status(void){ static bool cursor_f=true; if(timer_elapsed32(cursor_oled_timer) > 300){ cursor_oled_timer = timer_read32(); cursor_f=!cursor_f; } oled_write_P(PSTR("\n>:"), false); if(current_cursor_pos>(KEYLOG_LEN-1)){ current_cursor_pos=0; memset(keylog_str, ' ', sizeof(char)*KEYLOG_EOL_LEN); /* Here the EOL is to clear with white spaces all the keylog area */ keylog_str[KEYLOG_EOL_LEN-1] = '\0'; oled_write(keylog_str, false); /* Reset EOL to the begining of the keylog string */ keylog_str[0] = '\0'; } oled_write(keylog_str, false); oled_write_char(last_c, cursor_f); } /** @brief displays the current active layout. * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ void render_layout_state(void){ CUSTOM_LAYERS_T current_layer; current_layer=(CUSTOM_LAYERS_T)get_highest_layer(layer_state); #ifdef OLED_VERTICAL oled_write_P(PSTR("Lyt:\n"), false); #else oled_write_P(PSTR("Layout: "), false); #endif switch (current_layer){ #ifdef OLED_VERTICAL case _COLEMAK: oled_write_P(PSTR("Clmak\n"), false); break; case _DVORAK: oled_write_P(PSTR("Dvak\n"), false); break; case _QWERTY: oled_write_P(PSTR("Qwty\n"), false); break; default: oled_write_P(PSTR("Undf\n"), false); break; #else case _COLEMAK: oled_write_P(PSTR("Colemak\n"), false); break; case _DVORAK: oled_write_P(PSTR("Dvorak\n"), false); break; case _QWERTY: oled_write_P(PSTR("Qwerty\n"), false); break; default: oled_write_P(PSTR("Undefined\n"), false); break; #endif } } /** @brief displays the current active layer. * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ void render_layer_state(void){ CUSTOM_LAYERS_T current_layer; current_layer=(CUSTOM_LAYERS_T)get_highest_layer(layer_state); #ifdef OLED_VERTICAL oled_write_P(PSTR("Lyr:\n"), false); #else oled_write_P(PSTR("Layer:"), false); #endif switch(current_layer){ #ifdef OLED_VERTICAL case _LOWER: oled_write_P(PSTR("Lwr\n"), true); break; case _RAISE: oled_write_P(PSTR("Ris\n"), true); break; case _ADJUST: oled_write_P(PSTR("Adj\n"), true); break; case _NUMPAD: oled_write_P(PSTR("Num\n"), true); break; default: oled_write_P(PSTR("Def\n"), false); break; #else case _LOWER: oled_write_P(PSTR(" Lower "), true); break; case _RAISE: oled_write_P(PSTR(" Raise "), true); break; case _ADJUST: oled_write_P(PSTR(" Adjust "), true); break; case _NUMPAD: oled_write_P(PSTR(" Numpad "), true); break; default: oled_write_P(PSTR(" Default "), false); break; #endif } } /** @brief displays the current status of the main display/ * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ void render_status(void){ render_layout_state(); oled_write_P(PSTR("\n"), false); render_layer_state(); render_keylogger_status(); } /** @brief renders the logo to be displayed. * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ static void render_logo(void){ static const char PROGMEM qmk_logo[] = { 0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F, 0x90, 0x91, 0x92, 0x93, 0x94, 0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF, 0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF, 0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0x00}; oled_write_P(qmk_logo, false); } /** @brief executes the actions for both displays. * * If any argument is invalid, the function has no effect. * * @param void. * @return void. */ void oled_task_user(void){ if (timer_elapsed32(standby_oled_timer) > 15000){ oled_off(); }else{ oled_on(); if(true==is_keyboard_master()){ render_status(); }else{ render_logo(); oled_write_P(PSTR("\n"), false); oled_scroll_left(); } } } /** @brief process the current key and add it to the keylog string. * * If any argument is invalid, the function has no effect. * * @param keycode pressed key. * @return void. */ extern void add_keylog(uint16_t keycode){ if ((keycode >= QK_MOD_TAP && keycode <= QK_MOD_TAP_MAX) || (keycode >= QK_LAYER_TAP && keycode <= QK_LAYER_TAP_MAX)){ keycode&=0x00FF; } if(current_cursor_pos>(KEYLOG_LEN-1)||(current_cursor_pos>KEYLOG_STRING_STARTUP)){ current_cursor_pos=0; last_c=get_ascii(keycode); current_cursor_pos++; }else{ if(keycode <= KC_TILD){ keylog_str[current_cursor_pos]=last_c; last_c=get_ascii(keycode); current_cursor_pos++; } keylog_str[current_cursor_pos] = '\0'; } log_timer = timer_read(); standby_oled_timer = timer_read32(); }