b2a12d091a
* Menu app. Lib: add mlib submodule. * Menu: new startup lib dependency definition * Menu: hierarchy in menu. Cube: fix heap1/4 inconsistency, stack protection. * GUI: rendering pipeline initial version. * GUI: layered widget composing, FURI record. Menu: FURI record, api. * GUI: input dispatching. Menu: switch to input from GUI. * GUI, MENU: code style cleanup, fix type conversion warnings. * GUI, Menu: syntax check. * Makefile: check and reinit submodules, format. * Menu: lock on event processing. Makefile: proper submodule initialization. * Menu: fix stack corruption by queue. * GUI: refactor. * Makefile: format rule fix, st-util pid. * GUI, Menu, FURI: format with clang-format. * GUI, MENU: locks in critical sections, fix stack corruption, ready signaling. * Makefile: clang format rule cleanup. * GUI,MENU: migrate to new API. * Applications: PRODUCTION_HW variable, skip drivers build on local target. * refactor production build * add git to dockerfile * GUI: uncomment lock block Co-authored-by: Aleksandr Kutuzov <aku@plooks.com>
177 lines
4.3 KiB
C
177 lines
4.3 KiB
C
#include "gui.h"
|
|
#include "gui_i.h"
|
|
|
|
#include <flipper.h>
|
|
#include <stdio.h>
|
|
#include <m-array.h>
|
|
|
|
#include "gui_event.h"
|
|
#include "canvas.h"
|
|
#include "canvas_i.h"
|
|
#include "widget.h"
|
|
#include "widget_i.h"
|
|
|
|
ARRAY_DEF(WidgetArray, Widget*, M_PTR_OPLIST);
|
|
|
|
struct GUI {
|
|
GUIEvent* event;
|
|
Canvas* canvas;
|
|
WidgetArray_t widgets_status_bar;
|
|
WidgetArray_t widgets;
|
|
WidgetArray_t widgets_fs;
|
|
WidgetArray_t widgets_dialog;
|
|
};
|
|
|
|
void gui_widget_status_bar_add(GUI* gui, Widget* widget) {
|
|
assert(gui);
|
|
assert(widget);
|
|
|
|
gui_event_lock(gui->event);
|
|
WidgetArray_push_back(gui->widgets_status_bar, widget);
|
|
widget_gui_set(widget, gui);
|
|
gui_event_unlock(gui->event);
|
|
|
|
gui_update(gui);
|
|
}
|
|
|
|
void gui_widget_add(GUI* gui, Widget* widget) {
|
|
assert(gui);
|
|
assert(widget);
|
|
|
|
gui_event_lock(gui->event);
|
|
WidgetArray_push_back(gui->widgets, widget);
|
|
widget_gui_set(widget, gui);
|
|
gui_event_unlock(gui->event);
|
|
|
|
gui_update(gui);
|
|
}
|
|
|
|
void gui_widget_fs_add(GUI* gui, Widget* widget) {
|
|
assert(gui);
|
|
assert(widget);
|
|
|
|
gui_event_lock(gui->event);
|
|
WidgetArray_push_back(gui->widgets_fs, widget);
|
|
widget_gui_set(widget, gui);
|
|
gui_event_unlock(gui->event);
|
|
|
|
gui_update(gui);
|
|
}
|
|
|
|
void gui_widget_dialog_add(GUI* gui, Widget* widget) {
|
|
assert(gui);
|
|
assert(widget);
|
|
|
|
gui_event_lock(gui->event);
|
|
WidgetArray_push_back(gui->widgets_dialog, widget);
|
|
widget_gui_set(widget, gui);
|
|
gui_event_unlock(gui->event);
|
|
|
|
gui_update(gui);
|
|
}
|
|
|
|
void gui_update(GUI* gui) {
|
|
assert(gui);
|
|
GUIMessage message;
|
|
message.type = GUIMessageTypeRedraw;
|
|
gui_event_messsage_send(gui->event, &message);
|
|
}
|
|
|
|
Widget* gui_widget_find_enabled(WidgetArray_t array) {
|
|
size_t widgets_count = WidgetArray_size(array);
|
|
for(size_t i = 0; i < widgets_count; i++) {
|
|
Widget* widget = *WidgetArray_get(array, widgets_count - i - 1);
|
|
if(widget_is_enabled(widget)) {
|
|
return widget;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool gui_redraw_fs(GUI* gui) {
|
|
canvas_frame_set(gui->canvas, 0, 0, 128, 64);
|
|
Widget* widget = gui_widget_find_enabled(gui->widgets_fs);
|
|
if(widget) {
|
|
widget_draw(widget, gui->canvas);
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void gui_redraw_status_bar(GUI* gui) {
|
|
canvas_frame_set(gui->canvas, 0, 0, 128, 64);
|
|
Widget* widget = gui_widget_find_enabled(gui->widgets_status_bar);
|
|
if(widget) widget_draw(widget, gui->canvas);
|
|
}
|
|
|
|
void gui_redraw_normal(GUI* gui) {
|
|
canvas_frame_set(gui->canvas, 0, 9, 128, 55);
|
|
Widget* widget = gui_widget_find_enabled(gui->widgets);
|
|
if(widget) widget_draw(widget, gui->canvas);
|
|
}
|
|
|
|
void gui_redraw_dialogs(GUI* gui) {
|
|
canvas_frame_set(gui->canvas, 10, 20, 118, 44);
|
|
Widget* widget = gui_widget_find_enabled(gui->widgets_dialog);
|
|
if(widget) widget_draw(widget, gui->canvas);
|
|
}
|
|
|
|
void gui_redraw(GUI* gui) {
|
|
assert(gui);
|
|
|
|
if(!gui_redraw_fs(gui)) {
|
|
gui_redraw_status_bar(gui);
|
|
gui_redraw_normal(gui);
|
|
}
|
|
gui_redraw_dialogs(gui);
|
|
|
|
canvas_commit(gui->canvas);
|
|
}
|
|
|
|
void gui_input(GUI* gui, InputEvent* input_event) {
|
|
assert(gui);
|
|
|
|
Widget* widget = gui_widget_find_enabled(gui->widgets_dialog);
|
|
if(!widget) widget = gui_widget_find_enabled(gui->widgets_fs);
|
|
if(!widget) widget = gui_widget_find_enabled(gui->widgets);
|
|
|
|
if(widget) {
|
|
widget_input(widget, input_event);
|
|
}
|
|
}
|
|
|
|
GUI* gui_alloc() {
|
|
GUI* gui = furi_alloc(sizeof(GUI));
|
|
// Initialize widget arrays
|
|
WidgetArray_init(gui->widgets_status_bar);
|
|
WidgetArray_init(gui->widgets);
|
|
WidgetArray_init(gui->widgets_fs);
|
|
WidgetArray_init(gui->widgets_dialog);
|
|
// Event dispatcher
|
|
gui->event = gui_event_alloc();
|
|
// Drawing canvas
|
|
gui->canvas = canvas_alloc();
|
|
|
|
return gui;
|
|
}
|
|
|
|
void gui_task(void* p) {
|
|
GUI* gui = gui_alloc();
|
|
// Create FURI record
|
|
if(!furi_create_deprecated("gui", gui, sizeof(gui))) {
|
|
printf("[gui_task] cannot create the gui record\n");
|
|
furiac_exit(NULL);
|
|
}
|
|
furiac_ready();
|
|
// Forever dispatch
|
|
while(1) {
|
|
GUIMessage message = gui_event_message_next(gui->event);
|
|
if(message.type == GUIMessageTypeRedraw) {
|
|
gui_redraw(gui);
|
|
} else if(message.type == GUIMessageTypeInput) {
|
|
gui_input(gui, &message.input);
|
|
}
|
|
}
|
|
}
|