aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--apps/Makefile2
-rw-r--r--apps/chess.c57
-rw-r--r--apps/init.c2
-rw-r--r--apps/wm.c1
-rw-r--r--libs/libgui/gui.c125
-rw-r--r--libs/libgui/gui.h5
6 files changed, 169 insertions, 23 deletions
diff --git a/apps/Makefile b/apps/Makefile
index f2ab790..f689727 100644
--- a/apps/Makefile
+++ b/apps/Makefile
@@ -1,6 +1,6 @@
# MIT License, Copyright (c) 2020 Marvin Borner
-COBJS = init.o idle.o wm.o test.o window.o #mandelbrot.o window.o exec.o files.o test.o cc.o browser.o server.o
+COBJS = init.o idle.o wm.o test.o chess.o #mandelbrot.o window.o exec.o files.o test.o cc.o browser.o server.o
CC = ccache ../cross/opt/bin/i686-elf-gcc
LD = ccache ../cross/opt/bin/i686-elf-ld
OC = ccache ../cross/opt/bin/i686-elf-objcopy
diff --git a/apps/chess.c b/apps/chess.c
new file mode 100644
index 0000000..d339534
--- /dev/null
+++ b/apps/chess.c
@@ -0,0 +1,57 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+
+#include <assert.h>
+#include <libgui/gui.h>
+#include <print.h>
+
+#define SIZE 8
+#define TILE 24
+#define WHITE_STARTS 1
+
+typedef u32 board[SIZE][SIZE];
+
+static u32 win = 0; // Window
+static board tiles = { 0 }; // Matrix
+
+static void mouseclick(u32 widget_id, vec2 pos)
+{
+ UNUSED(pos);
+ /* log("%d: %d %d\n", widget_id, pos.x, pos.y); */
+
+ u32 x = widget_id / SIZE;
+ u32 y = (widget_id % SIZE) - 1;
+ u32 widget = tiles[x][y];
+ assert(gui_fill(win, widget, COLOR_MAGENTA) == EOK);
+ gui_redraw_widget(win, widget);
+}
+
+static void create_board(void)
+{
+ u32 widget;
+ for (u8 x = 0; x < 8; x++) {
+ for (u8 y = 0; y < 8; y++) {
+ widget = gui_new_widget(win, vec2(TILE, TILE), vec2(TILE * x, TILE * y));
+ assert(widget > 0);
+ tiles[x][y] = widget;
+
+ u8 colored = (x + y + 1) % 2 == 0;
+#if !WHITE_STARTS
+ colored = !colored;
+#endif
+
+ assert(gui_fill(win, widget, colored ? COLOR_BLACK : COLOR_WHITE) == EOK);
+ assert(gui_listen_widget(win, widget, GUI_LISTEN_MOUSECLICK,
+ (u32)mouseclick) == EOK);
+ }
+ }
+
+ assert(gui_redraw_window(win) == EOK);
+}
+
+int main(void)
+{
+ assert((win = gui_new_window()) > 0);
+ create_board();
+ gui_loop();
+ return 0;
+}
diff --git a/apps/init.c b/apps/init.c
index f1fcdf6..4ad63f2 100644
--- a/apps/init.c
+++ b/apps/init.c
@@ -10,7 +10,7 @@ int main(int argc, char **argv)
UNUSED(argv);
assert(exec("/bin/wm", "wm", NULL) == 0);
- assert(exec("/bin/window", "test", NULL) == 0);
+ assert(exec("/bin/chess", "chess", NULL) == 0);
return 0;
}
diff --git a/apps/wm.c b/apps/wm.c
index 52c5932..1fca72e 100644
--- a/apps/wm.c
+++ b/apps/wm.c
@@ -369,6 +369,7 @@ static void handle_event_mouse(struct event_mouse *event)
msg.header.state = MSG_GO_ON;
msg.id = win->id;
msg.pos = vec2_sub(mouse.pos, win->pos);
+ msg.bits.click = event->but1;
msg_send(win->client.pid, GUI_MOUSE, &msg, sizeof(msg));
}
diff --git a/libs/libgui/gui.c b/libs/libgui/gui.c
index f4c4c6c..99b3b5c 100644
--- a/libs/libgui/gui.c
+++ b/libs/libgui/gui.c
@@ -7,7 +7,6 @@
#include <libgui/msg.h>
#include <list.h>
#include <print.h>
-#include <random.h>
#define WM_PATH "/bin/wm"
@@ -18,7 +17,8 @@ struct gui_widget {
struct list *children;
struct {
- void (*mousemove)(vec2 pos);
+ void (*mousemove)(u32 widget_id, vec2 pos);
+ void (*mouseclick)(u32 widget_id, vec2 pos);
} event;
};
@@ -96,7 +96,7 @@ static struct gui_widget *gui_widget_in_win(struct gui_window *win, u32 widget_i
return NULL;
}
-static struct gui_widget *widget_by_id(u32 win_id, u32 widget_id)
+static struct gui_widget *gui_widget_by_id(u32 win_id, u32 widget_id)
{
struct gui_window *win = gui_window_by_id(win_id);
return gui_widget_in_win(win, widget_id);
@@ -108,7 +108,7 @@ static struct gui_widget *widget_by_id(u32 win_id, u32 widget_id)
res gui_fill(u32 win_id, u32 widget_id, u32 c)
{
- struct gui_widget *widget = widget_by_id(win_id, widget_id);
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
if (!widget)
return_errno(ENOENT);
@@ -157,9 +157,12 @@ static res gui_widget_at(u32 win_id, vec2 pos, struct gui_widget *widget)
return_errno(EFAULT);
struct gui_window *win = gui_window_by_id(win_id);
- if (!win || !win->widgets)
+ if (!win)
return_errno(ENOENT);
+ if (!win->widgets)
+ return_errno(EOK);
+
struct gui_widget *ret = NULL;
struct gui_widget sub = { 0 };
@@ -185,6 +188,24 @@ static res gui_widget_at(u32 win_id, vec2 pos, struct gui_widget *widget)
}
}
+static res gui_sync_sub_widgets(struct gui_widget *widget)
+{
+ if (!widget)
+ return_errno(EFAULT);
+
+ if (!widget->children)
+ return_errno(EOK);
+
+ struct node *iterator = widget->children->head;
+ while (iterator) {
+ struct gui_widget *w = iterator->data;
+ gfx_ctx_on_ctx(&widget->ctx, &w->ctx, w->pos);
+ iterator = iterator->next;
+ }
+
+ return_errno(EOK);
+}
+
static res gui_sync_widget(u32 win_id, u32 widget_id)
{
struct gui_window *win = gui_window_by_id(win_id);
@@ -192,22 +213,72 @@ static res gui_sync_widget(u32 win_id, u32 widget_id)
if (!widget)
return_errno(ENOENT);
+ gui_sync_sub_widgets(widget);
gfx_ctx_on_ctx(&win->ctx, &widget->ctx, widget->pos);
return_errno(EOK);
}
+static res gui_sync_widgets(u32 win_id)
+{
+ struct gui_window *win = gui_window_by_id(win_id);
+ if (!win)
+ return_errno(ENOENT);
+
+ if (!win->widgets)
+ return_errno(EOK);
+
+ struct node *iterator = win->widgets->head;
+ while (iterator) {
+ struct gui_widget *widget = iterator->data;
+ gui_sync_sub_widgets(widget);
+ gfx_ctx_on_ctx(&win->ctx, &widget->ctx, widget->pos);
+ iterator = iterator->next;
+ }
+
+ return_errno(EOK);
+}
+
static struct gui_widget *gui_win_main_widget(struct gui_window *win)
{
return win->widgets->head->data;
}
+// TODO: This is very recursive and inefficient -> improve!
+static vec2 gui_offset_widget(struct gui_widget *parent, struct gui_widget *child)
+{
+ if (!parent || !parent->children || !child)
+ return vec2(0, 0);
+
+ vec2 offset = vec2(0, 0);
+
+ struct node *iterator = parent->children->head;
+ while (iterator) {
+ struct gui_widget *w = iterator->data;
+ if (w == child) {
+ offset = vec2_add(offset, w->pos);
+ break;
+ }
+ struct gui_widget *sub = gui_widget_in_widget(w, child->id);
+ if (sub) {
+ offset = vec2_add(offset, w->pos);
+ gui_offset_widget(w, child);
+ break;
+ }
+
+ iterator = iterator->next;
+ }
+
+ return offset;
+}
+
static struct gui_widget *gui_new_plain_widget(vec2 size, vec2 pos, u8 bpp)
{
struct gui_widget *widget = zalloc(sizeof(*widget));
struct context *ctx = zalloc(sizeof(*ctx));
- widget->id = rand();
+ static u32 id = 0;
+ widget->id = id++;
widget->pos = pos;
widget->ctx = *gfx_new_ctx(ctx, size, bpp);
widget->children = list_new();
@@ -217,7 +288,7 @@ static struct gui_widget *gui_new_plain_widget(vec2 size, vec2 pos, u8 bpp)
res gui_add_widget(u32 win_id, u32 widget_id, vec2 size, vec2 pos)
{
- struct gui_widget *parent = widget_by_id(win_id, widget_id);
+ struct gui_widget *parent = gui_widget_by_id(win_id, widget_id);
if (!parent)
return_errno(ENOENT);
@@ -245,13 +316,16 @@ res gui_listen_widget(u32 win_id, u32 widget_id, enum gui_listener listener, u32
if (!func)
return_errno(EFAULT);
- struct gui_widget *widget = widget_by_id(win_id, widget_id);
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
if (!widget)
return_errno(ENOENT);
switch (listener) {
case GUI_LISTEN_MOUSEMOVE:
- widget->event.mousemove = (void (*)(vec2))func;
+ widget->event.mousemove = (void (*)(u32, vec2))func;
+ break;
+ case GUI_LISTEN_MOUSECLICK:
+ widget->event.mouseclick = (void (*)(u32, vec2))func;
break;
default:
return_errno(ENOENT);
@@ -315,11 +389,15 @@ res gui_new_window(void)
res gui_redraw_window(u32 id)
{
+ res ret = gui_sync_widgets(id);
+ if (ret != 0)
+ return_errno(ENOENT);
+
struct message_redraw_window msg = { .id = id, .header.state = MSG_NEED_ANSWER };
if (msg_send(pidof(WM_PATH), GUI_REDRAW_WINDOW, &msg, sizeof(msg)) > 0 &&
msg_receive(&msg, sizeof(msg)) > 0 &&
msg.header.type == (GUI_REDRAW_WINDOW | MSG_SUCCESS))
- return id;
+ return EOK;
return_errno(EINVAL);
}
@@ -328,16 +406,16 @@ res gui_redraw_window(u32 id)
* Message handling
*/
-static res handle_error(const char *op, res code)
+static res gui_handle_error(const char *op, res code)
{
log("GUI error at '%s': %s (%d)\n", op, strerror(code), code);
return code;
}
-static res handle_ping(struct message_ping *msg)
+static res gui_handle_ping(struct message_ping *msg)
{
if (msg->ping != MSG_PING_SEND)
- return handle_error("ping", EINVAL);
+ return gui_handle_error("ping", EINVAL);
msg->header.type |= MSG_SUCCESS;
msg->ping = MSG_PING_RECV;
@@ -346,7 +424,7 @@ static res handle_ping(struct message_ping *msg)
return errno;
}
-static res handle_mouse(struct message_mouse *msg)
+static res gui_handle_mouse(struct message_mouse *msg)
{
if (msg->header.state == MSG_NEED_ANSWER) {
msg_send(msg->header.src, msg->header.type | MSG_SUCCESS, msg, sizeof(*msg));
@@ -359,13 +437,19 @@ static res handle_mouse(struct message_mouse *msg)
if (gui_widget_at(msg->id, msg->pos, &widget) != EOK)
return_errno(EOK);
+ struct gui_window *win = gui_window_by_id(msg->id);
+ vec2 offset = gui_offset_widget(gui_win_main_widget(win), &widget);
+
if (widget.event.mousemove)
- widget.event.mousemove(msg->pos);
+ widget.event.mousemove(widget.id, vec2_sub(msg->pos, offset));
+
+ if (widget.event.mouseclick && msg->bits.click)
+ widget.event.mouseclick(widget.id, vec2_sub(msg->pos, offset));
return_errno(EOK);
}
-static void handle_exit(void)
+static void gui_handle_exit(void)
{
if (!windows)
return;
@@ -387,7 +471,7 @@ static void handle_exit(void)
void gui_loop(void)
{
- atexit(handle_exit);
+ atexit(gui_handle_exit);
if (!windows)
err(1, "Create some windows first\n");
@@ -397,13 +481,14 @@ void gui_loop(void)
struct message_header *head = msg;
switch (head->type) {
case GUI_PING:
- handle_ping(msg);
+ gui_handle_ping(msg);
break;
case GUI_MOUSE:
- handle_mouse(msg);
+ gui_handle_mouse(msg);
break;
default:
- handle_error("loop", EINVAL);
+ // TODO: Fix random unknown msg types
+ gui_handle_error("loop", EINVAL);
}
}
}
diff --git a/libs/libgui/gui.h b/libs/libgui/gui.h
index 511033f..ff475ae 100644
--- a/libs/libgui/gui.h
+++ b/libs/libgui/gui.h
@@ -8,7 +8,10 @@
#include <libgui/gfx.h>
#include <vec.h>
-enum gui_listener { GUI_LISTEN_MOUSEMOVE };
+enum gui_listener {
+ GUI_LISTEN_MOUSEMOVE,
+ GUI_LISTEN_MOUSECLICK,
+};
res gui_new_window(void);
res gui_redraw_window(u32 id);