aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--apps/window.c19
-rw-r--r--apps/wm.c78
-rw-r--r--libgui/gui.c61
-rw-r--r--libgui/inc/gfx.h12
-rw-r--r--libgui/inc/gui.h22
5 files changed, 134 insertions, 58 deletions
diff --git a/apps/window.c b/apps/window.c
index 601f187..ffd860d 100644
--- a/apps/window.c
+++ b/apps/window.c
@@ -8,19 +8,22 @@
#include <print.h>
#include <str.h>
+void on_click()
+{
+ print("CLICK!\n");
+}
+
int main()
{
print("[test context loaded]\n");
struct element *container = gui_init("test", 0, 0);
- gui_add_button(container, 10, 10, 100, 20, "hallo", COLOR_RED);
+ struct element_button *button =
+ gui_add_button(container, 10, 10, 100, 20, "hallo", COLOR_RED);
+
+ button->on_click = on_click;
+
+ gui_event_loop(container);
- struct message *msg;
- while (1) {
- if (!(msg = msg_receive())) {
- yield();
- continue;
- }
- }
return 0;
}
diff --git a/apps/wm.c b/apps/wm.c
index 0ce9a18..bbe2c1f 100644
--- a/apps/wm.c
+++ b/apps/wm.c
@@ -84,7 +84,34 @@ static void redraw_all()
memcpy(direct.fb, exchange.fb, exchange.pitch * exchange.height);
}
-// TODO: Send relative mouse position event to focused context
+#define SHIFT_PRESSED 1 << 0
+#define ALT_PRESSED 1 << 1
+#define CTRL_PRESSED 1 << 2
+static u32 special_keys_pressed;
+static void handle_keyboard(struct event_keyboard *event)
+{
+ if (event->magic != KEYBOARD_MAGIC || !focused)
+ return;
+
+ if (event->scancode == KEY_LEFTSHIFT || event->scancode == KEY_RIGHTSHIFT)
+ special_keys_pressed ^= SHIFT_PRESSED;
+ else if (event->scancode == KEY_LEFTALT || event->scancode == KEY_RIGHTALT)
+ special_keys_pressed ^= ALT_PRESSED;
+ else if (event->scancode == KEY_LEFTCTRL || event->scancode == KEY_RIGHTCTRL)
+ special_keys_pressed ^= CTRL_PRESSED;
+
+ struct gui_event_keyboard *msg = malloc(sizeof(*msg));
+
+ if (special_keys_pressed & SHIFT_PRESSED)
+ msg->ch = keymap->shift_map[event->scancode];
+ else
+ msg->ch = keymap->map[event->scancode];
+
+ msg->press = event->press;
+ msg->scancode = event->scancode;
+ msg_send(focused->pid, GUI_KEYBOARD, msg);
+}
+
static int mouse_skip = 0;
static int mouse_pressed[3] = { 0 };
static void handle_mouse(struct event_mouse *event)
@@ -109,12 +136,14 @@ static void handle_mouse(struct event_mouse *event)
// Restore cursor buffer backup
gfx_copy(&direct, &exchange, cursor.x, cursor.y, cursor.width, cursor.height);
+ int mod_pressed = special_keys_pressed & ALT_PRESSED;
+
// Context focus
if (!mouse_pressed[0] && !mouse_pressed[1])
focused = context_at(mouse_x, mouse_y);
// Context position
- if (event->but1 && !mouse_pressed[1]) {
+ if (mod_pressed && event->but1 && !mouse_pressed[1]) {
mouse_pressed[0] = 1;
if (focused && !(focused->flags & WF_NO_DRAG)) {
focused->x = mouse_x;
@@ -124,13 +153,13 @@ static void handle_mouse(struct event_mouse *event)
redraw_all(); // TODO: Function to redraw one context
}
}
- } else if (mouse_pressed[0]) {
+ } else if (mod_pressed && mouse_pressed[0]) {
mouse_pressed[0] = 0;
redraw_all();
}
// Context size
- if (event->but2 && !mouse_pressed[0]) {
+ if (mod_pressed && event->but2 && !mouse_pressed[0]) {
if (focused && !mouse_pressed[1]) {
mouse_x = focused->x + focused->width;
mouse_y = focused->y + focused->height;
@@ -144,7 +173,7 @@ static void handle_mouse(struct event_mouse *event)
redraw_all(); // TODO: Function to redraw one context
}
mouse_pressed[1] = 1;
- } else if (mouse_pressed[1]) {
+ } else if (mod_pressed && mouse_pressed[1]) {
mouse_pressed[1] = 0;
redraw_all();
}
@@ -153,33 +182,16 @@ static void handle_mouse(struct event_mouse *event)
cursor.y = mouse_y;
gfx_ctx_on_ctx(&direct, &cursor, cursor.x, cursor.y);
mouse_skip++;
-}
-#define SHIFT_PRESSED 1 << 0
-#define ALT_PRESSED 1 << 1
-#define CTRL_PRESSED 1 << 2
-static u32 special_keys_pressed;
-static void handle_keyboard(struct event_keyboard *event)
-{
- if (event->magic != KEYBOARD_MAGIC || !focused)
+ if (!focused)
return;
-
- struct msg_keyboard *msg = malloc(sizeof(*msg));
- if (event->scancode == KEY_LEFTSHIFT || event->scancode == KEY_RIGHTSHIFT)
- special_keys_pressed ^= SHIFT_PRESSED;
- else if (event->scancode == KEY_LEFTALT || event->scancode == KEY_RIGHTALT)
- special_keys_pressed ^= ALT_PRESSED;
- else if (event->scancode == KEY_LEFTCTRL || event->scancode == KEY_RIGHTCTRL)
- special_keys_pressed ^= CTRL_PRESSED;
-
- if (special_keys_pressed & SHIFT_PRESSED)
- msg->ch = keymap->shift_map[event->scancode];
- else
- msg->ch = keymap->map[event->scancode];
-
- msg->press = event->press;
- msg->scancode = event->scancode;
- msg_send(focused->pid, WM_KEYBOARD, msg);
+ struct gui_event_mouse *msg = malloc(sizeof(*msg));
+ msg->x = mouse_x - focused->x;
+ msg->y = mouse_y - focused->y;
+ msg->but1 = event->but1;
+ msg->but2 = event->but2;
+ msg->but3 = event->but3;
+ msg_send(focused->pid, GUI_MOUSE, msg);
}
// TODO: Clean this god-function
@@ -224,7 +236,7 @@ int main(int argc, char **argv)
}
switch (msg->type) {
- case WM_NEW_CONTEXT:
+ case GFX_NEW_CONTEXT:
printf("New context for pid %d\n", msg->src);
struct context *ctx = msg->data;
int width = ctx->width ? ctx->width : 1000;
@@ -237,9 +249,9 @@ int main(int argc, char **argv)
if (!(ctx->flags & WF_RELATIVE))
focused = ctx;
redraw_all();
- msg_send(msg->src, WM_NEW_CONTEXT, ctx);
+ msg_send(msg->src, GFX_NEW_CONTEXT, ctx);
break;
- case WM_REDRAW:
+ case GFX_REDRAW:
redraw_all();
break;
case EVENT_MOUSE:
diff --git a/libgui/gui.c b/libgui/gui.c
index 7ea341c..aa5c494 100644
--- a/libgui/gui.c
+++ b/libgui/gui.c
@@ -6,13 +6,15 @@
#include <gui.h>
#include <list.h>
#include <mem.h>
+#include <print.h>
+#include <sys.h>
#define MAX_WINDOWS 10
u32 window_count = 0;
static struct window windows[MAX_WINDOWS] = { 0 };
-struct window *new_window(const char *title, int x, int y, u32 width, u32 height, int flags)
+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;
@@ -37,9 +39,9 @@ struct window *new_window(const char *title, int x, int y, u32 width, u32 height
return win;
}
-void merge_elements(struct element *container)
+static void merge_elements(struct element *container)
{
- if (!container->childs || !container->childs->head)
+ if (!container || !container->childs || !container->childs->head)
return;
struct node *iterator = container->childs->head;
@@ -52,8 +54,26 @@ void merge_elements(struct element *container)
}
}
-struct element *gui_add_button(struct element *container, int x, int y, u32 width, u32 height,
- const char *text, u32 color)
+static struct element *element_at(struct element *container, int x, int y)
+{
+ if (!container || !container->childs || !container->childs->head)
+ return NULL;
+
+ struct element *ret = NULL;
+ struct node *iterator = container->childs->head;
+ while (iterator != NULL) {
+ struct context *ctx = ((struct element *)iterator->data)->ctx;
+ if (ctx != container->ctx && ctx->flags & WF_RELATIVE && x >= ctx->x &&
+ x <= ctx->x + (int)ctx->width && y >= ctx->y && y <= ctx->y + (int)ctx->height)
+ ret = iterator->data;
+ iterator = iterator->next;
+ }
+
+ return ret;
+}
+
+struct element_button *gui_add_button(struct element *container, int x, int y, u32 width,
+ u32 height, const char *text, u32 color)
{
if (!container || !container->childs)
return NULL;
@@ -76,7 +96,36 @@ struct element *gui_add_button(struct element *container, int x, int y, u32 widt
list_add(container->childs, button);
merge_elements(container);
- return button;
+ return button->data;
+}
+
+void gui_event_loop(struct element *container)
+{
+ if (!container)
+ return;
+
+ struct message *msg;
+ while (1) {
+ if (!(msg = msg_receive())) {
+ yield();
+ continue;
+ }
+
+ switch (msg->type) {
+ case GUI_MOUSE: {
+ struct gui_event_mouse *event = msg->data;
+ struct element *elem = element_at(container, event->x, event->y);
+ if (!elem)
+ continue;
+
+ if (elem->type == GUI_TYPE_BUTTON) {
+ struct element_button *button = elem->data;
+ if (event->but1 && button->on_click)
+ button->on_click();
+ }
+ }
+ }
+ }
}
struct element *gui_init(const char *title, u32 width, u32 height)
diff --git a/libgui/inc/gfx.h b/libgui/inc/gfx.h
index fe48ac2..2ce588c 100644
--- a/libgui/inc/gfx.h
+++ b/libgui/inc/gfx.h
@@ -39,7 +39,7 @@
#define WF_NO_RESIZE (1 << 2)
#define WF_RELATIVE (1 << 3)
-enum message_type { WM_NEW_CONTEXT = EVENT_MAX + 1, WM_REDRAW, WM_KEYBOARD };
+enum message_type { GFX_NEW_CONTEXT = EVENT_MAX + 1, GFX_REDRAW };
// Generalized font struct
struct font {
@@ -61,12 +61,6 @@ struct context {
int flags;
};
-struct msg_keyboard {
- char ch;
- int press;
- int scancode;
-};
-
void gfx_write_char(struct context *ctx, int x, int y, u32 c, char ch);
void gfx_write(struct context *ctx, int x, int y, u32 c, char *text);
void gfx_load_image(struct context *ctx, char *path, int x, int y);
@@ -86,7 +80,7 @@ int gfx_font_width();
*/
#define gfx_new_ctx(ctx) \
- (msg_send(2, WM_NEW_CONTEXT, (ctx)), (struct context *)msg_receive_loop()->data)
-#define gfx_redraw() (msg_send(2, WM_REDRAW, NULL)) // TODO: Partial redraw (optimization)
+ (msg_send(2, GFX_NEW_CONTEXT, (ctx)), (struct context *)msg_receive_loop()->data)
+#define gfx_redraw() (msg_send(2, GFX_REDRAW, NULL)) // TODO: Partial redraw (optimization)
#endif
diff --git a/libgui/inc/gui.h b/libgui/inc/gui.h
index ea1e40e..e7573f8 100644
--- a/libgui/inc/gui.h
+++ b/libgui/inc/gui.h
@@ -10,11 +10,14 @@
#define MAX_CHILDS 100
+// TODO: Improve event types (maybe as struct header)
+enum window_event_type { GUI_KEYBOARD = 10, GUI_MOUSE, GUI_MAX };
enum element_type { GUI_TYPE_CONTAINER, GUI_TYPE_BUTTON, GUI_TYPE_TEXTBOX };
struct element_button {
const char *text;
u32 color;
+ void (*on_click)();
};
struct element_textbox {
@@ -37,8 +40,23 @@ struct window {
struct context *ctx;
};
+struct gui_event_keyboard {
+ char ch;
+ int press;
+ int scancode;
+};
+
+struct gui_event_mouse {
+ int x;
+ int y;
+ int but1;
+ int but2;
+ int but3;
+};
+
struct element *gui_init(const char *title, u32 width, u32 height);
-struct element *gui_add_button(struct element *container, int x, int y, u32 width, u32 height,
- const char *text, u32 color);
+void gui_event_loop(struct element *container);
+struct element_button *gui_add_button(struct element *container, int x, int y, u32 width,
+ u32 height, const char *text, u32 color);
#endif