summaryrefslogtreecommitdiffstats
path: root/keyboards/nullbitsco
diff options
context:
space:
mode:
authorlokher <lokher@gmail.com>2022-12-06 17:04:10 +0800
committerlokher <lokher@gmail.com>2022-12-06 17:04:10 +0800
commit27fc28fd2ff52e079a5bc58d6aaea4c752420615 (patch)
tree7ac943fb1ba4f430a7220efd18f66f6a77205c30 /keyboards/nullbitsco
parente736133392fe6427cfb995da0787337189828272 (diff)
parent2709b6ed616f8012ff4cfd3ee69a822a8d188351 (diff)
Merge upstream master
Diffstat (limited to 'keyboards/nullbitsco')
-rw-r--r--keyboards/nullbitsco/nibble/config.h62
-rw-r--r--keyboards/nullbitsco/nibble/keymaps/oled_bongocat/keymap.c263
-rw-r--r--keyboards/nullbitsco/nibble/keymaps/snailmap/keymap.c543
-rw-r--r--keyboards/nullbitsco/nibble/keymaps/via/keymap.c175
-rw-r--r--keyboards/nullbitsco/nibble/nibble.c21
-rw-r--r--keyboards/nullbitsco/tidbit/tidbit.c153
6 files changed, 1217 insertions, 0 deletions
diff --git a/keyboards/nullbitsco/nibble/config.h b/keyboards/nullbitsco/nibble/config.h
new file mode 100644
index 0000000000..16b069dae0
--- /dev/null
+++ b/keyboards/nullbitsco/nibble/config.h
@@ -0,0 +1,62 @@
+/* Copyright 2021 Jay Greco
+ *
+ * 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/>.
+ */
+#pragma once
+
+#include "config_common.h"
+
+/* Used to set host for remote KB if VUSB detect doesn't work. */
+// #define KEYBOARD_HOST // Force host mode
+// #define KEYBOARD_REMOTE // Force remote mode
+
+// Workaround for freezing after MacOS sleep
+#define NO_USB_STARTUP_CHECK
+
+/* key matrix size */
+#define MATRIX_ROWS 5
+#define MATRIX_COLS 16
+#define MATRIX_MUX_COLS 4
+
+/* Set 0 if debouncing isn't needed */
+#define DEBOUNCE 10
+
+/*
+ * Keyboard Matrix Assignments
+ * The nibble uses a demultiplexer for the cols.
+ * to free up more IOs for awesomeness!
+ * See matrix.c for more details.
+*/
+#define MATRIX_ROW_PINS { B1, B3, B2, B6, D4 }
+#define MATRIX_COL_MUX_PINS { F4, F5, F6, F7 }
+#define MATRIX_COL_PINS { }
+
+/* Optional SMT LED pins */
+#define RGB_DI_PIN E6
+#define RGBLED_NUM 10
+#define RGBLIGHT_EFFECT_BREATHING
+#define RGBLIGHT_EFFECT_RAINBOW_MOOD
+#define RGBLIGHT_EFFECT_RAINBOW_SWIRL
+#define RGBLIGHT_EFFECT_SNAKE
+#define RGBLIGHT_EFFECT_KNIGHT
+#define RGBLIGHT_EFFECT_CHRISTMAS
+#define RGBLIGHT_EFFECT_STATIC_GRADIENT
+#define RGBLIGHT_EFFECT_RGB_TEST
+#define RGBLIGHT_EFFECT_ALTERNATING
+#define RGBLIGHT_EFFECT_TWINKLE
+#define RGBLIGHT_SLEEP
+
+/* Optional encoder pins */
+#define ENCODERS_PAD_A { B5 }
+#define ENCODERS_PAD_B { B4 }
diff --git a/keyboards/nullbitsco/nibble/keymaps/oled_bongocat/keymap.c b/keyboards/nullbitsco/nibble/keymaps/oled_bongocat/keymap.c
new file mode 100644
index 0000000000..631f28171a
--- /dev/null
+++ b/keyboards/nullbitsco/nibble/keymaps/oled_bongocat/keymap.c
@@ -0,0 +1,263 @@
+/* Copyright 2021 Jonathan Law, Jay Greco
+ *
+ * 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/>.
+ *
+ * Original: j-inc's kyria keymap
+ */
+#include QMK_KEYBOARD_H
+#include "animation_frames.h"
+
+enum layer_names {
+ _BASE,
+ _VIA1,
+ _VIA2,
+ _VIA3
+};
+
+#define KC_DISC_MUTE KC_F23
+#define KC_DISC_DEAF KC_F24
+
+enum custom_keycodes {
+ PROG = USER00,
+ DISC_MUTE,
+ DISC_DEAF,
+ SUPER_ALT_TAB,
+ _NUM_CUST_KCS,
+};
+
+// Macro variables
+bool is_alt_tab_active = false;
+uint16_t alt_tab_timer = 0;
+bool muted = false;
+bool deafened = false;
+
+const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
+ [_BASE] = LAYOUT_all(
+ QK_GESC, 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_HOME,
+ KC_F13, 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_F14, 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_PGUP,
+ KC_F15, KC_LSFT, KC_NUBS, 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_PGDN,
+ KC_F16, KC_LCTL, KC_LGUI, KC_LALT, KC_SPC, MO(_VIA1), KC_RALT, KC_RCTL, KC_LEFT, KC_DOWN, KC_RGHT
+ ),
+
+ [_VIA1] = LAYOUT_all(
+ QK_BOOT, 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_END,
+ RGB_TOG, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, KC_MPRV, KC_MPLY, KC_MNXT
+ ),
+
+ [_VIA2] = LAYOUT_all(
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
+ ),
+
+ [_VIA3] = LAYOUT_all(
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
+ ),
+
+};
+
+bool encoder_update_user(uint8_t index, bool clockwise) {
+ if (clockwise) {
+ tap_code(KC_VOLU);
+ } else {
+ tap_code(KC_VOLD);
+ }
+ return true;
+}
+
+#ifdef OLED_ENABLE
+#define IDLE_FRAME_DURATION 200 // Idle animation iteration rate in ms
+
+oled_rotation_t oled_init_user(oled_rotation_t rotation) { return OLED_ROTATION_270; }
+
+uint32_t anim_timer = 0;
+uint32_t anim_sleep = 0;
+uint8_t current_idle_frame = 0;
+
+bool tap_anim = false;
+bool tap_anim_toggle = false;
+
+
+// Decompress and write a precompressed bitmap frame to the OLED.
+// Documentation and python compression script available at:
+// https://github.com/nullbitsco/squeez-o
+#ifdef USE_OLED_BITMAP_COMPRESSION
+static void oled_write_compressed_P(const char* input_block_map, const char* input_block_list) {
+ uint16_t block_index = 0;
+ for (uint16_t i=0; i<NUM_OLED_BYTES; i++) {
+ uint8_t bit = i%8;
+ uint8_t map_index = i/8;
+ uint8_t _block_map = (uint8_t)pgm_read_byte_near(input_block_map + map_index);
+ uint8_t nonzero_byte = (_block_map & (1 << bit));
+ if (nonzero_byte) {
+ const char data = (const char)pgm_read_byte_near(input_block_list + block_index++);
+ oled_write_raw_byte(data, i);
+ } else {
+ const char data = (const char)0x00;
+ oled_write_raw_byte(data, i);
+ }
+ }
+}
+#endif
+
+static void render_anim(void) {
+ // Idle animation
+ void animation_phase(void) {
+ if (!tap_anim) {
+ current_idle_frame = (current_idle_frame + 1) % NUM_IDLE_FRAMES;
+ uint8_t idx = abs((NUM_IDLE_FRAMES - 1) - current_idle_frame);
+ #ifdef USE_OLED_BITMAP_COMPRESSION
+ oled_write_compressed_P(idle_block_map[idx], idle_frames[idx]);
+ #else
+ oled_write_raw_P(idle_frames[idx], NUM_OLED_BYTES);
+ #endif
+ }
+ }
+
+ // Idle behaviour
+ if (get_current_wpm() != 000) { // prevent sleep
+ oled_on();
+ if (timer_elapsed32(anim_timer) > IDLE_FRAME_DURATION) {
+ anim_timer = timer_read32();
+ animation_phase();
+ }
+ anim_sleep = timer_read32();
+ } else { // Turn off screen when timer threshold elapsed or reset time since last input
+ if (timer_elapsed32(anim_sleep) > OLED_TIMEOUT) {
+ oled_off();
+ } else {
+ if (timer_elapsed32(anim_timer) > IDLE_FRAME_DURATION) {
+ anim_timer = timer_read32();
+ animation_phase();
+ }
+ }
+ }
+}
+
+bool oled_task_user(void) {
+ render_anim();
+ oled_set_cursor(0, 14);
+
+ uint8_t n = get_current_wpm();
+ char wpm_counter[6];
+ wpm_counter[5] = '\0';
+ wpm_counter[4] = '0' + n % 10;
+ wpm_counter[3] = '0' + (n /= 10) % 10;
+ wpm_counter[2] = '0' + n / 10 ;
+ wpm_counter[1] = '0';
+ wpm_counter[0] = '>';
+ oled_write_ln(wpm_counter, false);
+
+ return false;
+}
+#endif
+
+// Animate tap
+bool process_record_user(uint16_t keycode, keyrecord_t *record) {
+ #ifdef OLED_ENABLE
+ // Check if non-mod
+ if ((keycode >= KC_A && keycode <= KC_0) || (keycode >= KC_TAB && keycode <= KC_SLASH)) {
+ if (record->event.pressed) {
+ // Display tap frames
+ tap_anim_toggle = !tap_anim_toggle;
+ #ifdef USE_OLED_BITMAP_COMPRESSION
+ oled_write_compressed_P(tap_block_map[tap_anim_toggle], tap_frames[tap_anim_toggle]);
+ #else
+ oled_write_raw_P(tap_frames[tap_anim_toggle], NUM_OLED_BYTES);
+ #endif
+ }
+ }
+ #endif
+
+ switch(keycode) {
+ case PROG:
+ if (record->event.pressed) {
+ rgblight_disable_noeeprom();
+ #ifdef OLED_ENABLE
+ oled_off();
+ #endif
+ bootloader_jump();
+ }
+ break;
+
+ case DISC_MUTE:
+ if (record->event.pressed) {
+ tap_code(KC_DISC_MUTE);
+ if (!rgblight_is_enabled()) break;
+
+ if (muted) {
+ rgblight_enable_noeeprom();
+ } else {
+ rgblight_timer_disable();
+ uint8_t val = rgblight_get_val();
+ rgblight_sethsv_range(255, 255, val, 0, 1);
+ }
+ muted = !muted;
+ }
+ break;
+
+ case DISC_DEAF:
+ if (record->event.pressed) {
+ tap_code(KC_DISC_DEAF);
+ if (!rgblight_is_enabled()) break;
+
+ if (deafened) {
+ rgblight_enable_noeeprom();
+ } else {
+ rgblight_timer_disable();
+ uint8_t val = rgblight_get_val();
+ rgblight_sethsv_range(255, 255, val, 0, RGBLED_NUM-1);
+ }
+ deafened = !deafened;
+ }
+ break;
+
+ case SUPER_ALT_TAB:
+ if (record->event.pressed) {
+ if (!is_alt_tab_active) {
+ is_alt_tab_active = true;
+ register_code(KC_LALT);
+ }
+ alt_tab_timer = timer_read();
+ register_code(KC_TAB);
+ } else {
+ unregister_code(KC_TAB);
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ return true;
+}
+
+void matrix_scan_user(void) {
+ if (is_alt_tab_active) {
+ if (timer_elapsed(alt_tab_timer) > 1000) {
+ unregister_code(KC_LALT);
+ is_alt_tab_active = false;
+ }
+ }
+}
diff --git a/keyboards/nullbitsco/nibble/keymaps/snailmap/keymap.c b/keyboards/nullbitsco/nibble/keymaps/snailmap/keymap.c
new file mode 100644
index 0000000000..a53335003f
--- /dev/null
+++ b/keyboards/nullbitsco/nibble/keymaps/snailmap/keymap.c
@@ -0,0 +1,543 @@
+/* Copyright 2021 dogspace <https://github.com/dogspace>
+ *
+ * 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/>.
+ */
+
+#include QMK_KEYBOARD_H
+
+enum custom_keycodes {
+ KC_CUST = SAFE_RANGE,
+};
+
+enum layer_names {
+ _MA,
+ _L1,
+ _L2,
+ _L3
+};
+
+// NOTE: Default keymap layers were designed for ANSI split-space layout http://www.keyboard-layout-editor.com/#/gists/f28bd5ff4e62f69e89896df3a59671c6
+const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {
+ [_MA] = LAYOUT_ansi(
+ KC_ESC, 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_DEL,
+ KC_MUTE, 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, LCTL(KC_F),
+ KC_CAPS, MO(_L2), 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_CALC,
+ TG(_L2), 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_WHOM,
+ MO(_L3), KC_LCTL, KC_LGUI, KC_LALT, KC_SPC, MO(_L1), LCTL(KC_C), LCTL(KC_V), KC_LEFT, KC_DOWN, KC_RGHT
+ ),
+ [_L1] = LAYOUT_ansi(
+ KC_GRAVE, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ RGB_TOG, _______, _______, _______, _______, _______, _______, _______, KC_PGUP, KC_UP, _______, _______, _______, _______, _______, _______,
+ _______, LCTL(KC_Z), KC_LCTL, KC_LSFT, _______, _______, _______, KC_HOME, KC_LEFT, KC_DOWN, KC_RIGHT, KC_END, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, KC_PGDN, _______, _______, LCTL(KC_SLSH), _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, LCTL(KC_X), _______, _______, _______, _______
+ ),
+ [_L2] = LAYOUT_ansi(
+ KC_GRAVE, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ RGB_TOG, _______, _______, _______, _______, _______, _______, KC_PAST, KC_7, KC_8, KC_9, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, KC_PPLS, KC_4, KC_5, KC_6, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, KC_PMNS, KC_1, KC_2, KC_3, _______, _______, _______, _______,
+ _______, _______, _______, _______, KC_0, KC_PSLS, _______, _______, _______, _______, _______
+ ),
+ [_L3] = LAYOUT_ansi(
+ _______, KC_F1, KC_F2, KC_F3, KC_F4, KC_F5, KC_F6, KC_F7, KC_F8, KC_F9, KC_F10, KC_F11, KC_F12, _______, _______,
+ RGB_TOG, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ KC_SYRQ, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______,
+ _______, _______, _______, _______, _______, _______, _______, _______, _______, _______, _______
+ )
+};
+
+#ifdef OLED_ENABLE
+/*=========================================== OLED CONFIGURATION ===========================================*/
+bool oled_horizontal = true; // OLED rotation (true = horizontal, false = vertical)
+bool ansi_layout = true; // ANSI or ISO layout (true = ANSI, false = ISO)
+bool split_space = true; // Split spacebar (true = split spacebar, false = 6.25u or 7u spacebar)
+bool three_mods_left = true; // Left mods layout (true = 3x 1.25u keys, false = 2x 1.5u keys)
+bool three_mods_right = false; // Right mods layout (true = 3x 1u keys, false = 2x 1.5u keys)
+bool graph_direction = true; // Graph movement (true = right to left, false = left to right)
+float graph_top_wpm = 100.0; // Minimum WPM required to reach the top of the graph
+int graph_refresh = 1000; // In milliseconds, determines the graph-line frequency
+int icon_med_wpm = 50; // WPM required to display the medium snail
+int icon_fast_wpm = 72; // WPM required to display the fast snail
+// Layer names: Should be exactly 5 characters in length if vertical display, or 6 characters if horizontal
+#define MA_LAYER_NAME "QWERTY" // Layer _MA name
+#define L1_LAYER_NAME "ARROWS" // Layer _L1 name
+#define L2_LAYER_NAME "NUMPAD" // Layer _L2 name
+#define L3_LAYER_NAME "FUNCTN" // Layer _L3 name
+/*================================================================================================================*/
+bool first_loop = true;
+int timer = 0;
+int wpm_limit = 20;
+int max_wpm = -1;
+int wpm_icon = -1;
+int graph_lines[64];
+
+// Set OLED rotation
+oled_rotation_t oled_init_user(oled_rotation_t rotation) {
+ if (oled_horizontal) {
+ return OLED_ROTATION_180;
+ } else {
+ return OLED_ROTATION_90;
+ }
+}
+
+// Toggles pixel on/off, converts horizontal coordinates to vertical equivalent if necessary
+static void write_pixel(int x, int y, bool onoff) {
+ if (oled_horizontal) {
+ oled_write_pixel(x, y, onoff);
+ } else {
+ oled_write_pixel(y, 127 - x, onoff);
+ }
+}
+
+// Draw static background image to OLED (keyboard with no bottom row)
+static void render_background(void) {
+ if (oled_horizontal) {
+ static const char PROGMEM oled_keymap_horizontal[] = {
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x00, 0x00,
+ 0x84, 0x80, 0x80, 0x80, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04,
+ 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00,
+ 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00,
+ 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00, 0x80, 0x04, 0x00, 0x00, 0x00,
+ 0x80, 0x04, 0x04, 0x04, 0x04, 0x84, 0x84, 0x84, 0x84, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00,
+ 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10,
+ 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00,
+ 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00,
+ 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00,
+ 0x00, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00,
+ 0x02, 0x02, 0x02, 0x02, 0x02, 0x02, 0x02, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00,
+ 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00,
+ 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02,
+ 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x02,
+ 0x02, 0x02, 0x02, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
+ };
+ oled_write_raw_P(oled_keymap_horizontal, sizeof(oled_keymap_horizontal));
+ } else {
+ static const char PROGMEM oled_keymap_vertical[] = {
+ 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00,
+ 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x7f, 0x00, 0x00, 0x00, 0x00, 0x87, 0x00, 0x00, 0x00, 0x00, 0x7f, 0x00, 0x00, 0x00,
+ 0x00, 0xf0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00,
+ 0x00, 0x43, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x21, 0x00, 0x00, 0x00, 0x00, 0x42, 0x00, 0x00, 0x00, 0x00, 0x21, 0x00, 0x00, 0x00,
+ 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00,
+ 0x00, 0x21, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x21, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00,
+ 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x42, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x42, 0x00, 0x00, 0x00,
+ 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00,
+ 0x00, 0x42, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x21, 0x00, 0x00, 0x00, 0x00, 0x42, 0x00, 0x00, 0x00, 0x00, 0x21, 0x00, 0x00, 0x00,
+ 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0xf8, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x00, 0x00, 0x00,
+ 0x00, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00,
+ 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
+ };
+ oled_write_raw_P(oled_keymap_vertical, sizeof(oled_keymap_vertical));
+ }
+}
+
+// Location of OLED keyboard's top left pixel, relative to the display
+static const int keymap_template[2] = {41, 0};
+// Location of key highlights top left pixels, relative to keymap_template {X, Y, Key length in px}
+static int keymap_coords[MATRIX_ROWS][MATRIX_COLS][3] = {
+ { {12, 15, 1}, {5, 0, 1}, {10, 0, 1}, {15, 0, 1}, {20, 0, 1}, {25, 0, 1}, {30, 0, 1}, {35, 0, 1}, {40, 0, 1}, {45, 0, 1}, {50, 0, 1}, {55, 0, 1}, {60, 0, 1}, {65, 0, 1}, {70, 0, 8}, {82, 0, 1} },
+ { {0, 5, 1}, {5, 5, 5}, {14, 5, 1}, {19, 5, 1}, {24, 5, 1}, {29, 5, 1}, {34, 5, 1}, {39, 5, 1}, {44, 5, 1}, {49, 5, 1}, {54, 5, 1}, {59, 5, 1}, {64, 5, 1}, {69, 5, 1}, {74, 5, 4}, {82, 5, 1} },
+ { {0, 10, 1}, {5, 10, 6}, {15, 10, 1}, {20, 10, 1}, {25, 10, 1}, {30, 10, 1}, {35, 10, 1}, {40, 10, 1}, {45, 10, 1}, {50, 10, 1}, {55, 10, 1}, {60, 10, 1}, {65, 10, 1}, {0, 0, 0}, {70, 10, 8}, {82, 10, 1} },
+ { {0, 15, 1}, {5, 15, 8}, {17, 15, 1}, {22, 15, 1}, {27, 15, 1}, {32, 15, 1}, {37, 15, 1}, {42, 15, 1}, {47, 15, 1}, {52, 15, 1}, {57, 15, 1}, {62, 15, 1}, {67, 15, 6}, {0, 0, 0}, {77, 15, 1}, {82, 15, 1} },
+ { {0, 20, 1}, {5, 20, 2}, {11, 20, 2}, {17, 20, 2}, {0, 0, 0}, {0, 0, 0}, {23, 20, 12}, {0, 0, 0}, {0, 0, 0}, {39, 20, 3}, {56, 20, 4}, {64, 20, 4}, {72, 20, 1}, {0, 0, 0}, {77, 20, 1}, {82, 20, 1} }
+};
+
+// Draw the bottom row of the keyboard (based on OLED config variables), update coordinates
+static void render_fn_row(void) {
+ // Update locations of spacebar and modifier key highlights
+ if ((split_space == false) && (three_mods_left == false)) {
+ keymap_coords[4][1][2] = 3;
+ keymap_coords[4][2][0] = 12;
+ keymap_coords[4][2][2] = 3;
+ keymap_coords[4][3][0] = 0;
+ keymap_coords[4][3][1] = 0;
+ keymap_coords[4][3][2] = 0;
+ keymap_coords[4][6][0] = 19;
+ keymap_coords[4][6][2] = 34;
+ } else if ((split_space == false) && (three_mods_left == true)) {
+ keymap_coords[4][6][2] = 30;
+ }
+ if ((split_space == false) && (three_mods_right == true)) {
+ keymap_coords[4][9][0] = 57;
+ keymap_coords[4][9][2] = 1;
+ keymap_coords[4][10][0] = 62;
+ keymap_coords[4][10][2] = 1;
+ keymap_coords[4][11][0] = 67;
+ keymap_coords[4][11][2] = 1;
+ }
+ // Draw modifiers
+ for (int i = 0; i < 16; i++) {
+ if (keymap_coords[4][i][2] != 0) {
+ for (int p = 0; p < keymap_coords[4][i][2]; p++) {
+ int x = keymap_template[0] + keymap_coords[4][i][0] + 2 + p;
+ write_pixel(x, 22, true);
+ }
+ }
+ }
+ // Draw second line for split spacebar
+ if (split_space == true) {
+ for (int i = 0; i < 6; i++) {
+ int x = keymap_template[0] + 46 + 2 + i;
+ write_pixel(x, 22, true);
+ }
+ }
+}
+
+// Update OLED keyboard with ISO layout, update coordinates
+static void render_iso(void) {
+ for (int i = 0; i < 6; i++) {
+ // Turn off ANSI enter
+ write_pixel(keymap_template[0] + 73 + i, keymap_template[1] + 12, false);
+ if (i < 4) {
+ // Turn off part of ANSI left shift
+ write_pixel(keymap_template[0] + 10 + i, keymap_template[1] + 17, false);
+ // Draw vertical line for ISO enter
+ write_pixel(keymap_template[0] + 79, keymap_template[1] + 8 + i, true);
+ }
+ }
+ // Update locations of shift and grave key highlights
+ keymap_coords[3][1][2] = 3;
+ keymap_coords[1][14][0] = 70;
+ keymap_coords[1][14][1] = 10;
+ keymap_coords[1][14][2] = 1;
+}
+
+// Toggles pixels surrounding key
+static void render_keymap(uint8_t key_row, uint8_t key_col, bool onoff) {
+ int length = keymap_coords[key_row][key_col][2] + 4;
+ int left = keymap_coords[key_row][key_col][0] + keymap_template[0];
+ int top = keymap_coords[key_row][key_col][1] + keymap_template[1];
+ int right = left + length - 1;
+ int bottom = top + 4;
+
+ // Special case 1 - Draw enter key on ISO layout, return
+ if ((ansi_layout == false) && (key_row == 2) && (key_col == 14)) {
+ for (int i = 0; i < 10; i++) {
+ write_pixel(keymap_template[0] + 81, keymap_template[1] + 5 + i, onoff);
+ if (i < 5) {
+ write_pixel(keymap_template[0] + 74, keymap_template[1] + 5 + i, onoff);
+ }
+ if (i < 6) {
+ write_pixel(keymap_template[0] + 75, keymap_template[1] + 9 + i, onoff);
+ }
+ if (i < 7) {
+ write_pixel(keymap_template[0] + 75 + i, keymap_template[1] + 5, onoff);
+ write_pixel(keymap_template[0] + 75 + i, keymap_template[1] + 14, onoff);
+ }
+ }
+ return;
+ }
+ // Draw top and bottom walls (horizontal for <length>px)
+ for (int x = 0; x < length; x++) {
+ write_pixel(left + x, top, onoff);
+ write_pixel(left + x, bottom, onoff);
+ }
+ // Draw left and right walls (vertical for 5px)
+ for (int y = 0; y < 5; y++) {
+ write_pixel(left, top + y, onoff);
+ write_pixel(right, top + y, onoff);
+ }
+ // Special case 2 - Draw right spacebar on split-space layout
+ if ((split_space == true) && (key_row == 4) && (key_col == 6)) {
+ int start = keymap_template[0] + 46;
+ int stop = keymap_template[0] + 55;
+ for (int x = start; x < stop; x++) {
+ write_pixel(x, top, onoff);
+ write_pixel(x, bottom, onoff);
+ }
+ for (int y = 0; y < 5; y++) {
+ write_pixel(start, top + y, onoff);
+ write_pixel(stop, top + y, onoff);
+ }
+ }
+}
+
+// Write active layer name
+static void render_layer_state(void) {
+ if (oled_horizontal) {
+ oled_set_cursor(0, 0);
+ } else {
+ oled_set_cursor(0, 15);
+ }
+ switch (get_highest_layer(layer_state)) {
+ case _MA:
+ oled_write_P(PSTR(MA_LAYER_NAME), false);
+ break;
+ case _L1:
+ oled_write_P(PSTR(L1_LAYER_NAME), false);
+ break;
+ case _L2:
+ oled_write_P(PSTR(L2_LAYER_NAME), false);
+ break;
+ case _L3:
+ oled_write_P(PSTR(L3_LAYER_NAME), false);
+ break;
+ default:
+ oled_write("ERROR", false);
+ break;
+ }
+}
+
+// Update WPM counters
+static void render_wpm_counters(int current_wpm) {
+ int cursorposition_cur = 2;
+ int cursorposition_max = 1;
+ if (oled_horizontal == false) {
+ cursorposition_cur = 13;
+ cursorposition_max = 14;
+ }
+
+ char wpm_counter[4];
+ wpm_counter[3] = '\0';
+ wpm_counter[2] = '0' + current_wpm % 10;
+ wpm_counter[1] = '0' + (current_wpm / 10) % 10;
+ wpm_counter[0] = '0' + (current_wpm / 100) % 10;
+ oled_set_cursor(0, cursorposition_cur);
+ oled_write(wpm_counter, false);
+
+ if (current_wpm > max_wpm) {
+ max_wpm = current_wpm;
+ wpm_limit = max_wpm + 20;
+ oled_set_cursor(0, cursorposition_max);
+ oled_write(wpm_counter, false);
+ }
+}
+
+// Update WPM snail icon
+static void render_wpm_icon(int current_wpm) {
+ // wpm_icon is used to prevent unnecessary redraw
+ if ((current_wpm < icon_med_wpm) && (wpm_icon != 0)) {
+ wpm_icon = 0;
+ } else if ((current_wpm >= icon_med_wpm) && (current_wpm < icon_fast_wpm) && (wpm_icon != 1)) {
+ wpm_icon = 1;
+ } else if ((current_wpm >= icon_fast_wpm) && (wpm_icon != 2)) {
+ wpm_icon = 2;
+ } else {
+ return;
+ }
+ static const char PROGMEM snails[][2][24] = {
+ {{0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x40, 0x20, 0xA0, 0x20, 0x40, 0x40, 0x80, 0x00, 0x00, 0x00, 0x80, 0x40, 0x20, 0x50, 0x88, 0x04, 0x00, 0x00},
+ {0x40, 0x60, 0x50, 0x4E, 0x51, 0x64, 0x4A, 0x51, 0x54, 0x49, 0x41, 0x62, 0x54, 0x49, 0x46, 0x41, 0x40, 0x30, 0x09, 0x04, 0x02, 0x01, 0x00, 0x00}},
+ {{0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x40, 0x40, 0x40, 0x40, 0x40, 0x80, 0x80, 0x00, 0x00, 0x00, 0x04, 0x98, 0x60, 0x80, 0x00, 0x00, 0x00, 0x00},
+ {0x60, 0x50, 0x54, 0x4A, 0x51, 0x64, 0x4A, 0x51, 0x55, 0x49, 0x41, 0x62, 0x54, 0x49, 0x46, 0x41, 0x21, 0x10, 0x0A, 0x08, 0x05, 0x02, 0x00, 0x00}},
+ {{0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x40, 0x40, 0x40, 0x40, 0x80, 0x80, 0x10, 0x10, 0x10, 0x20, 0x40, 0x40, 0xC0, 0x80, 0x80, 0x00, 0x00, 0x00},
+ {0x60, 0x58, 0x54, 0x62, 0x49, 0x54, 0x52, 0x51, 0x55, 0x49, 0x62, 0x52, 0x4D, 0x45, 0x46, 0x22, 0x21, 0x11, 0x10, 0x0A, 0x08, 0x05, 0x02, 0x00}}
+ };
+ if (oled_horizontal) {
+ oled_set_cursor(3, 1);
+ oled_write_raw_P(snails[wpm_icon][0], sizeof(snails[wpm_icon][0]));
+ oled_set_cursor(3, 2);
+ oled_write_raw_P(snails[wpm_icon][1], sizeof(snails[wpm_icon][1]));
+ } else {
+ oled_set_cursor(0, 11);
+ oled_write_raw_P(snails[wpm_icon][0], sizeof(snails[wpm_icon][0]));
+ oled_set_cursor(0, 12);
+ oled_write_raw_P(snails[wpm_icon][1], sizeof(snails[wpm_icon][1]));
+ }
+}
+
+// Update WPM graph
+static void render_wpm_graph(int current_wpm) {
+ int line_height = ((current_wpm / graph_top_wpm) * 7);
+ if (line_height > 7) {
+ line_height = 7;
+ }
+ // Count graph line pixels, return if nothing to draw
+ int pixel_count = line_height;
+ for (int i = 0; i < 63; i++) {
+ pixel_count += graph_lines[i];
+ }
+ if (pixel_count == 0) {
+ return;
+ }
+ // Shift array elements left or right depending on graph_direction, append new graph line
+ if (graph_direction) {
+ for (int i = 0; i < 63; i++) {
+ graph_lines[i] = graph_lines[i + 1];
+ }
+ graph_lines[63] = line_height;
+ } else {
+ for (int i = 63; i > 0; i--) {
+ graph_lines[i] = graph_lines[i - 1];
+ }
+ graph_lines[0] = line_height;
+ }
+ // Draw all graph lines (left to right, bottom to top)
+ int draw_count, arrpos;
+ for (int x = 1; x <= 127; x += 2) {
+ arrpos = x / 2;
+ draw_count = graph_lines[arrpos];
+ for (int y = 31; y >= 25; y--) {
+ if (draw_count > 0) {
+ write_pixel(x, y, true);
+ draw_count--;
+ } else {
+ write_pixel(x, y, false);
+ }
+ }
+ }
+}
+
+// Call OLED functions
+bool oled_task_user(void) {
+ // Draw OLED keyboard, prevent redraw
+ if (first_loop) {
+ render_background();
+ render_fn_row();
+ if (ansi_layout == false) {
+ render_iso();
+ }
+ first_loop = false;
+ }
+ // Get current WPM, subtract 25% for accuracy and prevent large jumps caused by simultaneous keypresses
+ int current_wpm = get_current_wpm();
+ // Note: This will most likely be removed once QMK's WPM calculation is updated
+ current_wpm -= current_wpm >> 2;
+ if (current_wpm > wpm_limit) {
+ current_wpm = max_wpm;
+ set_current_wpm(max_wpm);
+ }
+ // Write active layer name to display
+ render_layer_state();
+ // Update WPM counters
+ render_wpm_counters(current_wpm);
+ // Update WPM snail icon
+ render_wpm_icon(current_wpm);
+ // Update WPM graph every graph_refresh milliseconds
+ if (timer_elapsed(timer) > graph_refresh) {
+ render_wpm_graph(current_wpm);
+ timer = timer_read();