aboutsummaryrefslogtreecommitdiff
path: root/libs
diff options
context:
space:
mode:
Diffstat (limited to 'libs')
-rw-r--r--libs/libgui/gfx.c85
-rw-r--r--libs/libgui/gfx.h31
-rw-r--r--libs/libgui/gui.c163
-rw-r--r--libs/libgui/gui.h30
-rw-r--r--libs/libgui/msg.h7
5 files changed, 259 insertions, 57 deletions
diff --git a/libs/libgui/gfx.c b/libs/libgui/gfx.c
index d963a1b..fe8cf1e 100644
--- a/libs/libgui/gfx.c
+++ b/libs/libgui/gfx.c
@@ -72,7 +72,7 @@ static void free_fonts(void)
static void write_char(struct context *ctx, vec2 pos, struct font *font, u32 c, char ch)
{
- int bypp = ctx->bpp >> 3;
+ u8 bypp = ctx->bpp >> 3;
char *draw = (char *)&ctx->fb[pos.x * bypp + pos.y * ctx->pitch];
@@ -262,10 +262,20 @@ void gfx_load_wallpaper(struct context *ctx, const char *path)
gfx_load_image(ctx, vec2(0, 0), path);
}
+void gfx_draw_pixel(struct context *ctx, vec2 pos1, u32 c)
+{
+ u8 bypp = ctx->bpp >> 3;
+ u8 *draw = &ctx->fb[pos1.x * bypp + pos1.y * ctx->pitch];
+ draw[0] = GET_BLUE(c);
+ draw[1] = GET_GREEN(c);
+ draw[2] = GET_RED(c);
+ draw[3] = GET_ALPHA(c);
+}
+
void gfx_draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c)
{
assert(pos1.x <= pos2.x && pos1.y <= pos2.y);
- int bypp = ctx->bpp >> 3;
+ u8 bypp = ctx->bpp >> 3;
u8 *draw = &ctx->fb[pos1.x * bypp + pos1.y * ctx->pitch];
for (u32 i = 0; i < pos2.y - pos1.y; i++) {
for (u32 j = 0; j < pos2.x - pos1.x; j++) {
@@ -278,9 +288,54 @@ void gfx_draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c)
}
}
+void gfx_draw_border(struct context *ctx, u32 width, u32 c)
+{
+ if (width <= 0)
+ return;
+
+ u8 bypp = ctx->bpp >> 3;
+ u8 *draw = ctx->fb;
+ for (u32 i = 0; i < ctx->size.y; i++) {
+ for (u32 j = 0; j < ctx->size.x; j++) {
+ if (j <= width - 1 || i <= width - 1 || j - ctx->size.x + width <= width ||
+ i - ctx->size.y + width <= width) {
+ draw[bypp * j + 0] = GET_BLUE(c);
+ draw[bypp * j + 1] = GET_GREEN(c);
+ draw[bypp * j + 2] = GET_RED(c);
+ draw[bypp * j + 3] = GET_ALPHA(c);
+ }
+ }
+ draw += ctx->pitch;
+ }
+}
+
+// Using Bresenham's algorithm
+void gfx_draw_line(struct context *ctx, vec2 pos1, vec2 pos2, u32 scale, u32 c)
+{
+ int dx = ABS(pos2.x - pos1.x), sx = pos1.x < pos2.x ? 1 : -1;
+ int dy = ABS(pos2.y - pos1.y), sy = pos1.y < pos2.y ? 1 : -1;
+ int err = (dx > dy ? dx : -dy) / 2, e2;
+
+ while (1) {
+ gfx_draw_rectangle(ctx, pos1, vec2_add(pos1, vec2(scale, scale)), c);
+ /* gfx_draw_pixel(ctx, pos1, c); */
+ if (pos1.x == pos2.x && pos1.y == pos2.y)
+ break;
+ e2 = err;
+ if (e2 > -dx) {
+ err -= dy;
+ pos1.x += sx;
+ }
+ if (e2 < dy) {
+ err += dx;
+ pos1.y += sy;
+ }
+ }
+}
+
void gfx_copy(struct context *dest, struct context *src, vec2 pos, vec2 size)
{
- int bypp = dest->bpp >> 3;
+ u8 bypp = dest->bpp >> 3;
u8 *srcfb = &src->fb[pos.x * bypp + pos.y * src->pitch];
u8 *destfb = &dest->fb[pos.x * bypp + pos.y * dest->pitch];
for (u32 cy = 0; cy < size.y; cy++) {
@@ -292,7 +347,7 @@ void gfx_copy(struct context *dest, struct context *src, vec2 pos, vec2 size)
// TODO: Support alpha values other than 0x0 and 0xff (blending)
// TODO: Optimize!
-void gfx_ctx_on_ctx(struct context *dest, struct context *src, vec2 pos, u8 alpha)
+HOT void gfx_ctx_on_ctx(struct context *dest, struct context *src, vec2 pos, u8 alpha)
{
// TODO: Some kind of alpha-acknowledging memcpy?
if (!alpha && src->size.x == dest->size.x && src->size.y == dest->size.y) {
@@ -332,28 +387,6 @@ void gfx_fill(struct context *ctx, u32 c)
gfx_draw_rectangle(ctx, vec2(0, 0), vec2(ctx->size.x, ctx->size.y), c);
}
-void gfx_border(struct context *ctx, u32 c, u32 width)
-{
- if (width <= 0)
- return;
-
- int bypp = ctx->bpp >> 3;
- u8 *draw = ctx->fb;
- for (u32 i = 0; i < ctx->size.y; i++) {
- for (u32 j = 0; j < ctx->size.x; j++) {
- if (j <= width - 1 || i <= width - 1 ||
- j - ctx->size.x + width + 1 <= width ||
- i - ctx->size.y + width <= width) {
- draw[bypp * j + 0] = GET_BLUE(c);
- draw[bypp * j + 1] = GET_GREEN(c);
- draw[bypp * j + 2] = GET_RED(c);
- draw[bypp * j + 3] = GET_ALPHA(c);
- }
- }
- draw += ctx->pitch;
- }
-}
-
int gfx_font_height(enum font_type font_type)
{
struct font *font = gfx_resolve_font(font_type);
diff --git a/libs/libgui/gfx.h b/libs/libgui/gfx.h
index eb98578..7c069f4 100644
--- a/libs/libgui/gfx.h
+++ b/libs/libgui/gfx.h
@@ -73,23 +73,46 @@ struct context {
};
struct context *gfx_new_ctx(struct context *ctx, vec2 size, u8 bpp) NONNULL;
+
+/**
+ * Text stuff
+ */
+
struct font *gfx_resolve_font(enum font_type font_type);
void gfx_write_char(struct context *ctx, vec2 pos, enum font_type font_type, u32 c,
char ch) NONNULL;
void gfx_write(struct context *ctx, vec2 pos, enum font_type font_type, u32 c,
const char *text) NONNULL;
+
+int gfx_font_height(enum font_type);
+int gfx_font_width(enum font_type);
+
+/**
+ * Image loading
+ */
+
void gfx_load_image(struct context *ctx, vec2 pos, const char *path) NONNULL;
void gfx_load_image_filter(struct context *ctx, vec2 pos, enum gfx_filter filter,
const char *path) NONNULL;
void gfx_load_wallpaper(struct context *ctx, const char *path) NONNULL;
+
+/**
+ * Context copying
+ */
+
void gfx_copy(struct context *dest, struct context *src, vec2 pos, vec2 size) NONNULL;
void gfx_ctx_on_ctx(struct context *dest, struct context *src, vec2 pos, u8 alpha) NONNULL;
+
+/**
+ * Drawing functions
+ */
+
+void gfx_draw_pixel(struct context *ctx, vec2 pos1, u32 c);
void gfx_draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c) NONNULL;
+void gfx_draw_line(struct context *ctx, vec2 pos1, vec2 pos2, u32 scale, u32 c);
+
void gfx_clear(struct context *ctx);
void gfx_fill(struct context *ctx, u32 c) NONNULL;
-void gfx_border(struct context *ctx, u32 c, u32 width) NONNULL;
-
-int gfx_font_height(enum font_type);
-int gfx_font_width(enum font_type);
+void gfx_draw_border(struct context *ctx, u32 width, u32 c) NONNULL;
#endif
diff --git a/libs/libgui/gui.c b/libs/libgui/gui.c
index 023bc98..859fb5a 100644
--- a/libs/libgui/gui.c
+++ b/libs/libgui/gui.c
@@ -17,13 +17,14 @@ struct gui_widget {
struct list *children;
struct {
- void (*mousemove)(u32 widget_id, vec2 pos);
- void (*mouseclick)(u32 widget_id, vec2 pos);
+ void (*mousemove)(struct gui_event_mouse *event);
+ void (*mouseclick)(struct gui_event_mouse *event);
} event;
};
struct gui_window {
u32 id;
+ vec2 off; // fb offset
vec2 pos;
struct context ctx;
struct list *widgets;
@@ -190,6 +191,50 @@ void gui_load_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, v
gui_load_image_filter(win_id, widget_id, layer, pos, size, GFX_FILTER_NONE, path);
}
+void gui_draw_rectangle(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos1, vec2 pos2,
+ u32 c)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ gui_error(ENOENT);
+
+ if (layer == GUI_LAYER_BG)
+ gfx_draw_rectangle(&widget->bg, pos1, pos2, c);
+ else if (layer == GUI_LAYER_FG)
+ gfx_draw_rectangle(&widget->fg, pos1, pos2, c);
+ else
+ gui_error(EINVAL);
+}
+
+void gui_draw_border(u32 win_id, u32 widget_id, enum gui_layer layer, u32 width, u32 c)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ gui_error(ENOENT);
+
+ if (layer == GUI_LAYER_BG)
+ gfx_draw_border(&widget->bg, width, c);
+ else if (layer == GUI_LAYER_FG)
+ gfx_draw_border(&widget->fg, width, c);
+ else
+ gui_error(EINVAL);
+}
+
+void gui_draw_line(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos1, vec2 pos2, u32 scale,
+ u32 c)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ gui_error(ENOENT);
+
+ if (layer == GUI_LAYER_BG)
+ gfx_draw_line(&widget->bg, pos1, pos2, scale, c);
+ else if (layer == GUI_LAYER_FG)
+ gfx_draw_line(&widget->fg, pos1, pos2, scale, c);
+ else
+ gui_error(EINVAL);
+}
+
/**
* Widgets
*/
@@ -266,6 +311,7 @@ static void gui_sync_sub_widgets(struct gui_widget *widget)
struct node *iterator = widget->children->head;
while (iterator) {
struct gui_widget *w = iterator->data;
+ gui_sync_sub_widgets(w);
gfx_ctx_on_ctx(&widget->bg, &w->bg, w->pos, GFX_NON_ALPHA);
gfx_ctx_on_ctx(&widget->fg, &w->fg, w->pos, GFX_NON_ALPHA);
iterator = iterator->next;
@@ -346,14 +392,14 @@ static vec2 gui_offset_widget(struct gui_widget *parent, struct gui_widget *chil
static struct gui_widget *gui_new_plain_widget(vec2 pos, vec2 size, u8 bpp)
{
struct gui_widget *widget = zalloc(sizeof(*widget));
- struct context *bg = zalloc(sizeof(*bg));
- struct context *fg = zalloc(sizeof(*fg));
+ struct context bg;
+ struct context fg;
static u32 id = 0;
widget->id = id++;
widget->pos = pos;
- widget->bg = *gfx_new_ctx(bg, size, bpp);
- widget->fg = *gfx_new_ctx(fg, size, bpp);
+ widget->bg = *gfx_new_ctx(&bg, size, bpp);
+ widget->fg = *gfx_new_ctx(&fg, size, bpp);
widget->children = list_new();
return widget;
@@ -371,6 +417,41 @@ void gui_add_widget(u32 *widget, u32 win_id, u32 widget_id, vec2 pos, vec2 size)
*widget = child->id;
}
+static void gui_destroy_widget(u32 win_id, u32 widget_id)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ return;
+
+ struct node *iterator = widget->children->head;
+ while (iterator) {
+ struct gui_widget *sub = iterator->data;
+ gui_destroy_widget(win_id, sub->id);
+ iterator = iterator->next;
+ }
+
+ list_destroy(widget->children);
+ free(widget->bg.fb);
+ free(widget->fg.fb);
+ free(widget);
+}
+
+static void gui_destroy_widgets(u32 win_id)
+{
+ struct gui_window *win = gui_window_by_id(win_id);
+
+ if (!win->widgets)
+ return;
+
+ struct node *iterator = win->widgets->head;
+ while (iterator) {
+ struct gui_widget *widget = iterator->data;
+ gui_destroy_widget(win_id, widget->id);
+ iterator = iterator->next;
+ }
+ list_destroy(win->widgets);
+}
+
void gui_new_widget(u32 *widget, u32 win_id, vec2 pos, vec2 size)
{
struct gui_window *win = gui_window_by_id(win_id);
@@ -391,10 +472,10 @@ void gui_listen_widget(u32 win_id, u32 widget_id, enum gui_listener listener, u3
switch (listener) {
case GUI_LISTEN_MOUSEMOVE:
- widget->event.mousemove = (void (*)(u32, vec2))func;
+ widget->event.mousemove = (void (*)(struct gui_event_mouse *))func;
break;
case GUI_LISTEN_MOUSECLICK:
- widget->event.mouseclick = (void (*)(u32, vec2))func;
+ widget->event.mouseclick = (void (*)(struct gui_event_mouse *))func;
break;
default:
gui_error(ENOENT);
@@ -404,7 +485,7 @@ void gui_listen_widget(u32 win_id, u32 widget_id, enum gui_listener listener, u3
void gui_redraw_widget(u32 win_id, u32 widget_id)
{
gui_sync_widget(win_id, widget_id);
- gui_redraw_window(win_id);
+ gui_redraw_window_only(win_id);
}
/**
@@ -467,6 +548,7 @@ void gui_new_custom_window(u32 *id, vec2 pos, vec2 size)
if (msg_send(GUI_NEW_WINDOW, &msg, sizeof(msg)) > 0 && msg_receive(&msg, sizeof(msg)) > 0 &&
msg.header.type == (GUI_NEW_WINDOW | MSG_SUCCESS)) {
win->id = msg.id;
+ win->off = msg.off;
win->ctx = msg.ctx;
win->pos = msg.pos;
u32 buf_size;
@@ -497,10 +579,8 @@ INLINE void gui_new_window(u32 *id)
gui_new_custom_window(id, vec2(0, 0), vec2(0, 0));
}
-void gui_redraw_window(u32 id)
+void gui_redraw_window_only(u32 id)
{
- gui_sync_widgets(id);
-
struct message_redraw_window msg = { .id = id, .header.state = MSG_NEED_ANSWER };
gui_connect_wm();
if (msg_send(GUI_REDRAW_WINDOW, &msg, sizeof(msg)) > 0 &&
@@ -511,6 +591,37 @@ void gui_redraw_window(u32 id)
gui_error(EINVAL);
}
+void gui_redraw_window(u32 id)
+{
+ gui_sync_widgets(id);
+ gui_redraw_window_only(id);
+}
+
+static void gui_destroy_window(u32 id)
+{
+ gui_destroy_widgets(id);
+
+ struct gui_window *win = gui_window_by_id(id);
+ u8 *fb = win->ctx.fb - (win->off.y * win->ctx.pitch);
+ assert(sys_free(fb) == EOK);
+ free(win);
+}
+
+static void gui_destroy_all(void)
+{
+ struct node *iterator = windows->head;
+ while (iterator) {
+ struct gui_window *win = iterator->data;
+ struct message_destroy_window msg = { .id = win->id };
+ gui_destroy_window(win->id);
+ gui_connect_wm();
+ msg_send(GUI_DESTROY_WINDOW, &msg, sizeof(msg));
+ iterator = iterator->next;
+ }
+
+ list_destroy(windows);
+}
+
/**
* Message handling
*/
@@ -542,12 +653,23 @@ static void gui_handle_mouse(struct message_mouse *msg)
struct gui_window *win = gui_window_by_id(msg->id);
vec2 offset = gui_offset_widget(gui_main_widget(win), &widget);
+ vec2 pos = vec2_sub(msg->pos, offset);
+
+ struct gui_event_mouse event = {
+ .win = win->id,
+ .widget = widget.id,
+ .pos = pos,
+ .scroll = msg->scroll,
+ .but.left = msg->but.left,
+ .but.right = msg->but.right,
+ .but.middle = msg->but.middle,
+ };
if (widget.event.mousemove)
- widget.event.mousemove(widget.id, vec2_sub(msg->pos, offset));
+ widget.event.mousemove(&event);
- if (widget.event.mouseclick && msg->bits.click)
- widget.event.mouseclick(widget.id, vec2_sub(msg->pos, offset));
+ if (widget.event.mouseclick && msg->but.left)
+ widget.event.mouseclick(&event);
}
static void gui_handle_exit(void)
@@ -555,16 +677,7 @@ static void gui_handle_exit(void)
if (!windows)
return;
- struct node *iterator = windows->head;
- while (iterator) {
- struct gui_window *win = iterator->data;
- struct message_destroy_window msg = { .id = win->id };
- gui_connect_wm();
- msg_send(GUI_DESTROY_WINDOW, &msg, sizeof(msg));
- iterator = iterator->next;
- }
-
- list_destroy(windows);
+ gui_destroy_all();
}
/**
diff --git a/libs/libgui/gui.h b/libs/libgui/gui.h
index c7dc24f..66c6733 100644
--- a/libs/libgui/gui.h
+++ b/libs/libgui/gui.h
@@ -18,9 +18,30 @@ enum gui_layer {
GUI_LAYER_FG,
};
+struct gui_event_mouse {
+ u32 win;
+ u32 widget;
+ vec2 pos;
+ s8 scroll; // Dir: -1 neg, +1 pos
+ struct {
+ u8 left : 1;
+ u8 right : 1;
+ u8 middle : 1;
+ } but;
+};
+
+/**
+ * Window operations
+ */
+
void gui_new_custom_window(u32 *id, vec2 pos, vec2 size);
void gui_new_window(u32 *id);
void gui_redraw_window(u32 id);
+void gui_redraw_window_only(u32 id); // Without widgets
+
+/**
+ * GFX wrappers
+ */
void gui_clear(u32 win_id, u32 widget_id, enum gui_layer layer);
void gui_fill(u32 win_id, u32 widget_id, enum gui_layer layer, u32 c);
@@ -30,6 +51,15 @@ void gui_load_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, v
const char *path) NONNULL;
void gui_load_image_filter(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size,
enum gfx_filter filter, const char *path) NONNULL;
+void gui_draw_rectangle(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos1, vec2 pos2,
+ u32 c);
+void gui_draw_border(u32 win_id, u32 widget_id, enum gui_layer layer, u32 width, u32 c);
+void gui_draw_line(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos1, vec2 pos2, u32 scale,
+ u32 c);
+
+/**
+ * Widget operations
+ */
void gui_add_widget(u32 *widget, u32 win_id, u32 widget_id, vec2 pos, vec2 size);
void gui_new_widget(u32 *widget, u32 win_id, vec2 pos, vec2 size);
diff --git a/libs/libgui/msg.h b/libs/libgui/msg.h
index b29d0c8..4bffb74 100644
--- a/libs/libgui/msg.h
+++ b/libs/libgui/msg.h
@@ -55,9 +55,12 @@ struct message_mouse {
struct message_header header;
u32 id;
vec2 pos;
+ s8 scroll; // Dir: -1 neg, +1 pos
struct {
- u8 click : 1;
- } bits;
+ u8 left : 1;
+ u8 right : 1;
+ u8 middle : 1;
+ } but;
};
enum message_type {