2023-07-18 14:43:30 -07:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Copyright (c) 2023 The ZMK Contributors
|
|
|
|
* SPDX-License-Identifier: MIT
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <zephyr/kernel.h>
|
|
|
|
|
|
|
|
#include <zephyr/logging/log.h>
|
|
|
|
LOG_MODULE_DECLARE(zmk, CONFIG_ZMK_LOG_LEVEL);
|
|
|
|
|
2024-01-16 16:28:32 -08:00
|
|
|
#include <zmk/battery.h>
|
2023-07-18 14:43:30 -07:00
|
|
|
#include <zmk/display.h>
|
|
|
|
#include "status.h"
|
|
|
|
#include <zmk/events/usb_conn_state_changed.h>
|
|
|
|
#include <zmk/event_manager.h>
|
|
|
|
#include <zmk/events/battery_state_changed.h>
|
|
|
|
#include <zmk/events/ble_active_profile_changed.h>
|
2023-08-27 16:33:29 -07:00
|
|
|
#include <zmk/events/endpoint_changed.h>
|
2023-07-18 14:43:30 -07:00
|
|
|
#include <zmk/events/wpm_state_changed.h>
|
|
|
|
#include <zmk/events/layer_state_changed.h>
|
|
|
|
#include <zmk/usb.h>
|
|
|
|
#include <zmk/ble.h>
|
|
|
|
#include <zmk/endpoints.h>
|
|
|
|
#include <zmk/keymap.h>
|
|
|
|
#include <zmk/wpm.h>
|
|
|
|
|
|
|
|
static sys_slist_t widgets = SYS_SLIST_STATIC_INIT(&widgets);
|
|
|
|
|
|
|
|
struct output_status_state {
|
2023-08-27 16:33:29 -07:00
|
|
|
struct zmk_endpoint_instance selected_endpoint;
|
2023-10-08 16:30:23 -07:00
|
|
|
int active_profile_index;
|
2023-07-18 14:43:30 -07:00
|
|
|
bool active_profile_connected;
|
|
|
|
bool active_profile_bonded;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct layer_status_state {
|
|
|
|
uint8_t index;
|
|
|
|
const char *label;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct wpm_status_state {
|
|
|
|
uint8_t wpm;
|
|
|
|
};
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
static void draw_top(lv_obj_t *widget, lv_color_t cbuf[], const struct status_state *state) {
|
2023-07-18 14:43:30 -07:00
|
|
|
lv_obj_t *canvas = lv_obj_get_child(widget, 0);
|
|
|
|
|
|
|
|
lv_draw_label_dsc_t label_dsc;
|
|
|
|
init_label_dsc(&label_dsc, LVGL_FOREGROUND, &lv_font_montserrat_16, LV_TEXT_ALIGN_RIGHT);
|
|
|
|
lv_draw_label_dsc_t label_dsc_wpm;
|
|
|
|
init_label_dsc(&label_dsc_wpm, LVGL_FOREGROUND, &lv_font_unscii_8, LV_TEXT_ALIGN_RIGHT);
|
|
|
|
lv_draw_rect_dsc_t rect_black_dsc;
|
|
|
|
init_rect_dsc(&rect_black_dsc, LVGL_BACKGROUND);
|
|
|
|
lv_draw_rect_dsc_t rect_white_dsc;
|
|
|
|
init_rect_dsc(&rect_white_dsc, LVGL_FOREGROUND);
|
|
|
|
lv_draw_line_dsc_t line_dsc;
|
|
|
|
init_line_dsc(&line_dsc, LVGL_FOREGROUND, 1);
|
|
|
|
|
|
|
|
// Fill background
|
|
|
|
lv_canvas_draw_rect(canvas, 0, 0, CANVAS_SIZE, CANVAS_SIZE, &rect_black_dsc);
|
|
|
|
|
|
|
|
// Draw battery
|
|
|
|
draw_battery(canvas, state);
|
|
|
|
|
|
|
|
// Draw output status
|
|
|
|
char output_text[10] = {};
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
switch (state->selected_endpoint.transport) {
|
|
|
|
case ZMK_TRANSPORT_USB:
|
2023-07-18 14:43:30 -07:00
|
|
|
strcat(output_text, LV_SYMBOL_USB);
|
|
|
|
break;
|
2023-08-27 16:33:29 -07:00
|
|
|
case ZMK_TRANSPORT_BLE:
|
|
|
|
if (state->active_profile_bonded) {
|
|
|
|
if (state->active_profile_connected) {
|
2023-07-18 14:43:30 -07:00
|
|
|
strcat(output_text, LV_SYMBOL_WIFI);
|
|
|
|
} else {
|
|
|
|
strcat(output_text, LV_SYMBOL_CLOSE);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
strcat(output_text, LV_SYMBOL_SETTINGS);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_canvas_draw_text(canvas, 0, 0, CANVAS_SIZE, &label_dsc, output_text);
|
|
|
|
|
|
|
|
// Draw WPM
|
|
|
|
lv_canvas_draw_rect(canvas, 0, 21, 68, 42, &rect_white_dsc);
|
|
|
|
lv_canvas_draw_rect(canvas, 1, 22, 66, 40, &rect_black_dsc);
|
|
|
|
|
|
|
|
char wpm_text[6] = {};
|
2023-08-27 16:33:29 -07:00
|
|
|
snprintf(wpm_text, sizeof(wpm_text), "%d", state->wpm[9]);
|
2023-07-18 14:43:30 -07:00
|
|
|
lv_canvas_draw_text(canvas, 42, 52, 24, &label_dsc_wpm, wpm_text);
|
|
|
|
|
|
|
|
int max = 0;
|
|
|
|
int min = 256;
|
|
|
|
|
|
|
|
for (int i = 0; i < 10; i++) {
|
2023-08-27 16:33:29 -07:00
|
|
|
if (state->wpm[i] > max) {
|
|
|
|
max = state->wpm[i];
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
2023-08-27 16:33:29 -07:00
|
|
|
if (state->wpm[i] < min) {
|
|
|
|
min = state->wpm[i];
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int range = max - min;
|
|
|
|
if (range == 0) {
|
|
|
|
range = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_point_t points[10];
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
points[i].x = 2 + i * 7;
|
2023-08-27 16:33:29 -07:00
|
|
|
points[i].y = 60 - (state->wpm[i] - min) * 36 / range;
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
lv_canvas_draw_line(canvas, points, 10, &line_dsc);
|
|
|
|
|
|
|
|
// Rotate canvas
|
|
|
|
rotate_canvas(canvas, cbuf);
|
|
|
|
}
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
static void draw_middle(lv_obj_t *widget, lv_color_t cbuf[], const struct status_state *state) {
|
2023-07-18 14:43:30 -07:00
|
|
|
lv_obj_t *canvas = lv_obj_get_child(widget, 1);
|
|
|
|
|
|
|
|
lv_draw_rect_dsc_t rect_black_dsc;
|
|
|
|
init_rect_dsc(&rect_black_dsc, LVGL_BACKGROUND);
|
|
|
|
lv_draw_rect_dsc_t rect_white_dsc;
|
|
|
|
init_rect_dsc(&rect_white_dsc, LVGL_FOREGROUND);
|
|
|
|
lv_draw_arc_dsc_t arc_dsc;
|
|
|
|
init_arc_dsc(&arc_dsc, LVGL_FOREGROUND, 2);
|
|
|
|
lv_draw_arc_dsc_t arc_dsc_filled;
|
|
|
|
init_arc_dsc(&arc_dsc_filled, LVGL_FOREGROUND, 9);
|
|
|
|
lv_draw_label_dsc_t label_dsc;
|
|
|
|
init_label_dsc(&label_dsc, LVGL_FOREGROUND, &lv_font_montserrat_18, LV_TEXT_ALIGN_CENTER);
|
|
|
|
lv_draw_label_dsc_t label_dsc_black;
|
|
|
|
init_label_dsc(&label_dsc_black, LVGL_BACKGROUND, &lv_font_montserrat_18, LV_TEXT_ALIGN_CENTER);
|
|
|
|
|
|
|
|
// Fill background
|
|
|
|
lv_canvas_draw_rect(canvas, 0, 0, CANVAS_SIZE, CANVAS_SIZE, &rect_black_dsc);
|
|
|
|
|
|
|
|
// Draw circles
|
|
|
|
int circle_offsets[5][2] = {
|
|
|
|
{13, 13}, {55, 13}, {34, 34}, {13, 55}, {55, 55},
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i = 0; i < 5; i++) {
|
2023-10-08 16:30:23 -07:00
|
|
|
bool selected = i == state->active_profile_index;
|
2023-07-18 14:43:30 -07:00
|
|
|
|
2023-11-15 16:06:21 -08:00
|
|
|
lv_canvas_draw_arc(canvas, circle_offsets[i][0], circle_offsets[i][1], 13, 0, 360,
|
2023-07-18 14:43:30 -07:00
|
|
|
&arc_dsc);
|
|
|
|
|
|
|
|
if (selected) {
|
|
|
|
lv_canvas_draw_arc(canvas, circle_offsets[i][0], circle_offsets[i][1], 9, 0, 359,
|
|
|
|
&arc_dsc_filled);
|
|
|
|
}
|
|
|
|
|
|
|
|
char label[2];
|
|
|
|
snprintf(label, sizeof(label), "%d", i + 1);
|
|
|
|
lv_canvas_draw_text(canvas, circle_offsets[i][0] - 8, circle_offsets[i][1] - 10, 16,
|
|
|
|
(selected ? &label_dsc_black : &label_dsc), label);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rotate canvas
|
|
|
|
rotate_canvas(canvas, cbuf);
|
|
|
|
}
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
static void draw_bottom(lv_obj_t *widget, lv_color_t cbuf[], const struct status_state *state) {
|
2023-07-18 14:43:30 -07:00
|
|
|
lv_obj_t *canvas = lv_obj_get_child(widget, 2);
|
|
|
|
|
|
|
|
lv_draw_rect_dsc_t rect_black_dsc;
|
|
|
|
init_rect_dsc(&rect_black_dsc, LVGL_BACKGROUND);
|
|
|
|
lv_draw_label_dsc_t label_dsc;
|
|
|
|
init_label_dsc(&label_dsc, LVGL_FOREGROUND, &lv_font_montserrat_14, LV_TEXT_ALIGN_CENTER);
|
|
|
|
|
|
|
|
// Fill background
|
|
|
|
lv_canvas_draw_rect(canvas, 0, 0, CANVAS_SIZE, CANVAS_SIZE, &rect_black_dsc);
|
|
|
|
|
|
|
|
// Draw layer
|
2023-08-27 16:33:29 -07:00
|
|
|
if (state->layer_label == NULL) {
|
2023-11-15 16:06:21 -08:00
|
|
|
char text[10] = {};
|
2023-07-18 14:43:30 -07:00
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
sprintf(text, "LAYER %i", state->layer_index);
|
2023-07-18 14:43:30 -07:00
|
|
|
|
|
|
|
lv_canvas_draw_text(canvas, 0, 5, 68, &label_dsc, text);
|
|
|
|
} else {
|
2023-08-27 16:33:29 -07:00
|
|
|
lv_canvas_draw_text(canvas, 0, 5, 68, &label_dsc, state->layer_label);
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rotate canvas
|
|
|
|
rotate_canvas(canvas, cbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void set_battery_status(struct zmk_widget_status *widget,
|
|
|
|
struct battery_status_state state) {
|
|
|
|
#if IS_ENABLED(CONFIG_USB_DEVICE_STACK)
|
|
|
|
widget->state.charging = state.usb_present;
|
|
|
|
#endif /* IS_ENABLED(CONFIG_USB_DEVICE_STACK) */
|
|
|
|
|
|
|
|
widget->state.battery = state.level;
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
draw_top(widget->obj, widget->cbuf, &widget->state);
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void battery_status_update_cb(struct battery_status_state state) {
|
|
|
|
struct zmk_widget_status *widget;
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&widgets, widget, node) { set_battery_status(widget, state); }
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct battery_status_state battery_status_get_state(const zmk_event_t *eh) {
|
2024-02-19 16:15:19 -08:00
|
|
|
const struct zmk_battery_state_changed *ev = as_zmk_battery_state_changed(eh);
|
|
|
|
|
2023-07-18 14:43:30 -07:00
|
|
|
return (struct battery_status_state) {
|
2024-02-19 16:15:19 -08:00
|
|
|
.level = (ev != NULL) ? ev->state_of_charge : zmk_battery_state_of_charge(),
|
2023-07-18 14:43:30 -07:00
|
|
|
#if IS_ENABLED(CONFIG_USB_DEVICE_STACK)
|
|
|
|
.usb_present = zmk_usb_is_powered(),
|
|
|
|
#endif /* IS_ENABLED(CONFIG_USB_DEVICE_STACK) */
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
ZMK_DISPLAY_WIDGET_LISTENER(widget_battery_status, struct battery_status_state,
|
|
|
|
battery_status_update_cb, battery_status_get_state)
|
|
|
|
|
|
|
|
ZMK_SUBSCRIPTION(widget_battery_status, zmk_battery_state_changed);
|
|
|
|
#if IS_ENABLED(CONFIG_USB_DEVICE_STACK)
|
|
|
|
ZMK_SUBSCRIPTION(widget_battery_status, zmk_usb_conn_state_changed);
|
|
|
|
#endif /* IS_ENABLED(CONFIG_USB_DEVICE_STACK) */
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
static void set_output_status(struct zmk_widget_status *widget,
|
|
|
|
const struct output_status_state *state) {
|
|
|
|
widget->state.selected_endpoint = state->selected_endpoint;
|
2023-10-08 16:30:23 -07:00
|
|
|
widget->state.active_profile_index = state->active_profile_index;
|
2023-08-27 16:33:29 -07:00
|
|
|
widget->state.active_profile_connected = state->active_profile_connected;
|
|
|
|
widget->state.active_profile_bonded = state->active_profile_bonded;
|
2023-07-18 14:43:30 -07:00
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
draw_top(widget->obj, widget->cbuf, &widget->state);
|
|
|
|
draw_middle(widget->obj, widget->cbuf2, &widget->state);
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void output_status_update_cb(struct output_status_state state) {
|
|
|
|
struct zmk_widget_status *widget;
|
2023-08-27 16:33:29 -07:00
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&widgets, widget, node) { set_output_status(widget, &state); }
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct output_status_state output_status_get_state(const zmk_event_t *_eh) {
|
2023-08-27 16:33:29 -07:00
|
|
|
return (struct output_status_state){
|
|
|
|
.selected_endpoint = zmk_endpoints_selected(),
|
2023-10-08 16:30:23 -07:00
|
|
|
.active_profile_index = zmk_ble_active_profile_index(),
|
2023-08-27 16:33:29 -07:00
|
|
|
.active_profile_connected = zmk_ble_active_profile_is_connected(),
|
|
|
|
.active_profile_bonded = !zmk_ble_active_profile_is_open(),
|
|
|
|
};
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ZMK_DISPLAY_WIDGET_LISTENER(widget_output_status, struct output_status_state,
|
|
|
|
output_status_update_cb, output_status_get_state)
|
2023-08-27 16:33:29 -07:00
|
|
|
ZMK_SUBSCRIPTION(widget_output_status, zmk_endpoint_changed);
|
2023-07-18 14:43:30 -07:00
|
|
|
|
|
|
|
#if IS_ENABLED(CONFIG_USB_DEVICE_STACK)
|
|
|
|
ZMK_SUBSCRIPTION(widget_output_status, zmk_usb_conn_state_changed);
|
|
|
|
#endif
|
|
|
|
#if defined(CONFIG_ZMK_BLE)
|
|
|
|
ZMK_SUBSCRIPTION(widget_output_status, zmk_ble_active_profile_changed);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void set_layer_status(struct zmk_widget_status *widget, struct layer_status_state state) {
|
|
|
|
widget->state.layer_index = state.index;
|
|
|
|
widget->state.layer_label = state.label;
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
draw_bottom(widget->obj, widget->cbuf3, &widget->state);
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layer_status_update_cb(struct layer_status_state state) {
|
|
|
|
struct zmk_widget_status *widget;
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&widgets, widget, node) { set_layer_status(widget, state); }
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct layer_status_state layer_status_get_state(const zmk_event_t *eh) {
|
|
|
|
uint8_t index = zmk_keymap_highest_layer_active();
|
2023-10-06 18:04:41 -07:00
|
|
|
return (struct layer_status_state){.index = index, .label = zmk_keymap_layer_name(index)};
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ZMK_DISPLAY_WIDGET_LISTENER(widget_layer_status, struct layer_status_state, layer_status_update_cb,
|
|
|
|
layer_status_get_state)
|
|
|
|
|
|
|
|
ZMK_SUBSCRIPTION(widget_layer_status, zmk_layer_state_changed);
|
|
|
|
|
|
|
|
static void set_wpm_status(struct zmk_widget_status *widget, struct wpm_status_state state) {
|
|
|
|
for (int i = 0; i < 9; i++) {
|
|
|
|
widget->state.wpm[i] = widget->state.wpm[i + 1];
|
|
|
|
}
|
|
|
|
widget->state.wpm[9] = state.wpm;
|
|
|
|
|
2023-08-27 16:33:29 -07:00
|
|
|
draw_top(widget->obj, widget->cbuf, &widget->state);
|
2023-07-18 14:43:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void wpm_status_update_cb(struct wpm_status_state state) {
|
|
|
|
struct zmk_widget_status *widget;
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&widgets, widget, node) { set_wpm_status(widget, state); }
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wpm_status_state wpm_status_get_state(const zmk_event_t *eh) {
|
|
|
|
return (struct wpm_status_state){.wpm = zmk_wpm_get_state()};
|
|
|
|
};
|
|
|
|
|
|
|
|
ZMK_DISPLAY_WIDGET_LISTENER(widget_wpm_status, struct wpm_status_state, wpm_status_update_cb,
|
|
|
|
wpm_status_get_state)
|
|
|
|
ZMK_SUBSCRIPTION(widget_wpm_status, zmk_wpm_state_changed);
|
|
|
|
|
|
|
|
int zmk_widget_status_init(struct zmk_widget_status *widget, lv_obj_t *parent) {
|
|
|
|
widget->obj = lv_obj_create(parent);
|
|
|
|
lv_obj_set_size(widget->obj, 160, 68);
|
|
|
|
lv_obj_t *top = lv_canvas_create(widget->obj);
|
|
|
|
lv_obj_align(top, LV_ALIGN_TOP_RIGHT, 0, 0);
|
|
|
|
lv_canvas_set_buffer(top, widget->cbuf, CANVAS_SIZE, CANVAS_SIZE, LV_IMG_CF_TRUE_COLOR);
|
|
|
|
lv_obj_t *middle = lv_canvas_create(widget->obj);
|
|
|
|
lv_obj_align(middle, LV_ALIGN_TOP_LEFT, 24, 0);
|
|
|
|
lv_canvas_set_buffer(middle, widget->cbuf2, CANVAS_SIZE, CANVAS_SIZE, LV_IMG_CF_TRUE_COLOR);
|
|
|
|
lv_obj_t *bottom = lv_canvas_create(widget->obj);
|
|
|
|
lv_obj_align(bottom, LV_ALIGN_TOP_LEFT, -44, 0);
|
|
|
|
lv_canvas_set_buffer(bottom, widget->cbuf3, CANVAS_SIZE, CANVAS_SIZE, LV_IMG_CF_TRUE_COLOR);
|
|
|
|
|
|
|
|
sys_slist_append(&widgets, &widget->node);
|
|
|
|
widget_battery_status_init();
|
|
|
|
widget_output_status_init();
|
|
|
|
widget_layer_status_init();
|
|
|
|
widget_wpm_status_init();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_obj_t *zmk_widget_status_obj(struct zmk_widget_status *widget) { return widget->obj; }
|