aboutsummaryrefslogtreecommitdiff
path: root/libgui/gui.c
diff options
context:
space:
mode:
Diffstat (limited to 'libgui/gui.c')
-rw-r--r--libgui/gui.c627
1 files changed, 14 insertions, 613 deletions
diff --git a/libgui/gui.c b/libgui/gui.c
index 8eb7ce9..2083f23 100644
--- a/libgui/gui.c
+++ b/libgui/gui.c
@@ -1,624 +1,25 @@
-// MIT License, Copyright (c) 2020 Marvin Borner
-// Mostly GFX function wrappers
-// TODO: Reduce code duplication
+// MIT License, Copyright (c) 2021 Marvin Borner
-#include <assert.h>
#include <def.h>
-#include <gfx.h>
#include <gui.h>
-#include <input.h>
-#include <list.h>
-#include <mem.h>
-#include <msg.h>
-#include <str.h>
-#include <sys.h>
+#include <print.h>
-// TODO: Use list (and add index-based access)
-#define MAX_WINDOWS 10
-#define BORDER 2
-#define PERC(c, e) ((u32)((double)(c) * (double)(e) / 100.0))
+#define WM_PATH "/bin/wm"
-u32 window_count = 0;
-static struct window windows[MAX_WINDOWS] = { 0 };
-
-static struct window *new_window(const char *title, int x, int y, u32 width, u32 height, int flags)
-{
- assert(window_count + 1 <= MAX_WINDOWS);
-
- struct window *win = &windows[window_count + 1];
- win->ctx = malloc(sizeof(*win->ctx));
- win->ctx->x = x > 0 ? x : 50;
- win->ctx->y = y > 0 ? y : 50;
- win->ctx->width = width > 0 ? width : 600;
- win->ctx->height = height > 0 ? height : 400;
- win->ctx->flags = flags;
- win->id = window_count + 1;
- win->title = title;
- win->childs = list_new();
-
- gfx_new_ctx(win->ctx);
-
- if (!win->ctx->fb)
- return NULL;
-
- window_count++;
-
- return win;
-}
-
-static struct element *get_root(u32 window_id)
-{
- struct list *childs = windows[window_id].childs;
- if (!childs || !childs->head || !childs->head->data)
- return NULL;
- return childs->head->data;
-}
-
-static void merge_elements(struct element *container)
-{
- if (!container || !container->childs || !container->childs->head)
- return;
-
- struct node *iterator = container->childs->head;
- while (iterator != NULL) {
- struct element *elem = iterator->data;
- struct context *ctx = elem->ctx;
- merge_elements(elem);
- gfx_ctx_on_ctx(container->ctx, ctx, ctx->x, ctx->y);
- iterator = iterator->next;
- }
-}
-
-static void free_context(struct context *ctx)
-{
- if (!ctx)
- return;
-
- free(ctx->fb);
- ctx->fb = NULL;
- free(ctx);
- ctx = NULL;
-}
-
-static void remove_childs(struct element *elem);
-static void remove_element(struct element *elem)
-{
- if (!elem)
- return;
-
- remove_childs(elem);
- free_context(elem->ctx);
- free(elem->data);
- elem->data = NULL;
- free(elem);
- elem = NULL;
-}
-
-static void remove_childs(struct element *elem)
-{
- if (!elem || !elem->childs || !elem->childs->head)
- return;
-
- struct node *iterator = elem->childs->head;
- while (iterator != NULL) {
- struct element *child = iterator->data;
- remove_element(child);
- iterator = iterator->next;
- }
-
- list_destroy(elem->childs);
-}
-
-static void remove_window(struct window *win)
-{
- if (!win || !win->childs || !win->childs->head)
- return;
-
- struct node *iterator = win->childs->head;
- while (iterator != NULL) {
- struct element *child = iterator->data;
- remove_element(child);
- iterator = iterator->next;
- }
-
- free_context(win->ctx);
- list_destroy(win->childs);
-}
-
-static void remove_all()
-{
- for (int i = 0; i < MAX_WINDOWS; i++) {
- remove_window(&windows[i]);
- }
-}
-
-static struct element *element_at(struct element *container, int x, int y)
-{
- if (!container || !container->childs || !container->childs->head)
- return NULL;
-
- struct node *iterator = container->childs->head;
- while (iterator != NULL) {
- struct context *ctx = ((struct element *)iterator->data)->ctx;
-
- int relative_x, relative_y;
- if (container->type == GUI_TYPE_ROOT) {
- relative_x = ctx->x;
- relative_y = ctx->y;
- } else {
- relative_x = ctx->x + container->ctx->x;
- relative_y = ctx->y + container->ctx->y;
- }
-
- if (ctx != container->ctx && ctx->flags & WF_RELATIVE && x >= relative_x &&
- x <= relative_x + (int)ctx->width && y >= relative_y &&
- y <= relative_y + (int)ctx->height) {
- struct element *recursive = NULL;
- if ((recursive = element_at(iterator->data, x, y)))
- return recursive;
- else
- return iterator->data;
- }
-
- iterator = iterator->next;
- }
-
- return NULL;
-}
-
-static int absolute_x_off(struct element *elem)
-{
- if (!elem->parent)
- return 0;
-
- int x = 0;
-
- struct element *iterator = elem;
- while ((iterator = iterator->parent) && iterator->ctx)
- if (iterator->parent)
- x += iterator->ctx->x;
-
- return x;
-}
-
-static int absolute_y_off(struct element *elem)
-{
- if (!elem->parent)
- return 0;
-
- int y = 0;
-
- struct element *iterator = elem;
- while ((iterator = iterator->parent) && iterator->ctx) {
- if (iterator->parent)
- y += iterator->ctx->y;
- }
-
- return y;
-}
-
-struct context *gui_get_context(int x, int y, u32 width, u32 height)
+s32 gui_new_window(struct gui_window *win)
{
- struct context *ctx = malloc(sizeof(*ctx));
- ctx->pid = getpid();
- ctx->x = x;
- ctx->y = y;
- ctx->width = width;
- ctx->height = height;
- ctx->bpp = 32; // TODO: Dynamic bpp
- ctx->pitch = ctx->width * (ctx->bpp >> 3);
- ctx->fb = malloc(ctx->height * ctx->pitch);
- memset(ctx->fb, 0, ctx->height * ctx->pitch);
- ctx->flags = WF_RELATIVE;
- return ctx;
-}
-
-void gui_sync_button(struct element *elem)
-{
- assert(elem->type == GUI_TYPE_BUTTON);
- struct element_button *button = elem->data;
-
- if (!elem->ctx) {
- elem->ctx =
- gui_get_context(button->x, button->y,
- strlen(button->text) * gfx_font_width(button->font_type),
- gfx_font_height(button->font_type));
- }
-
- gfx_fill(elem->ctx, button->color_bg);
- gfx_write(elem->ctx, 0, 0, button->font_type, button->color_fg, button->text);
-}
-
-void gui_sync_label(struct element *elem)
-{
- assert(elem->type == GUI_TYPE_LABEL);
- struct element_label *label = elem->data;
-
- if (!elem->ctx) {
- elem->ctx = gui_get_context(label->x, label->y,
- strlen(label->text) * gfx_font_width(label->font_type),
- gfx_font_height(label->font_type));
- }
-
- gfx_fill(elem->ctx, label->color_bg);
- gfx_write(elem->ctx, 0, 0, label->font_type, label->color_fg, label->text);
-}
-
-void gui_sync_text_box(struct element *elem)
-{
- assert(elem->type == GUI_TYPE_TEXT_BOX);
- struct element_text_box *text_box = elem->data;
-
- int abs_x = absolute_x_off(elem) + text_box->x;
- int abs_y = absolute_y_off(elem) + text_box->y;
- if (!elem->ctx ||
- PERC(elem->parent->ctx->width, text_box->width) - abs_x != elem->ctx->width ||
- PERC(elem->parent->ctx->height, text_box->height) - abs_y != elem->ctx->height) {
- free_context(elem->ctx);
- elem->ctx =
- gui_get_context(text_box->x, text_box->y,
- PERC(elem->parent->ctx->width, text_box->width) - abs_x,
- PERC(elem->parent->ctx->height, text_box->height) - abs_y);
- }
-
- gfx_fill(elem->ctx, text_box->color_bg);
- gfx_write(elem->ctx, 0, 0, text_box->font_type, text_box->color_fg, text_box->text);
-}
-
-void gui_sync_text_input(struct element *elem)
-{
- assert(elem->type == GUI_TYPE_TEXT_INPUT);
- struct element_text_input *text_input = elem->data;
-
- int abs_x = absolute_x_off(elem) + text_input->x;
- if (!elem->ctx ||
- PERC(elem->parent->ctx->width, text_input->width) - abs_x != elem->ctx->width ||
- (u32)gfx_font_height(text_input->font_type) != elem->ctx->height) {
- free_context(elem->ctx);
- elem->ctx =
- gui_get_context(text_input->x, text_input->y,
- PERC(elem->parent->ctx->width, text_input->width) - abs_x,
- gfx_font_height(text_input->font_type));
- }
-
- gfx_fill(elem->ctx, text_input->color_bg);
- gfx_write(elem->ctx, 0, 0, text_input->font_type, text_input->color_fg, text_input->text);
-}
-
-void gui_sync_container(struct element *elem)
-{
- assert(elem->type == GUI_TYPE_CONTAINER);
- struct element_container *container = elem->data;
-
- int abs_x = absolute_x_off(elem) + container->x;
- int abs_y = absolute_y_off(elem) + container->y;
- if (!elem->ctx ||
- PERC(elem->parent->ctx->width, container->width) - abs_x != elem->ctx->width ||
- PERC(elem->parent->ctx->height, container->height) - abs_y != elem->ctx->height) {
- free_context(elem->ctx);
- elem->ctx =
- gui_get_context(container->x, container->y,
- PERC(elem->parent->ctx->width, container->width) - abs_x,
- PERC(elem->parent->ctx->height, container->height) - abs_y);
- }
-
- gfx_fill(elem->ctx, container->color_bg);
- // TODO: Handle container flags
-}
-
-void gui_only_sync(struct element *elem)
-{
- switch (elem->type) {
- case GUI_TYPE_BUTTON:
- gui_sync_button(elem);
- break;
- case GUI_TYPE_LABEL:
- gui_sync_label(elem);
- break;
- case GUI_TYPE_TEXT_BOX:
- gui_sync_text_box(elem);
- break;
- case GUI_TYPE_TEXT_INPUT:
- gui_sync_text_input(elem);
- break;
- case GUI_TYPE_CONTAINER:
- gui_sync_container(elem);
- break;
- default:
- break;
- }
-}
-
-void gui_sync(struct element *elem)
-{
- gui_only_sync(elem);
- merge_elements(get_root(elem->window_id));
- gfx_redraw_focused();
-}
-
-void gui_sync_recursive(struct element *container)
-{
- if (!container || !container->childs || !container->childs->head)
- return;
-
- struct node *iterator = container->childs->head;
- while (iterator != NULL) {
- gui_only_sync(iterator->data);
- gui_sync_recursive(iterator->data);
- iterator = iterator->next;
- }
-}
-
-void gui_sync_window(u32 window_id)
-{
- struct element *root = get_root(window_id);
-
- if (!root || !root->childs || !root->childs->head)
- return;
-
- struct node *iterator = root->childs->head;
- while (iterator != NULL) {
- gui_only_sync(iterator->data);
- gui_sync_recursive(iterator->data);
- iterator = iterator->next;
- }
-
- merge_elements(root);
- gfx_redraw_focused();
-}
-
-struct element *gui_add_button(struct element *container, int x, int y, enum font_type font_type,
- const char *text, u32 color_bg, u32 color_fg)
-{
- if (!container || !container->childs || !gfx_resolve_font(font_type))
- return NULL;
-
- struct element *button = malloc(sizeof(*button));
- button->type = GUI_TYPE_BUTTON;
- button->window_id = container->window_id;
- button->ctx = NULL;
- button->parent = container;
- button->childs = list_new();
- button->data = malloc(sizeof(struct element_button));
- ((struct element_button *)button->data)->x = x;
- ((struct element_button *)button->data)->y = y;
- ((struct element_button *)button->data)->text = strdup(text);
- ((struct element_button *)button->data)->color_fg = color_fg;
- ((struct element_button *)button->data)->color_bg = color_bg;
- ((struct element_button *)button->data)->font_type = font_type;
-
- list_add(container->childs, button);
- gui_sync(button);
-
- return button;
-}
-
-struct element *gui_add_label(struct element *container, int x, int y, enum font_type font_type,
- const char *text, u32 color_bg, u32 color_fg)
-{
- if (!container || !container->childs || !gfx_resolve_font(font_type))
- return NULL;
-
- struct element *label = malloc(sizeof(*label));
- label->type = GUI_TYPE_LABEL;
- label->window_id = container->window_id;
- label->ctx = NULL;
- label->parent = container;
- label->childs = list_new();
- label->data = malloc(sizeof(struct element_label));
- ((struct element_label *)label->data)->x = x;
- ((struct element_label *)label->data)->y = y;
- ((struct element_label *)label->data)->text = strdup(text);
- ((struct element_label *)label->data)->color_fg = color_fg;
- ((struct element_label *)label->data)->color_bg = color_bg;
- ((struct element_label *)label->data)->font_type = font_type;
-
- list_add(container->childs, label);
- gui_sync(label);
-
- return label;
-}
-
-struct element *gui_add_text_box(struct element *container, int x, int y, u32 width, u32 height,
- enum font_type font_type, const char *text, u32 color_bg,
- u32 color_fg)
-{
- if (!container || !container->childs || !gfx_resolve_font(font_type))
- return NULL;
-
- struct element *text_box = malloc(sizeof(*text_box));
- text_box->type = GUI_TYPE_TEXT_BOX;
- text_box->window_id = container->window_id;
- text_box->ctx = NULL;
- text_box->parent = container;
- text_box->childs = list_new();
- text_box->data = malloc(sizeof(struct element_text_box));
- ((struct element_text_box *)text_box->data)->x = x;
- ((struct element_text_box *)text_box->data)->y = y;
- ((struct element_text_box *)text_box->data)->width = width;
- ((struct element_text_box *)text_box->data)->height = height;
- ((struct element_text_box *)text_box->data)->text = strdup(text);
- ((struct element_text_box *)text_box->data)->color_fg = color_fg;
- ((struct element_text_box *)text_box->data)->color_bg = color_bg;
- ((struct element_text_box *)text_box->data)->font_type = font_type;
-
- list_add(container->childs, text_box);
- gui_sync(text_box);
-
- return text_box;
-}
-
-struct element *gui_add_text_input(struct element *container, int x, int y, u32 width,
- enum font_type font_type, u32 color_bg, u32 color_fg)
-{
- if (!container || !container->childs || !gfx_resolve_font(font_type))
- return NULL;
-
- struct element *text_input = malloc(sizeof(*text_input));
- text_input->type = GUI_TYPE_TEXT_INPUT;
- text_input->window_id = container->window_id;
- text_input->ctx = NULL;
- text_input->parent = container;
- text_input->childs = list_new();
- text_input->data = malloc(sizeof(struct element_text_input));
- ((struct element_text_input *)text_input->data)->x = x;
- ((struct element_text_input *)text_input->data)->y = y;
- ((struct element_text_input *)text_input->data)->width = width;
- ((struct element_text_input *)text_input->data)->text[0] = '\0';
- ((struct element_text_input *)text_input->data)->color_fg = color_fg;
- ((struct element_text_input *)text_input->data)->color_bg = color_bg;
- ((struct element_text_input *)text_input->data)->font_type = font_type;
-
- list_add(container->childs, text_input);
- gui_sync(text_input);
-
- return text_input;
-}
-
-struct element *gui_add_container(struct element *container, int x, int y, u32 width, u32 height,
- u32 color_bg)
-{
- if (!container || !container->childs)
- return NULL;
-
- struct element *new_container = malloc(sizeof(*new_container));
- new_container->type = GUI_TYPE_CONTAINER;
- new_container->window_id = container->window_id;
- new_container->ctx = NULL;
- new_container->parent = container;
- new_container->childs = list_new();
- new_container->data = malloc(sizeof(struct element_container));
- ((struct element_container *)new_container->data)->x = x;
- ((struct element_container *)new_container->data)->y = y;
- ((struct element_container *)new_container->data)->width = width;
- ((struct element_container *)new_container->data)->height = height;
- ((struct element_container *)new_container->data)->color_bg = color_bg;
- ((struct element_container *)new_container->data)->flags = 0;
-
- list_add(container->childs, new_container);
- gui_sync(new_container);
-
- return new_container;
-}
-
-void gui_remove_childs(struct element *elem)
-{
- remove_childs(elem);
- elem->childs = list_new();
- gui_sync(elem);
-}
-
-void gui_remove_element(struct element *elem)
-{
- if (!elem)
- return;
-
- u32 id = elem->window_id;
- struct element *root = get_root(id);
- u8 is_root = root == elem;
- remove_element(elem);
- elem = NULL;
- if (!is_root) {
- merge_elements(get_root(id));
- gfx_redraw_focused();
- }
-}
-
-// TODO: Split into small functions
-void gui_event_loop(struct element *container)
-{
- if (!container)
- return;
-
struct message msg = { 0 };
- struct element *focused = NULL;
- while (1) {
- assert(msg_receive(&msg) > 0);
-
- switch (msg.type) {
- case GUI_KILL: {
- remove_all();
- exit(0);
- }
- case GUI_MOUSE: {
- struct gui_event_mouse *event = msg.data;
- focused = element_at(container, event->x, event->y);
- if (focused && focused->event.on_click && event->but1)
- focused->event.on_click(event, focused);
- break;
- }
- case GUI_KEYBOARD: {
- struct gui_event_keyboard *event = msg.data;
-
- if (focused && focused->type == GUI_TYPE_TEXT_INPUT && event->press) {
- char *s = ((struct element_text_input *)focused->data)->text;
- u32 l = strlen(s);
- if (event->ch >= ' ') {
- if (l >= MAX_INPUT_LENGTH)
- continue;
- s[l] = event->ch;
- s[l + 1] = '\0';
- gui_sync(focused);
- } else if (event->scancode == KEY_BACKSPACE && l > 0) {
- s[l - 1] = '\0';
- gui_sync(focused);
- }
- }
-
- if (focused && focused->event.on_submit && event->press &&
- event->scancode == KEY_ENTER) {
- focused->event.on_submit(event, focused);
- // Clear!
- char *t = ((struct element_text_input *)focused->data)->text;
- memset(t, 0, strlen(t));
- gui_sync(focused);
- }
-
- if (focused && focused->event.on_key && event->press && event->ch)
- focused->event.on_key(event, focused);
-
- break;
- }
- case GUI_RESIZE: {
- struct gui_event_resize *event = msg.data;
- struct element *root = get_root(container->window_id);
- root->ctx = event->new_ctx;
- gui_sync_window(container->window_id);
- break;
- }
- default: {
- log("Unknown GUI request %d\n", msg.type);
- }
- }
- }
-
- exit(1);
+ if (msg_send(pidof(WM_PATH), GUI_NEW_WINDOW, win) > 0 && msg_receive(&msg) > 0 &&
+ msg.type == (GUI_NEW_WINDOW | MSG_SUCCESS))
+ return win->id;
+ return -1;
}
-struct element *gui_init(const char *title, u32 width, u32 height, u32 color_bg)
+s32 gui_redraw_window(u32 id)
{
- if (window_count != 0)
- return NULL;
-
- // TODO: Add center flag
- struct window *win =
- new_window(title, 30, 30, width + BORDER * 2, height + BORDER * 2, WF_DEFAULT);
- if (!win)
- return NULL;
-
- gfx_fill(win->ctx, color_bg);
-
- struct element *container = malloc(sizeof(*container));
- container->type = GUI_TYPE_ROOT;
- container->window_id = win->id;
- container->ctx = win->ctx;
- container->childs = list_new();
- container->data = NULL;
- list_add(win->childs, container);
-
- struct element *root = gui_add_container(container, BORDER, BORDER, 100, 100, COLOR_BLACK);
- if (!root)
- return NULL;
-
- return root;
+ struct message msg = { 0 };
+ if (msg_send(pidof(WM_PATH), GUI_REDRAW_WINDOW, &id) > 0 && msg_receive(&msg) > 0 &&
+ msg.type == (GUI_REDRAW_WINDOW | MSG_SUCCESS))
+ return id;
+ return -1;
}