diff options
Diffstat (limited to 'libs')
-rw-r--r-- | libs/libc/inc/math.h | 7 | ||||
-rw-r--r-- | libs/libc/math.c | 26 | ||||
-rw-r--r-- | libs/libgui/Makefile | 1 | ||||
-rw-r--r-- | libs/libgui/bmp.c | 29 | ||||
-rw-r--r-- | libs/libgui/bmp.h | 39 | ||||
-rw-r--r-- | libs/libgui/gfx.c | 124 | ||||
-rw-r--r-- | libs/libgui/gfx.h | 20 | ||||
-rw-r--r-- | libs/libgui/gui.c | 10 | ||||
-rw-r--r-- | libs/libgui/gui.h | 4 |
9 files changed, 151 insertions, 109 deletions
diff --git a/libs/libc/inc/math.h b/libs/libc/inc/math.h index 4a38c5f..813ebf1 100644 --- a/libs/libc/inc/math.h +++ b/libs/libc/inc/math.h @@ -19,9 +19,14 @@ f32 powf(f32 base, f32 exp); f64 pow(f64 base, f64 exp); -f32 sqrtf(f64 num); +f32 sqrtf(f32 num); f64 sqrt(f64 num); +f32 lerpf(f32 from, f32 to, f32 trans); +f64 lerp(f64 from, f64 to, f64 trans); +f32 blerpf(f32 a, f32 b, f32 c, f32 d, f32 transx, f32 transy); +f64 blerp(f64 a, f64 b, f64 c, f64 d, f64 transx, f64 transy); + f32 sinf(f32 angle); f64 sin(f64 angle); f32 cosf(f32 angle); diff --git a/libs/libc/math.c b/libs/libc/math.c index 23b1ca4..606281c 100644 --- a/libs/libc/math.c +++ b/libs/libc/math.c @@ -32,7 +32,7 @@ f64 pow(f64 base, f64 exp) // TODO: More efficient sqrt? -f32 sqrtf(f64 num) +f32 sqrtf(f32 num) { return powf(num, .5); } @@ -43,6 +43,30 @@ f64 sqrt(f64 num) } /** + * Interpolations + */ + +f32 lerpf(f32 from, f32 to, f32 trans) +{ + return from + (to - from) * trans; +} + +f64 lerp(f64 from, f64 to, f64 trans) +{ + return from + (to - from) * trans; +} + +f32 blerpf(f32 a, f32 b, f32 c, f32 d, f32 transx, f32 transy) +{ + return lerpf(lerpf(a, b, transx), lerpf(c, d, transx), transy); +} + +f64 blerp(f64 a, f64 b, f64 c, f64 d, f64 transx, f64 transy) +{ + return lerp(lerp(a, b, transx), lerp(c, d, transx), transy); +} + +/** * Trigonometric functions */ diff --git a/libs/libgui/Makefile b/libs/libgui/Makefile index 2f76e61..a8f3a62 100644 --- a/libs/libgui/Makefile +++ b/libs/libgui/Makefile @@ -1,7 +1,6 @@ # MIT License, Copyright (c) 2020 Marvin Borner COBJS = psf.o \ - bmp.o \ png.o \ gfx.o \ gui.o \ diff --git a/libs/libgui/bmp.c b/libs/libgui/bmp.c deleted file mode 100644 index 5576d16..0000000 --- a/libs/libgui/bmp.c +++ /dev/null @@ -1,29 +0,0 @@ -// MIT License, Copyright (c) 2020 Marvin Borner - -#include <def.h> -#include <libgui/bmp.h> -#include <mem.h> -#include <print.h> -#include <sys.h> - -struct bmp *bmp_load(const char *path) -{ - void *buf = sread(path); - if (!buf) - return NULL; - - struct bmp_header *h = buf; - if (h->signature[0] != 'B' || h->signature[1] != 'M') - return NULL; - - // TODO: Support padding with odd widths - struct bmp_info *info = (struct bmp_info *)((u32)buf + sizeof(*h)); - struct bmp *bmp = malloc(sizeof(*bmp)); - bmp->size.x = info->width; - bmp->size.y = info->height; - bmp->data = (u8 *)((u32)buf + h->offset); - bmp->bpp = info->bpp; - bmp->pitch = bmp->size.x * (bmp->bpp >> 3); - - return bmp; -} diff --git a/libs/libgui/bmp.h b/libs/libgui/bmp.h deleted file mode 100644 index f7fb57f..0000000 --- a/libs/libgui/bmp.h +++ /dev/null @@ -1,39 +0,0 @@ -// MIT License, Copyright (c) 2020 Marvin Borner - -#ifndef BMP_H -#define BMP_H - -#include <def.h> -#include <vec.h> - -struct bmp_header { - u8 signature[2]; - u32 size; - u32 reserved; - u32 offset; -} PACKED; - -struct bmp_info { - u32 size; - u32 width; - u32 height; - u16 planes; - u16 bpp; - u32 compression; - u32 compressed_size; - u32 x_pixel_meter; - u32 y_pixel_meter; - u32 colors; - u32 important_colors; -}; - -struct bmp { - vec2 size; - u8 *data; - u32 bpp; - u32 pitch; -}; - -struct bmp *bmp_load(const char *path) NONNULL; - -#endif diff --git a/libs/libgui/gfx.c b/libs/libgui/gfx.c index 373c099..1341a5a 100644 --- a/libs/libgui/gfx.c +++ b/libs/libgui/gfx.c @@ -5,12 +5,12 @@ #include <assert.h> #include <crypto.h> -#include <libgui/bmp.h> #include <libgui/gfx.h> #include <libgui/msg.h> #include <libgui/png.h> #include <libgui/psf.h> #include <list.h> +#include <math.h> #include <mem.h> #include <str.h> #include <sys.h> @@ -102,6 +102,19 @@ struct gfx_context *gfx_new_ctx(struct gfx_context *ctx, vec2 size, u8 bpp) return ctx; } +struct gfx_context *gfx_clone(struct gfx_context *ctx) +{ + struct gfx_context *new = zalloc(sizeof(*new)); + gfx_new_ctx(new, ctx->size, ctx->bpp); + assert(new->bytes == ctx->bytes); + memcpy(new->fb, ctx->fb, ctx->bytes); + return new; +} + +/** + * Font/text + */ + // On-demand font loading static u8 fonts_loaded = 0; struct gfx_font *gfx_resolve_font(enum font_type font_type) @@ -147,9 +160,13 @@ void gfx_write(struct gfx_context *ctx, vec2 pos, enum font_type font_type, u32 } } +/** + * Image drawing/caching + */ + struct gfx_image_cache { u32 hash; - struct bmp *bmp; + struct gfx_context *ctx; }; struct list *gfx_image_cache_list = NULL; @@ -159,8 +176,8 @@ static void gfx_image_cache_clear(void) struct node *iterator = gfx_image_cache_list->head; while (iterator) { struct gfx_image_cache *cache = iterator->data; - free(cache->bmp->data); - free(cache->bmp); + free(cache->ctx->fb); + free(cache->ctx); free(cache); iterator = iterator->next; } @@ -176,7 +193,7 @@ static void gfx_image_cache_init(void) } } -static struct bmp *gfx_image_cache_get(const char *path) +static struct gfx_context *gfx_image_cache_get(const char *path) { gfx_image_cache_init(); @@ -185,61 +202,65 @@ static struct bmp *gfx_image_cache_get(const char *path) struct node *iterator = gfx_image_cache_list->head; while (iterator) { struct gfx_image_cache *cache = iterator->data; - if (cache->hash == hash) { - return cache->bmp; - } + if (cache->hash == hash) + return cache->ctx; iterator = iterator->next; } return NULL; } -static void gfx_image_cache_save(const char *path, struct bmp *bmp) +static void gfx_image_cache_save(const char *path, struct gfx_context *ctx) { gfx_image_cache_init(); struct gfx_image_cache *cache = zalloc(sizeof(*cache)); cache->hash = crc32(0, path, strlen(path)); - cache->bmp = bmp; + cache->ctx = ctx; list_add(gfx_image_cache_list, cache); } -void gfx_load_image_filter(struct gfx_context *ctx, vec2 pos, enum gfx_filter filter, +void gfx_draw_image_filter(struct gfx_context *ctx, vec2 pos, vec2 size, enum gfx_filter filter, const char *path) { // TODO: Detect image type - struct bmp *bmp = gfx_image_cache_get(path); + struct gfx_context *bmp = gfx_image_cache_get(path); if (!bmp) { bmp = zalloc(sizeof(*bmp)); - u32 error = png_decode32_file(&bmp->data, &bmp->size.x, &bmp->size.y, path); + u32 error = png_decode32_file(&bmp->fb, &bmp->size.x, &bmp->size.y, path); if (error) - err(1, "error %u: %s\n", error, png_error_text(error)); + err(1, "error %d: %s\n", error, png_error_text(error)); bmp->bpp = 32; bmp->pitch = bmp->size.x * (bmp->bpp >> 3); + bmp->bytes = bmp->size.y * bmp->pitch; gfx_image_cache_save(path, bmp); } + // Scaling clones! + bmp = gfx_scale(bmp, size); + assert(bmp->size.x + pos.x <= ctx->size.x); assert(bmp->size.y + pos.y <= ctx->size.y); - // TODO: Fix reversed png in decoder u8 bypp = bmp->bpp >> 3; - /* u8 *srcfb = &bmp->data[bypp + (bmp->size.y - 1) * bmp->pitch]; */ - u8 *srcfb = bmp->data; + u8 *srcfb = bmp->fb; u8 *destfb = &ctx->fb[pos.x * bypp + pos.y * ctx->pitch]; for (u32 cy = 0; cy < bmp->size.y && cy + pos.y < ctx->size.y; cy++) { int diff = 0; for (u32 cx = 0; cx < bmp->size.x && cx + pos.x < ctx->size.x; cx++) { if (srcfb[bypp - 1]) { if (filter == GFX_FILTER_NONE) { - memcpy(destfb, srcfb, bypp); + destfb[0] = srcfb[2]; + destfb[1] = srcfb[1]; + destfb[2] = srcfb[0]; + destfb[3] = srcfb[3]; } else if (filter == GFX_FILTER_INVERT) { - destfb[0] = 0xff - srcfb[0]; + destfb[0] = 0xff - srcfb[2]; destfb[1] = 0xff - srcfb[1]; - destfb[2] = 0xff - srcfb[2]; + destfb[2] = 0xff - srcfb[0]; destfb[3] = srcfb[3]; } } @@ -251,22 +272,75 @@ void gfx_load_image_filter(struct gfx_context *ctx, vec2 pos, enum gfx_filter fi srcfb += bmp->pitch - diff; destfb += ctx->pitch - diff; } + + free(bmp); + free(bmp->fb); } -void gfx_load_image(struct gfx_context *ctx, vec2 pos, const char *path) +void gfx_draw_image(struct gfx_context *ctx, vec2 pos, vec2 size, const char *path) { - gfx_load_image_filter(ctx, pos, GFX_FILTER_NONE, path); + gfx_draw_image_filter(ctx, pos, size, GFX_FILTER_NONE, path); } void gfx_load_wallpaper(struct gfx_context *ctx, const char *path) { - gfx_load_image(ctx, vec2(0, 0), path); + gfx_draw_image(ctx, vec2(0, 0), ctx->size, path); } -void gfx_draw_pixel(struct gfx_context *ctx, vec2 pos1, u32 c) +/** + * Context transformations + */ + +// Using bilinear interpolation +// TODO: Fix alpha channel scaling +struct gfx_context *gfx_scale(struct gfx_context *ctx, vec2 size) { + if (vec2_eq(ctx->size, size)) + return gfx_clone(ctx); + u8 bypp = ctx->bpp >> 3; - u8 *draw = &ctx->fb[pos1.x * bypp + pos1.y * ctx->pitch]; + + struct gfx_context *new = zalloc(sizeof(*new)); + gfx_new_ctx(new, size, ctx->bpp); + + for (u32 x = 0, y = 0; y < size.y; x++) { + if (x > size.x) { + x = 0; + y++; + } + + f32 gx = x / (f32)size.x * (ctx->size.x - 1); + f32 gy = y / (f32)size.y * (ctx->size.y - 1); + u32 gxi = (u32)gx; + u32 gyi = (u32)gy; + + u32 a = *(u32 *)&ctx->fb[(gxi + 0) * bypp + (gyi + 0) * ctx->pitch]; + u32 b = *(u32 *)&ctx->fb[(gxi + 1) * bypp + (gyi + 0) * ctx->pitch]; + u32 c = *(u32 *)&ctx->fb[(gxi + 0) * bypp + (gyi + 1) * ctx->pitch]; + u32 d = *(u32 *)&ctx->fb[(gxi + 1) * bypp + (gyi + 1) * ctx->pitch]; + + u32 color = 0; + for (u8 i = 0; i < bypp - 1; i++) { + color |= ((u8)blerpf(GET_COLOR(a, i), GET_COLOR(b, i), GET_COLOR(c, i), + GET_COLOR(d, i), gx - gxi, gy - gyi)) + << (i << 3); + } + + color |= 0xffu << ((bypp - 1) << 3); + gfx_draw_pixel(new, vec2(x, y), color); + } + + return new; +} + +/** + * General drawing functions + */ + +void gfx_draw_pixel(struct gfx_context *ctx, vec2 pos, u32 c) +{ + u8 bypp = ctx->bpp >> 3; + u8 *draw = &ctx->fb[pos.x * bypp + pos.y * ctx->pitch]; draw[0] = GET_BLUE(c); draw[1] = GET_GREEN(c); draw[2] = GET_RED(c); diff --git a/libs/libgui/gfx.h b/libs/libgui/gfx.h index 855e060..cb89ff5 100644 --- a/libs/libgui/gfx.h +++ b/libs/libgui/gfx.h @@ -10,10 +10,11 @@ #define WM_PATH "wm" -#define GET_ALPHA(color) ((color >> 24) & 0x000000FF) -#define GET_RED(color) ((color >> 16) & 0x000000FF) -#define GET_GREEN(color) ((color >> 8) & 0x000000FF) -#define GET_BLUE(color) ((color >> 0) & 0X000000FF) +#define GET_COLOR(color, n) (((color) >> ((n) << 3)) & 0xff) +#define GET_ALPHA(color) (GET_COLOR((color), 3)) +#define GET_RED(color) (GET_COLOR((color), 2)) +#define GET_GREEN(color) (GET_COLOR((color), 1)) +#define GET_BLUE(color) (GET_COLOR((color), 0)) #define COLOR_TRANSPARENT 0x00000000 #define COLOR_INVISIBLE 0x00000000 @@ -90,6 +91,7 @@ struct gfx_rect { }; struct gfx_context *gfx_new_ctx(struct gfx_context *ctx, vec2 size, u8 bpp) NONNULL; +struct gfx_context *gfx_clone(struct gfx_context *ctx) NONNULL; /** * Text stuff @@ -108,8 +110,8 @@ int gfx_font_width(enum font_type); * Image loading */ -void gfx_load_image(struct gfx_context *ctx, vec2 pos, const char *path) NONNULL; -void gfx_load_image_filter(struct gfx_context *ctx, vec2 pos, enum gfx_filter filter, +void gfx_draw_image(struct gfx_context *ctx, vec2 pos, vec2 size, const char *path) NONNULL; +void gfx_draw_image_filter(struct gfx_context *ctx, vec2 pos, vec2 size, enum gfx_filter filter, const char *path) NONNULL; void gfx_load_wallpaper(struct gfx_context *ctx, const char *path) NONNULL; @@ -121,6 +123,12 @@ void gfx_copy(struct gfx_context *dest, struct gfx_context *src, vec2 pos, vec2 void gfx_ctx_on_ctx(struct gfx_context *dest, struct gfx_context *src, vec2 pos, u8 alpha) NONNULL; /** + * Context transformations + */ + +struct gfx_context *gfx_scale(struct gfx_context *ctx, vec2 size) NONNULL; + +/** * Drawing functions */ diff --git a/libs/libgui/gui.c b/libs/libgui/gui.c index cc45bfb..1496239 100644 --- a/libs/libgui/gui.c +++ b/libs/libgui/gui.c @@ -167,7 +167,7 @@ void gui_write(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, enum f gui_error(EINVAL); } -void gui_load_image_filter(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, +void gui_draw_image_filter(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, enum gfx_filter filter, const char *path) { UNUSED(size); // TODO: Add image scaling @@ -177,17 +177,17 @@ void gui_load_image_filter(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 gui_error(ENOENT); if (layer == GUI_LAYER_BG) - gfx_load_image_filter(&widget->bg, pos, filter, path); + gfx_draw_image_filter(&widget->bg, pos, size, filter, path); else if (layer == GUI_LAYER_FG) - gfx_load_image_filter(&widget->fg, pos, filter, path); + gfx_draw_image_filter(&widget->fg, pos, size, filter, path); else gui_error(EINVAL); } -void gui_load_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, +void gui_draw_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, const char *path) { - gui_load_image_filter(win_id, widget_id, layer, pos, size, GFX_FILTER_NONE, path); + gui_draw_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, diff --git a/libs/libgui/gui.h b/libs/libgui/gui.h index c3e1d10..e148c45 100644 --- a/libs/libgui/gui.h +++ b/libs/libgui/gui.h @@ -49,9 +49,9 @@ 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); void gui_write(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, enum font_type font_type, u32 c, const char *text); -void gui_load_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, +void gui_draw_image(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, const char *path) NONNULL; -void gui_load_image_filter(u32 win_id, u32 widget_id, enum gui_layer layer, vec2 pos, vec2 size, +void gui_draw_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); |