// MIT License, Copyright (c) 2020 Marvin Borner // Mostly GFX function wrappers // TODO: Reduce code duplication #include #include #include #include #include #include #include #include #include #include // TODO: Use list (and add index-based access) #define MAX_WINDOWS 10 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) { if (window_count + 1 >= MAX_WINDOWS) return NULL; 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 remove_childs(struct element *elem); static void remove_element(struct element *elem) { if (!elem) return; remove_childs(elem); free(elem->ctx->fb); elem->ctx->fb = NULL; free(elem->ctx); elem->ctx = NULL; 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 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; } void gui_sync_button(struct element *elem) { struct element_button *button = elem->data; 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) { struct element_label *label = elem->data; 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_input(struct element *elem) { struct element_text_input *text_input = elem->data; 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) { struct element_container *container = elem->data; gfx_fill(elem->ctx, container->color_bg); // TODO: Handle container flags } struct element *gui_add_button(struct element *container, int x, int y, enum font_type font_type, 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 = malloc(sizeof(*button->ctx)); button->ctx->x = x; button->ctx->y = y; button->ctx->width = strlen(text) * gfx_font_width(font_type); button->ctx->height = gfx_font_height(font_type); button->ctx->flags = WF_RELATIVE; button->childs = list_new(); button->data = malloc(sizeof(struct element_button)); ((struct element_button *)button->data)->text = 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; gfx_new_ctx(button->ctx); list_add(container->childs, button); gui_sync_button(button); merge_elements(get_root(container->window_id)); gfx_redraw_focused(); return button; } struct element *gui_add_label(struct element *container, int x, int y, enum font_type font_type, 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 = malloc(sizeof(*label->ctx)); label->ctx->x = x; label->ctx->y = y; label->ctx->width = strlen(text) * gfx_font_width(font_type); label->ctx->height = gfx_font_height(font_type); label->ctx->flags = WF_RELATIVE; label->childs = list_new(); label->data = malloc(sizeof(struct element_label)); ((struct element_label *)label->data)->text = 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; gfx_new_ctx(label->ctx); list_add(container->childs, label); gui_sync_label(label); merge_elements(get_root(container->window_id)); gfx_redraw_focused(); return label; } 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 = malloc(sizeof(*text_input->ctx)); text_input->ctx->x = x; text_input->ctx->y = y; text_input->ctx->width = width; text_input->ctx->height = gfx_font_height(font_type); text_input->ctx->flags = WF_RELATIVE; text_input->childs = list_new(); text_input->data = malloc(sizeof(struct element_text_input)); ((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; gfx_new_ctx(text_input->ctx); list_add(container->childs, text_input); gui_sync_text_input(text_input); merge_elements(get_root(container->window_id)); gfx_redraw_focused(); 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 = malloc(sizeof(*new_container->ctx)); new_container->ctx->x = x; new_container->ctx->y = y; new_container->ctx->width = width; new_container->ctx->height = height; new_container->ctx->flags = WF_RELATIVE; new_container->childs = list_new(); new_container->data = malloc(sizeof(struct element_container)); ((struct element_container *)new_container->data)->color_bg = color_bg; ((struct element_container *)new_container->data)->flags = 0; gfx_new_ctx(new_container->ctx); list_add(container->childs, new_container); gui_sync_container(new_container); merge_elements(get_root(container->window_id)); gfx_redraw_focused(); return new_container; } void gui_remove_childs(struct element *elem) { remove_childs(elem); elem->childs = list_new(); gui_sync_container(elem); merge_elements(get_root(elem->window_id)); gfx_redraw_focused(); } 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; struct element *focused = NULL; while (1) { if (!(msg = msg_receive())) { yield(); continue; } switch (msg->type) { 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_text_input(focused); merge_elements(get_root(focused->window_id)); gfx_redraw_focused(); } else if (event->scancode == KEY_BACKSPACE && l > 0) { s[l - 1] = '\0'; gui_sync_text_input(focused); merge_elements(get_root(focused->window_id)); gfx_redraw_focused(); } } if (focused && focused->event.on_submit && event->press && event->scancode == KEY_ENTER) { focused->event.on_submit(event, focused); // Clear! ((struct element_text_input *)focused->data)->text[0] = '\0'; gui_sync_text_input(focused); merge_elements(get_root(focused->window_id)); gfx_redraw_focused(); } if (focused && focused->event.on_key && event->press && event->ch) focused->event.on_key(event, focused); break; } } } } struct element *gui_init(const char *title, u32 width, u32 height, u32 color_bg) { if (window_count != 0) return NULL; // TODO: Add center flag struct window *win = new_window(title, 30, 30, width, height, 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); return container; }