mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2025-01-09 03:08:47 +00:00
146 lines
No EOL
4.5 KiB
C
146 lines
No EOL
4.5 KiB
C
#include <furi.h>
|
|
#include <furi_hal.h>
|
|
|
|
#include <gui/elements.h>
|
|
|
|
#include <gui/gui.h>
|
|
#include <input/input.h>
|
|
|
|
#define TAG "Clock"
|
|
#define CLOCK_DATE_FORMAT "%.4d-%.2d-%.2d"
|
|
#define CLOCK_TIME_FORMAT "%.2d:%.2d:%.2d"
|
|
|
|
typedef enum {
|
|
EventTypeTick,
|
|
EventTypeKey,
|
|
} EventType;
|
|
|
|
typedef struct {
|
|
EventType type;
|
|
InputEvent input;
|
|
} PluginEvent;
|
|
|
|
typedef struct {
|
|
FuriHalRtcDateTime datetime;
|
|
} ClockState;
|
|
|
|
static void clock_input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
|
|
furi_assert(event_queue);
|
|
|
|
PluginEvent event = {.type = EventTypeKey, .input = *input_event};
|
|
furi_message_queue_put(event_queue, &event, FuriWaitForever);
|
|
}
|
|
|
|
static void clock_render_callback(Canvas* const canvas, void* ctx) {
|
|
canvas_clear(canvas);
|
|
canvas_set_color(canvas, ColorBlack);
|
|
|
|
ClockState* state = (ClockState*)acquire_mutex((ValueMutex*)ctx, 25);
|
|
|
|
char strings[2][20];
|
|
|
|
snprintf(
|
|
strings[0],
|
|
sizeof(strings[0]),
|
|
CLOCK_DATE_FORMAT,
|
|
state->datetime.year,
|
|
state->datetime.month,
|
|
state->datetime.day);
|
|
snprintf(
|
|
strings[1],
|
|
sizeof(strings[1]),
|
|
CLOCK_TIME_FORMAT,
|
|
state->datetime.hour,
|
|
state->datetime.minute,
|
|
state->datetime.second);
|
|
|
|
release_mutex((ValueMutex*)ctx, state);
|
|
canvas_set_font(canvas, FontBigNumbers);
|
|
canvas_draw_str_aligned(canvas, 64, 42 - 16, AlignCenter, AlignCenter, strings[1]);
|
|
canvas_set_font(canvas, FontSecondary);
|
|
canvas_draw_str_aligned(canvas, 64, 52 - 8, AlignCenter, AlignTop, strings[0]);
|
|
}
|
|
|
|
static void clock_state_init(ClockState* const state) {
|
|
furi_hal_rtc_get_datetime(&state->datetime);
|
|
}
|
|
|
|
// Runs every 1000ms by default
|
|
static void clock_tick(void* ctx) {
|
|
furi_assert(ctx);
|
|
FuriMessageQueue* event_queue = ctx;
|
|
PluginEvent event = {.type = EventTypeTick};
|
|
// It's OK to loose this event if system overloaded
|
|
furi_message_queue_put(event_queue, &event, 0);
|
|
}
|
|
|
|
int32_t clock_app(void* p) {
|
|
UNUSED(p);
|
|
FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
|
|
|
|
ClockState* plugin_state = malloc(sizeof(ClockState));
|
|
clock_state_init(plugin_state);
|
|
ValueMutex state_mutex;
|
|
if(!init_mutex(&state_mutex, plugin_state, sizeof(ClockState))) {
|
|
FURI_LOG_E(TAG, "cannot create mutex\r\n");
|
|
furi_message_queue_free(event_queue);
|
|
free(plugin_state);
|
|
return 255;
|
|
}
|
|
|
|
// Set system callbacks
|
|
ViewPort* view_port = view_port_alloc();
|
|
view_port_draw_callback_set(view_port, clock_render_callback, &state_mutex);
|
|
view_port_input_callback_set(view_port, clock_input_callback, event_queue);
|
|
FuriTimer* timer = furi_timer_alloc(clock_tick, FuriTimerTypePeriodic, event_queue);
|
|
furi_timer_start(timer, furi_kernel_get_tick_frequency());
|
|
|
|
// Open GUI and register view_port
|
|
Gui* gui = furi_record_open(RECORD_GUI);
|
|
gui_add_view_port(gui, view_port, GuiLayerFullscreen);
|
|
|
|
// Main loop
|
|
PluginEvent event;
|
|
for(bool processing = true; processing;) {
|
|
FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
|
|
ClockState* plugin_state = (ClockState*)acquire_mutex_block(&state_mutex);
|
|
|
|
if(event_status == FuriStatusOk) {
|
|
// press events
|
|
if(event.type == EventTypeKey) {
|
|
if(event.input.type == InputTypeShort || event.input.type == InputTypeRepeat) {
|
|
switch(event.input.key) {
|
|
case InputKeyUp:
|
|
case InputKeyDown:
|
|
case InputKeyRight:
|
|
case InputKeyLeft:
|
|
case InputKeyOk:
|
|
break;
|
|
case InputKeyBack:
|
|
// Exit the plugin
|
|
processing = false;
|
|
break;
|
|
}
|
|
}
|
|
} else if(event.type == EventTypeTick) {
|
|
furi_hal_rtc_get_datetime(&plugin_state->datetime);
|
|
}
|
|
} else {
|
|
FURI_LOG_D(TAG, "furi_message_queue: event timeout");
|
|
// event timeout
|
|
}
|
|
|
|
view_port_update(view_port);
|
|
release_mutex(&state_mutex, plugin_state);
|
|
}
|
|
|
|
furi_timer_free(timer);
|
|
view_port_enabled_set(view_port, false);
|
|
gui_remove_view_port(gui, view_port);
|
|
furi_record_close(RECORD_GUI);
|
|
view_port_free(view_port);
|
|
furi_message_queue_free(event_queue);
|
|
delete_mutex(&state_mutex);
|
|
|
|
return 0;
|
|
} |