diff options
-rw-r--r-- | apps/Makefile | 2 | ||||
-rw-r--r-- | apps/init.c | 4 | ||||
-rw-r--r-- | apps/wm.c | 401 | ||||
-rw-r--r-- | kernel/features/proc.c | 1 | ||||
-rw-r--r-- | libc/inc/sys.h | 4 | ||||
-rw-r--r-- | libc/inc/vec.h | 4 | ||||
-rw-r--r-- | libgui/Makefile | 1 | ||||
-rw-r--r-- | libgui/bmp.c | 6 | ||||
-rw-r--r-- | libgui/gfx.c | 88 | ||||
-rw-r--r-- | libgui/inc/bmp.h | 4 | ||||
-rw-r--r-- | libgui/inc/gfx.h | 27 | ||||
-rw-r--r-- | libgui/png.c | 4 | ||||
-rw-r--r-- | libgui/psf.c | 4 | ||||
-rw-r--r-- | libtxt/html.c | 18 |
14 files changed, 205 insertions, 363 deletions
diff --git a/apps/Makefile b/apps/Makefile index 8f0e150..6164010 100644 --- a/apps/Makefile +++ b/apps/Makefile @@ -1,6 +1,6 @@ # MIT License, Copyright (c) 2020 Marvin Borner -COBJS = init.o wm.o mandelbrot.o window.o exec.o files.o test.o cc.o browser.o server.o +COBJS = init.o wm.o test.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/init.c b/apps/init.c index bb9a556..08f2d80 100644 --- a/apps/init.c +++ b/apps/init.c @@ -12,7 +12,7 @@ int main(int argc, char **argv) log("%s loaded\n", argv[0]); int wm = exec("/bin/wm", "wm", argv[1], NULL); - int exec = exec("/bin/exec", "test", NULL); + /* int exec = exec("/bin/exec", "test", NULL); */ - return wm + exec; + return wm; //+ exec; } @@ -1,307 +1,174 @@ // MIT License, Copyright (c) 2020 Marvin Borner #include <assert.h> -#include <cpu.h> #include <def.h> #include <gfx.h> -#include <gui.h> #include <input.h> #include <keymap.h> #include <list.h> -#include <mem.h> -#include <msg.h> -#include <print.h> #include <random.h> -#include <sys.h> #include <vesa.h> -static int MOUSE_SKIP = 0; // => Every move % n != 0 gets skipped -static int context_count; - -static struct vbe vbe; -static struct context direct; // Direct video memory context -static struct context root; // Root context (wallpaper etc.) -static struct context exchange; // Exchange buffer -static struct context cursor; // Cursor bitmap context -static struct context *focused; // The focused context -static struct list *contexts; // List of all contexts - -static struct keymap *keymap; - -static int mouse_x = 0; -static int mouse_y = 0; - -static struct context *new_context(struct context *ctx, u32 pid, int x, int y, u16 width, - u16 height, int flags) +struct client { + u32 pid; +}; + +struct window { + u32 id; + const char *name; + struct context ctx; + struct client client; + u32 flags; + vec3 pos; +}; + +static struct vbe screen = { 0 }; +static struct list *windows = NULL; +static struct window *root = NULL; +static struct window *cursor = NULL; +static struct keymap *keymap = NULL; +static struct { + u8 shift : 1; + u8 alt : 1; + u8 ctrl : 1; +} special_keys = { 0 }; +static struct { + vec2 pos; + u8 left : 1; + u8 mid : 1; + u8 right : 1; +} mouse = { 0 }; + +static struct window *window_create(struct client client, const char *name, struct vec3 pos, + struct vec2 size, u32 flags) { - ctx->pid = pid; - ctx->x = x; - ctx->y = y; - ctx->width = width; - ctx->height = height; - ctx->bpp = vbe.bpp; - ctx->pitch = ctx->width * (ctx->bpp >> 3); - ctx->fb = malloc(height * ctx->pitch); - memset(ctx->fb, 0, height * ctx->pitch); - ctx->flags = flags; - if (!(flags & WF_RELATIVE)) { - context_count++; - if (context_count % 2 == 1) - MOUSE_SKIP++; - } - return ctx; + struct window *win = malloc(sizeof(*win)); + win->id = rand(); + win->name = name; + win->ctx.size = size; + win->ctx.bpp = screen.bpp; + win->ctx.pitch = size.x * (win->ctx.bpp >> 3); + if ((flags & WF_NO_FB) == 0) + win->ctx.fb = malloc(size.y * win->ctx.pitch); + win->client = client; + win->flags = flags; + win->pos = pos; + list_add(windows, win); + return win; } -static void remove_context(struct context *ctx) +static struct window *window_find(u32 id) { - assert(list_remove(contexts, list_first_data(contexts, ctx))); - - if (!(ctx->flags & WF_RELATIVE)) { - if (context_count % 2 == 1) - MOUSE_SKIP--; - context_count--; - } - - free(ctx->fb); - ctx->fb = NULL; - free(ctx); - ctx = NULL; -} - -static struct context *context_at(int x, int y) -{ - if (!contexts->head || !contexts->head->data) - return NULL; - - struct context *ret = NULL; - struct node *iterator = contexts->head; - while (iterator != NULL) { - struct context *ctx = iterator->data; - if (ctx != &root && !(ctx->flags & WF_RELATIVE) && x >= ctx->x && - x <= ctx->x + (int)ctx->width && y >= ctx->y && y <= ctx->y + (int)ctx->height) - ret = ctx; + struct node *iterator = windows->head; + while (iterator) { + struct window *win = iterator->data; + if (win->id == id) + return win; iterator = iterator->next; } - return ret; + return NULL; } -static void kill_focused() +static void window_destroy(struct window *win) { - if (!focused) - return; - msg_send(focused->pid, GUI_KILL, NULL); - remove_context(focused); - focused = context_at(mouse_x, mouse_y); + free(win->ctx.fb); + free(win); } -// This only works if window hasn't moved - TODO! -static void redraw_focused() +static void handle_event_keyboard(struct event_keyboard *event) { - gfx_ctx_on_ctx(&exchange, focused, focused->x, focused->y); - memcpy(direct.fb, exchange.fb, exchange.pitch * exchange.height); -} - -// TODO: Add dirty bitmap redraw (and clipping?): https://github.com/JMarlin/wsbe -static void redraw_all() -{ - if (!contexts->head || !contexts->head->data) + if (event->magic != KEYBOARD_MAGIC) { + log("Keyboard magic doesn't match!\n"); return; - - struct node *iterator = contexts->head; - while (iterator != NULL) { - struct context *ctx = iterator->data; - if (ctx != focused && !(ctx->flags & WF_RELATIVE)) - gfx_ctx_on_ctx(&exchange, ctx, ctx->x, ctx->y); - iterator = iterator->next; } - if (focused) - gfx_ctx_on_ctx(&exchange, focused, focused->x, focused->y); - - memcpy(direct.fb, exchange.fb, exchange.pitch * exchange.height); -} - -#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) - return; - if (event->scancode == KEY_LEFTSHIFT || event->scancode == KEY_RIGHTSHIFT) - special_keys_pressed ^= SHIFT_PRESSED; + special_keys.shift ^= 1; else if (event->scancode == KEY_LEFTALT || event->scancode == KEY_RIGHTALT) - special_keys_pressed ^= ALT_PRESSED; + special_keys.alt ^= 1; else if (event->scancode == KEY_LEFTCTRL || event->scancode == KEY_RIGHTCTRL) - special_keys_pressed ^= CTRL_PRESSED; - - // Special key combos - char ch = keymap->map[event->scancode]; - int mod = event->press && special_keys_pressed & ALT_PRESSED; - if (mod && focused && ch == 'q') { - kill_focused(); - redraw_all(); - return; - } else if (mod && ch == 'p') { - exec("/bin/exec", NULL); - return; - } + special_keys.ctrl ^= 1; - if (!focused) - return; - - // Key maps - struct gui_event_keyboard *msg = malloc(sizeof(*msg)); - if (special_keys_pressed & SHIFT_PRESSED) - msg->ch = keymap->shift_map[event->scancode]; - else if (special_keys_pressed & ALT_PRESSED) - msg->ch = keymap->alt_map[event->scancode]; + char ch; + if (special_keys.shift) + ch = keymap->shift_map[event->scancode]; + else if (special_keys.alt) + ch = keymap->alt_map[event->scancode]; else - msg->ch = ch; + ch = keymap->map[event->scancode]; - msg->press = event->press; - msg->scancode = event->scancode; - msg_send(focused->pid, GUI_KEYBOARD, msg); + (void)ch; } -static int mouse_skip = 0; -static int mouse_pressed[3] = { 0 }; -static void handle_mouse(struct event_mouse *event) +static void handle_event_mouse(struct event_mouse *event) { - if (event->magic != MOUSE_MAGIC) + if (event->magic != MOUSE_MAGIC) { + log("Mouse magic doesn't match!\n"); return; - - // Cursor movement - mouse_x += event->diff_x; - mouse_y -= event->diff_y; - - if (mouse_x < 0) - mouse_x = 0; - else if ((int)(mouse_x + cursor.width) > vbe.width - 1) - mouse_x = vbe.width - cursor.width - 1; - - if (mouse_y < 0) - mouse_y = 0; - else if ((int)(mouse_y + cursor.height) > vbe.height - 1) - mouse_y = vbe.height - cursor.height - 1; - - // 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 (mod_pressed && event->but1 && !mouse_pressed[1]) { - mouse_pressed[0] = 1; - if (focused && !(focused->flags & WF_NO_DRAG)) { - focused->x = mouse_x; - focused->y = mouse_y; - if (mouse_skip % MOUSE_SKIP == 0) { - mouse_skip = 0; - redraw_all(); // TODO: Function to redraw one context - } - } - } else if (mod_pressed && mouse_pressed[0]) { - mouse_pressed[0] = 0; - redraw_all(); } - // Context size - 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; - } else if (focused && !(focused->flags & WF_NO_RESIZE) && - mouse_skip % MOUSE_SKIP == 0) { - mouse_skip = 0; - if (mouse_x - focused->x > 0) - focused->width = mouse_x - focused->x; - if (mouse_y - focused->y > 0) { - focused->height = mouse_y - focused->y; - } - } - mouse_pressed[1] = 1; - } else if (mod_pressed && mouse_pressed[1]) { - mouse_pressed[1] = 0; - if (focused) { - struct context *resized = malloc(sizeof(*resized)); - new_context(resized, focused->pid, focused->x, focused->y, focused->width, - focused->height, focused->flags); - remove_context(focused); - list_add(contexts, resized); - focused = resized; - struct gui_event_resize *msg = malloc(sizeof(*msg)); - msg->new_ctx = resized; - msg_send(resized->pid, GUI_RESIZE, msg); - redraw_all(); - } + mouse.pos.x += event->diff_x; + mouse.pos.y -= event->diff_y; + + // Fix x overflow + if ((signed)mouse.pos.x < 0) + mouse.pos.x = 0; + else if (mouse.pos.x + cursor->ctx.size.x > (unsigned)screen.width - 1) + mouse.pos.x = screen.width - cursor->ctx.size.x - 1; + + // Fix y overflow + if ((signed)mouse.pos.y < 0) + mouse.pos.y = 0; + else if (mouse.pos.y + cursor->ctx.size.y > (unsigned)screen.height - 1) + mouse.pos.y = screen.height - cursor->ctx.size.y - 1; + + //log("%d %d\n", mouse.pos.x, mouse.pos.y); + cursor->pos = vec2to3(mouse.pos, U32_MAX); + + gfx_fill(&cursor->ctx, COLOR_RED); + if (event->but1) { + /* TODO: Fix gfx_copy! */ + /* gfx_copy(&root->ctx, &cursor->ctx, vec3to2(cursor->pos), cursor->ctx.size); */ + gfx_draw_rectangle(&root->ctx, vec3to2(cursor->pos), + vec2_add(cursor->pos, cursor->ctx.size), COLOR_RED); } - - cursor.x = mouse_x; - cursor.y = mouse_y; - gfx_ctx_on_ctx(&direct, &cursor, cursor.x, cursor.y); - mouse_skip++; - - if (!focused) - return; - 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 int main(int argc, char **argv) { (void)argc; int pid = getpid(); - vbe = *(struct vbe *)argv[1]; - log("WM loaded: %dx%d\n", vbe.width, vbe.height); + screen = *(struct vbe *)argv[1]; + log("WM loaded: %dx%d\n", screen.width, screen.height); + windows = list_new(); keymap = keymap_parse("/res/keymaps/en.keymap"); - contexts = list_new(); - new_context(&root, pid, 0, 0, vbe.width, vbe.height, - WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); - new_context(&exchange, pid, 0, 0, vbe.width, vbe.height, - WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); - new_context(&cursor, pid, 0, 0, 32, 32, WF_NO_FOCUS | WF_NO_RESIZE); - memcpy(&direct, &root, sizeof(direct)); - direct.fb = vbe.fb; - list_add(contexts, &root); + root = window_create((struct client){ pid }, "root", vec3(0, 0, 0), + vec2(screen.width, screen.height), WF_NO_FB); + root->ctx.fb = screen.fb; + root->flags ^= WF_NO_FB; + cursor = window_create((struct client){ pid }, "cursor", vec3(0, 0, 0), vec2(20, 20), + WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE); - gfx_write(&direct, 0, 0, FONT_32, COLOR_FG, "Welcome to Melvix!"); - gfx_write(&direct, 0, 32, FONT_32, COLOR_FG, "Loading resources..."); - gfx_fill(&root, COLOR_FG); - //gfx_load_wallpaper(&root, "/res/wall.png"); - //gfx_load_image(&cursor, "/res/cursor.png", 0, 0); - gfx_fill(&cursor, COLOR_BG); - redraw_all(); + gfx_fill(&root->ctx, COLOR_WHITE); struct message msg = { 0 }; - struct event_keyboard kbd_event = { 0 }; - struct event_mouse mouse_event = { 0 }; + struct event_keyboard event_keyboard = { 0 }; + struct event_mouse event_mouse = { 0 }; const char *listeners[] = { "/dev/kbd", "/dev/mouse", "/proc/self/msg" }; while (1) { int poll_ret = 0; if ((poll_ret = poll(listeners)) >= 0) { if (poll_ret == 0) { - if (read(listeners[poll_ret], &kbd_event, 0, sizeof(kbd_event)) > 0) - handle_keyboard(&kbd_event); + if (read(listeners[poll_ret], &event_keyboard, 0, + sizeof(event_keyboard)) > 0) + handle_event_keyboard(&event_keyboard); continue; } else if (poll_ret == 1) { - if (read(listeners[poll_ret], &mouse_event, 0, - sizeof(mouse_event)) > 0) - handle_mouse(&mouse_event); + if (read(listeners[poll_ret], &event_mouse, 0, + sizeof(event_mouse)) > 0) + handle_event_mouse(&event_mouse); continue; } else if (poll_ret == 2) { if (read(listeners[poll_ret], &msg, 0, sizeof(msg)) <= 0) @@ -311,41 +178,17 @@ int main(int argc, char **argv) err(1, "POLL ERROR!\n"); } - assert(msg.magic == MSG_MAGIC); - - switch (msg.type) { - case GFX_NEW_CONTEXT: { - struct context *ctx = msg.data; - int width = ctx->width; - int height = ctx->height; - int x = ctx->x; - int y = ctx->y; - ctx->pid = msg.src; - new_context(ctx, msg.src, x, y, width, height, ctx->flags); - list_add(contexts, ctx); - if (!(ctx->flags & WF_RELATIVE)) - focused = ctx; - redraw_all(); - msg_send(msg.src, GFX_NEW_CONTEXT, ctx); - - // Send mouse position - struct gui_event_mouse *mouse = malloc(sizeof(msg)); - mouse->x = mouse_x - focused->x; - mouse->y = mouse_y - focused->y; - msg_send(focused->pid, GUI_MOUSE, mouse); - break; - } - case GFX_REDRAW: - redraw_all(); - break; - case GFX_REDRAW_FOCUSED: - redraw_focused(); - break; - default: - log("Unknown WM request %d\n", msg.type); - break; + if (msg.magic != MSG_MAGIC) { + log("Message magic doesn't match!\n"); + continue; } + + log("not implemented!\n"); }; + // TODO: Execute? + free(keymap); + list_destroy(windows); + return 0; } diff --git a/kernel/features/proc.c b/kernel/features/proc.c index 74fa5d4..2ca213b 100644 --- a/kernel/features/proc.c +++ b/kernel/features/proc.c @@ -290,6 +290,7 @@ s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct devi path++; if (!memcmp(path, "msg", 4)) { + // TODO: Messages should be copied by value not by reference stack_push_bot(p->messages, buf); // TODO: Use offset and count proc_enable_waiting(pid, PROC_WAIT_MSG); return count; diff --git a/libc/inc/sys.h b/libc/inc/sys.h index aead212..be4352f 100644 --- a/libc/inc/sys.h +++ b/libc/inc/sys.h @@ -29,13 +29,13 @@ enum sys { }; struct event_keyboard { - int magic; + u32 magic; int press; int scancode; }; struct event_mouse { - int magic; + u32 magic; int diff_x; int diff_y; int but1; diff --git a/libc/inc/vec.h b/libc/inc/vec.h index 9eb706c..024e792 100644 --- a/libc/inc/vec.h +++ b/libc/inc/vec.h @@ -13,6 +13,8 @@ typedef struct vec3 { u32 x, y, z; } vec3; +#define vec2(x, y) ((vec2){ x, y }) +#define vec2to3(a, z) ((vec3){ a.x, a.y, z }) #define vec2_add(a, b) ((vec2){ a.x + b.x, a.y + b.y }) #define vec2_sub(a, b) ((vec2){ a.x - b.x, a.y - b.y }) #define vec2_mul(a, b) ((vec2){ a.x * b, a.y * b }) @@ -20,6 +22,8 @@ typedef struct vec3 { #define vec2_dot(a, b) ((u32)(a.x * b.x + a.y * b.y)) #define vec2_eq(a, b) (a.x == b.x && a.y == b.y) +#define vec3(x, y, z) ((vec3){ x, y, z }) +#define vec3to2(a) ((vec2){ a.x, a.y }) #define vec3_add(a, b) ((vec3){ a.x + b.x, a.y + b.y, a.z + b.z }) #define vec3_sub(a, b) ((vec3){ a.x - b.x, a.y - b.y, a.z - b.z }) #define vec3_mul(a, b) ((vec3){ a.x * b, a.y * b, a.z * b }) diff --git a/libgui/Makefile b/libgui/Makefile index b37bc3b..cb02fed 100644 --- a/libgui/Makefile +++ b/libgui/Makefile @@ -4,7 +4,6 @@ COBJS = psf.o \ bmp.o \ png.o \ gfx.o \ - gui.o \ msg.o CC = ccache ../cross/opt/bin/i686-elf-gcc LD = ccache ../cross/opt/bin/i686-elf-ld diff --git a/libgui/bmp.c b/libgui/bmp.c index 9fffbf5..c8795e8 100644 --- a/libgui/bmp.c +++ b/libgui/bmp.c @@ -19,11 +19,11 @@ struct bmp *bmp_load(const char *path) // TODO: Support padding with odd widths struct bmp_info *info = (struct bmp_info *)((u32)buf + sizeof(*h)); struct bmp *bmp = malloc(sizeof(*bmp)); - bmp->width = info->width; - bmp->height = info->height; + bmp->size.x = info->width; + bmp->size.y = info->height; bmp->data = (u8 *)((u32)buf + h->offset); bmp->bpp = info->bpp; - bmp->pitch = bmp->width * (bmp->bpp >> 3); + bmp->pitch = bmp->size.x * (bmp->bpp >> 3); return bmp; } diff --git a/libgui/gfx.c b/libgui/gfx.c index a6e23e6..bd7165a 100644 --- a/libgui/gfx.c +++ b/libgui/gfx.c @@ -59,16 +59,15 @@ static void load_font(enum font_type font_type) assert(fonts[font_type]); } -static void write_char(struct context *ctx, int x, int y, struct font *font, u32 c, char ch) +static void write_char(struct context *ctx, vec2 pos, struct font *font, u32 c, char ch) { int bypp = ctx->bpp >> 3; - int pos = x * bypp + y * ctx->pitch; - char *draw = (char *)&ctx->fb[pos]; + char *draw = (char *)&ctx->fb[pos.x * bypp + pos.y * ctx->pitch]; - u32 stride = font->char_size / font->height; - for (int cy = 0; cy < font->height; cy++) { - for (int cx = 0; cx < font->width; cx++) { + u32 stride = font->char_size / font->size.y; + for (u32 cy = 0; cy < font->size.y; cy++) { + for (u32 cx = 0; cx < font->size.x; cx++) { u8 bits = font->chars[ch * font->char_size + cy * stride + cx / 8]; u8 bit = bits >> (7 - cx % 8) & 1; if (bit) { @@ -82,12 +81,12 @@ static void write_char(struct context *ctx, int x, int y, struct font *font, u32 } } -static void draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c) +static void draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c) { int bypp = ctx->bpp >> 3; - u8 *draw = &ctx->fb[x1 * bypp + y1 * ctx->pitch]; - for (int i = 0; i < y2 - y1; i++) { - for (int j = 0; j < x2 - x1; j++) { + 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++) { draw[bypp * j] = GET_BLUE(c); draw[bypp * j + 1] = GET_GREEN(c); draw[bypp * j + 2] = GET_RED(c); @@ -114,14 +113,14 @@ struct font *gfx_resolve_font(enum font_type font_type) return fonts[font_type]; } -void gfx_write_char(struct context *ctx, int x, int y, enum font_type font_type, u32 c, char ch) +void gfx_write_char(struct context *ctx, vec2 pos, enum font_type font_type, u32 c, char ch) { struct font *font = gfx_resolve_font(font_type); - write_char(ctx, x, y, font, c, ch); + write_char(ctx, pos, font, c, ch); /* gfx_redraw(); */ } -void gfx_write(struct context *ctx, int x, int y, enum font_type font_type, u32 c, const char *text) +void gfx_write(struct context *ctx, vec2 pos, enum font_type font_type, u32 c, const char *text) { struct font *font = gfx_resolve_font(font_type); u32 cnt = 0; @@ -131,37 +130,37 @@ void gfx_write(struct context *ctx, int x, int y, enum font_type font_type, u32 cnt = 0; } else if (text[i] == '\n') { cnt = 0; - y += font->height; + pos.y += font->size.y; } else if (text[i] == '\t') { cnt += 4; } else { // TODO: Overflow on single line input - if ((cnt + 1) * font->width > ctx->width) { + if ((cnt + 1) * font->size.x > ctx->size.x) { cnt = 0; - y += font->height; + pos.y += font->size.y; } - write_char(ctx, x + cnt * font->width, y, font, c, text[i]); + write_char(ctx, vec2(pos.x + cnt * font->size.x, pos.y), font, c, text[i]); cnt++; } } /* gfx_redraw(); */ } -void gfx_load_image(struct context *ctx, const char *path, int x, int y) +void gfx_load_image(struct context *ctx, vec2 pos, const char *path) { // TODO: Support x, y // TODO: Detect image type // struct bmp *bmp = bmp_load(path); struct bmp *bmp = png_load(path); - assert(bmp && bmp->width + x <= ctx->width); - assert(bmp && bmp->height + y <= ctx->height); + assert(bmp && bmp->size.x + pos.x <= ctx->size.x); + assert(bmp && bmp->size.y + pos.y <= ctx->size.y); // TODO: Fix reversed png in decoder int bypp = bmp->bpp >> 3; - // u8 *srcfb = &bmp->data[bypp + (bmp->height - 1) * bmp->pitch]; + // u8 *srcfb = &bmp->data[bypp + (bmp->size.y - 1) * bmp->pitch]; u8 *srcfb = bmp->data; u8 *destfb = &ctx->fb[bypp]; - for (u32 cy = 0; cy < bmp->height; cy++) { + for (u32 cy = 0; cy < bmp->size.y; cy++) { memcpy(destfb, srcfb, bmp->pitch); // srcfb -= bmp->pitch; srcfb += bmp->pitch; @@ -172,16 +171,16 @@ void gfx_load_image(struct context *ctx, const char *path, int x, int y) void gfx_load_wallpaper(struct context *ctx, const char *path) { - gfx_load_image(ctx, path, 0, 0); + gfx_load_image(ctx, vec2(0, 0), path); } -void gfx_copy(struct context *dest, struct context *src, int x, int y, u32 width, u32 height) +void gfx_copy(struct context *dest, struct context *src, vec2 pos, vec2 size) { int bypp = dest->bpp >> 3; - u8 *srcfb = &src->fb[x * bypp + y * src->pitch]; - u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; - for (u32 cy = 0; cy < height; cy++) { - memcpy(destfb, srcfb, width * bypp); + 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++) { + memcpy(destfb, srcfb, size.x * bypp); srcfb += src->pitch; destfb += dest->pitch; } @@ -189,24 +188,23 @@ void gfx_copy(struct context *dest, struct context *src, int x, int y, u32 width // TODO: Support alpha values other than 0x0 and 0xff (blending) // TODO: Optimize! -void gfx_ctx_on_ctx(struct context *dest, struct context *src, int x, int y) +void gfx_ctx_on_ctx(struct context *dest, struct context *src, vec2 pos) { - if (src->width == dest->width && src->height == dest->height && src->x == 0 && - dest->y == 0) { - memcpy(dest->fb, src->fb, dest->pitch * dest->height); + if (src->size.x == dest->size.x && src->size.y == dest->size.y) { + memcpy(dest->fb, src->fb, dest->pitch * dest->size.y); return; } - if (src->width > dest->width || src->height > dest->height) + if (src->size.x > dest->size.x || src->size.y > dest->size.y) return; // TODO: Negative x and y int bypp = dest->bpp >> 3; u8 *srcfb = src->fb; - u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; - for (u32 cy = 0; cy < src->height && cy + y < dest->height; cy++) { + u8 *destfb = &dest->fb[pos.x * bypp + pos.y * dest->pitch]; + for (u32 cy = 0; cy < src->size.y && cy + pos.y < dest->size.y; cy++) { int diff = 0; - for (u32 cx = 0; cx < src->width && cx + x < dest->width; cx++) { + for (u32 cx = 0; cx < src->size.x && cx + pos.x < dest->size.x; cx++) { if (srcfb[3]) { destfb[0] = srcfb[0]; destfb[1] = srcfb[1]; @@ -223,15 +221,15 @@ void gfx_ctx_on_ctx(struct context *dest, struct context *src, int x, int y) } } -void gfx_draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c) +void gfx_draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c) { - draw_rectangle(ctx, x1, y1, x2, y2, c); + draw_rectangle(ctx, pos1, pos2, c); /* gfx_redraw(); */ } void gfx_fill(struct context *ctx, u32 c) { - draw_rectangle(ctx, 0, 0, ctx->width, ctx->height, c); + draw_rectangle(ctx, vec2(0, 0), vec2(ctx->size.x, ctx->size.y), c); /* gfx_redraw(); */ } @@ -242,11 +240,11 @@ void gfx_border(struct context *ctx, u32 c, u32 width) int bypp = ctx->bpp >> 3; u8 *draw = ctx->fb; - for (u32 i = 0; i < ctx->height; i++) { - for (u32 j = 0; j < ctx->width; j++) { + 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->width + width + 1 <= width || - i - ctx->height + width <= width) { + 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); @@ -261,11 +259,11 @@ void gfx_border(struct context *ctx, u32 c, u32 width) int gfx_font_height(enum font_type font_type) { struct font *font = gfx_resolve_font(font_type); - return font->height; + return font->size.y; } int gfx_font_width(enum font_type font_type) { struct font *font = gfx_resolve_font(font_type); - return font->width; + return font->size.x; } diff --git a/libgui/inc/bmp.h b/libgui/inc/bmp.h index ea7a07f..ff8360b 100644 --- a/libgui/inc/bmp.h +++ b/libgui/inc/bmp.h @@ -4,6 +4,7 @@ #define BMP_H #include <def.h> +#include <vec.h> struct bmp_header { u8 signature[2]; @@ -27,8 +28,7 @@ struct bmp_info { }; struct bmp { - u32 width; - u32 height; + vec2 size; u8 *data; u32 bpp; u32 pitch; diff --git a/libgui/inc/gfx.h b/libgui/inc/gfx.h index fd8d4ab..a283459 100644 --- a/libgui/inc/gfx.h +++ b/libgui/inc/gfx.h @@ -7,6 +7,7 @@ #include <def.h> #include <msg.h> #include <sys.h> +#include <vec.h> #include <vesa.h> #define WM_PATH "/bin/wm" @@ -41,40 +42,34 @@ #define WF_NO_FOCUS (1 << 0) #define WF_NO_DRAG (1 << 1) #define WF_NO_RESIZE (1 << 2) -#define WF_RELATIVE (1 << 3) +#define WF_NO_FB (1 << 3) +/* #define WF_RELATIVE (1 << 4) */ enum font_type { FONT_8, FONT_12, FONT_16, FONT_24, FONT_32, FONT_64 }; // Generalized font struct struct font { char *chars; - int height; - int width; + vec2 size; int char_size; }; struct context { - u32 pid; - int x; - int y; - u32 width; - u32 height; + vec2 size; u8 *fb; u32 bpp; u32 pitch; - int flags; }; struct context *gfx_new_ctx(struct context *ctx); struct font *gfx_resolve_font(enum font_type font_type); -void gfx_write_char(struct context *ctx, int x, int y, enum font_type font_type, u32 c, char ch); -void gfx_write(struct context *ctx, int x, int y, enum font_type font_type, u32 c, - const char *text); -void gfx_load_image(struct context *ctx, const char *path, int x, int y); +void gfx_write_char(struct context *ctx, vec2 pos, enum font_type font_type, u32 c, char ch); +void gfx_write(struct context *ctx, vec2 pos, enum font_type font_type, u32 c, const char *text); +void gfx_load_image(struct context *ctx, vec2 pos, const char *path); void gfx_load_wallpaper(struct context *ctx, const char *path); -void gfx_copy(struct context *dest, struct context *src, int x, int y, u32 width, u32 height); -void gfx_ctx_on_ctx(struct context *dest, struct context *src, int x, int y); -void gfx_draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c); +void gfx_copy(struct context *dest, struct context *src, vec2 pos, vec2 size); +void gfx_ctx_on_ctx(struct context *dest, struct context *src, vec2 pos); +void gfx_draw_rectangle(struct context *ctx, vec2 pos1, vec2 pos2, u32 c); void gfx_fill(struct context *ctx, u32 c); void gfx_border(struct context *ctx, u32 c, u32 width); diff --git a/libgui/png.c b/libgui/png.c index ad65eba..48089e3 100644 --- a/libgui/png.c +++ b/libgui/png.c @@ -1182,8 +1182,8 @@ struct bmp *png_load(const char *path) assert(png->error == PNG_EOK); struct bmp *bmp = malloc(sizeof(*bmp)); - bmp->width = png->width; - bmp->height = png->height; + bmp->size.x = png->width; + bmp->size.y = png->height; bmp->data = png->buffer; bmp->bpp = png_get_bpp(png); bmp->pitch = png->width * (bmp->bpp >> 3); diff --git a/libgui/psf.c b/libgui/psf.c index f7271a8..655fb07 100644 --- a/libgui/psf.c +++ b/libgui/psf.c @@ -52,8 +52,8 @@ struct font *psf_parse(char *data) struct font *font = malloc(sizeof(*font)); font->chars = chars; - font->height = height; - font->width = width; + font->size.x = width; + font->size.y = height; font->char_size = char_size; return font; diff --git a/libtxt/html.c b/libtxt/html.c index eca2feb..f40f7b3 100644 --- a/libtxt/html.c +++ b/libtxt/html.c @@ -169,33 +169,35 @@ static struct html_element *render_object(struct html_element *container, struct gui_add_label(container->obj, container->x_offset, container->y_offset, FONT_32, dom->content, COLOR_WHITE, COLOR_BLACK); container->x_offset = 0; - container->y_offset += obj->ctx->height; + container->y_offset += obj->ctx->size.y; return new_html_element(obj, dom); } else if (CMP(tag, "h2")) { struct element *obj = gui_add_label(container->obj, container->x_offset, container->y_offset, FONT_24, dom->content, COLOR_WHITE, COLOR_BLACK); container->x_offset = 0; - container->y_offset += obj->ctx->height; + container->y_offset += obj->ctx->size.y; return new_html_element(obj, dom); } else if (CMP(tag, "h3")) { struct element *obj = gui_add_label(container->obj, container->x_offset, container->y_offset, FONT_16, dom->content, COLOR_WHITE, COLOR_BLACK); container->x_offset = 0; - container->y_offset += obj->ctx->height; + container->y_offset += obj->ctx->size.y; return new_html_element(obj, dom); } else if (CMP(tag, "p")) { struct element *obj = gui_add_label(container->obj, container->x_offset, container->y_offset, FONT_16, dom->content, COLOR_WHITE, COLOR_BLACK); container->x_offset = 0; - container->y_offset += obj->ctx->height; + container->y_offset += obj->ctx->size.y; return new_html_element(obj, dom); } else if (CMP(tag, "hr")) { - gfx_draw_rectangle(container->obj->ctx, container->x_offset, container->y_offset, - container->obj->ctx->width - container->x_offset, - container->y_offset + 2, COLOR_BLACK); + gfx_draw_rectangle(container->obj->ctx, + vec2(container->x_offset, container->y_offset), + vec2(container->obj->ctx->size.x - container->x_offset, + container->y_offset + 2), + COLOR_BLACK); container->x_offset = 0; container->y_offset += 2; return container; @@ -208,7 +210,7 @@ static struct html_element *render_object(struct html_element *container, struct container->y_offset, FONT_16, dom->content, COLOR_WHITE, COLOR_BLACK); container->x_offset = 0; - container->y_offset += obj->ctx->height; + container->y_offset += obj->ctx->size.y; return new_html_element(obj, dom); } return container; |