From b317951c321051a456215cf202be586ffea33c38 Mon Sep 17 00:00:00 2001 From: Alessandro Mauri <alemauri001@gmail.com> Date: Wed, 29 Jan 2025 01:10:18 +0100 Subject: [PATCH] first working prototype of sprite drawing --- TODO | 3 +- lib/libmqoi.c3l/Makefile | 6 ++ lib/libmqoi.c3l/libmqoi.c3i | 122 ++++++++++++++++++++++++++ lib/libmqoi.c3l/manifest.json | 9 ++ lib/libmqoi.c3l/project.json | 31 +++++++ lib/libmqoi.c3l/thirdparty/mini-qoi | 1 + project.json | 16 ++-- resources/tux.qoi | Bin 0 -> 4092 bytes src/main.c3 | 70 ++++++++++----- src/ugui_atlas.c3 | 25 +++++- src/ugui_core.c3 | 6 ++ src/ugui_font.c3 | 3 +- src/ugui_sprite.c3 | 130 ++++++++++++++++++++++++++++ 13 files changed, 386 insertions(+), 36 deletions(-) create mode 100644 lib/libmqoi.c3l/Makefile create mode 100644 lib/libmqoi.c3l/libmqoi.c3i create mode 100644 lib/libmqoi.c3l/manifest.json create mode 100644 lib/libmqoi.c3l/project.json create mode 160000 lib/libmqoi.c3l/thirdparty/mini-qoi create mode 100644 resources/tux.qoi create mode 100644 src/ugui_sprite.c3 diff --git a/TODO b/TODO index 6252fdc..bf076ae 100644 --- a/TODO +++ b/TODO @@ -18,6 +18,8 @@ to maintain focus until mouse release (fix scroll bars) [ ] Animations, somehow [ ] Maybe cache codepoint converted strings [x] Fix scroll wheel when div is scrolled +[ ] Be consistent with the initialization methods some are foo.new() and some are foo.init() +[ ] Implement image loading (.bmp, .ff, .qoi and .png), in the future even lossy images like .jpg ## Layout @@ -63,4 +65,3 @@ _ border radius [ ] Text Input box [ ] Icon Buttons [ ] Switch - diff --git a/lib/libmqoi.c3l/Makefile b/lib/libmqoi.c3l/Makefile new file mode 100644 index 0000000..53230c5 --- /dev/null +++ b/lib/libmqoi.c3l/Makefile @@ -0,0 +1,6 @@ +all: thirdparty/mini-qoi/mqoi.a + cp thirdparty/mini-qoi/mqoi.a linux-x64/libmqoi.a + +thirdparty/mini-qoi/mqoi.a: + gcc -O2 -c -o thirdparty/mini-qoi/mqoi.o thirdparty/mini-qoi/src/mini_qoi.c + ar -rc thirdparty/mini-qoi/mqoi.a thirdparty/mini-qoi/mqoi.o diff --git a/lib/libmqoi.c3l/libmqoi.c3i b/lib/libmqoi.c3l/libmqoi.c3i new file mode 100644 index 0000000..330145d --- /dev/null +++ b/lib/libmqoi.c3l/libmqoi.c3i @@ -0,0 +1,122 @@ +module mqoi; + +macro rgb_hash(Rgb px) => ((px.r * 3 + px.g * 5 + px.b * 7) & 0b00111111); +macro mqoi_rgba_hash(Rgba px) => ((px.r * 3 + px.g * 5 + px.b * 7 + px.a * 11) & 0b00111111); + +const uint MQOI_HEADER_SIZE = 14; + +const char MQOI_MASK_OP_2B = 0b11000000; +const char MQOI_MASK_OP_8B = 0b11111111; + +const char MQOI_MASK_OP_LUMA_DG = 0b00111111; +const char MQOI_MASK_OP_RUN = 0b00111111; + +// basic types + +enum DescErr : uint (uint value) { + MQOI_DESC_OK = 0, // The descriptor is valid + MQOI_DESC_INVALID_MAGIC = 1, // The magic value isn't correct + MQOI_DESC_INVALID_CHANNELS = 2, // The channel number isn't valid + MQOI_DESC_INVALID_COLORSPACE = 3, // The colorspace isn't valid +} + +enum Op : char (char value) { + MQOI_OP2_INDEX = (0b00 << 6), + MQOI_OP2_DIFF = (0b01 << 6), + MQOI_OP2_LUMA = (0b10 << 6), + MQOI_OP2_RUN = (0b11 << 6), + MQOI_OP8_RUN_RGB = (0b11111110), + MQOI_OP8_RUN_RGBA = (0b11111111), +} + +enum Channels : char (char value) { + MQOI_CHANNELS_RGB = 3, + MQOI_CHANNELS_RGBA = 4, +} + +enum Colorspace : char (char value) { + MQOI_COLORSPACE_SRGB = 0, + MQOI_COLORSPACE_LINEAR = 1, +} + +union Rgb { + struct { char r, g, b; } + char[3] value; +} + +union Rgba{ + struct { char r, g, b, a; } + char[4] value; +} + +struct Desc { + char head; + + char[4] magic; + char[4] width; // big-endian width + char[4] height; // big-endian height + char channels; + char colorspace; +} + +// ==== chunks ==== + +union Chunk { + struct { + char head; + union { + Rgb rgb; + Rgba rgba; + char drdb; + } + } + char[5] value; +} + +// ==== codecs ==== + +struct Encoding { + Rgba[64] hashtable; + Rgba prev_px; + Chunk working_chunk; + char working_chunk_size; +} + +struct Dec { + Rgba[64] hashtable; + Rgba prev_px; + Chunk curr_chunk; + bitstruct : char { + char curr_chunk_head : 0..3; + char curr_chunk_size : 4..7; + } + uint pix_left; +} + +// ==== utilities ==== + +fn void u32_write(uint * n, char * dest) @extern("mqoi_u32_write"); +fn void u32_read(char * src, uint * n) @extern("mqoi_u32_read"); + +// ==== Desc ==== + +fn void desc_init(Desc* desc) @extern("mqoi_desc_init"); +fn void desc_push(Desc* desc, char byte) @extern("mqoi_desc_push"); +fn char* desc_pop(Desc* desc) @extern("mqoi_desc_pop"); +fn char desc_verify(Desc* desc, uint* w, uint* h) @extern("mqoi_desc_verify"); +fn bool desc_done(Desc* desc) @extern("mqoi_desc_done"); + +/* the encoder is still WIP +void mqoi_enc_init(mqoi_enc_t * enc); +void mqoi_enc_push(mqoi_enc_t * enc, Rgba * pix) +Chunk * mqoi_enc_pop(mqoi_enc_t * enc, char * size); +*/ + +// ==== Dec ==== + +fn void dec_init(Dec* dec, uint n_pix) @extern("mqoi_dec_init"); +fn void dec_push(Dec* dec, char byte) @extern("mqoi_dec_push"); +fn char dec_take(Dec* dec, char* bytes) @extern("mqoi_dec_take"); +fn Rgba* dec_pop(Dec* dec) @extern("mqoi_dec_pop"); +fn bool dec_done(Dec* dec) @extern("mqoi_dec_done"); + diff --git a/lib/libmqoi.c3l/manifest.json b/lib/libmqoi.c3l/manifest.json new file mode 100644 index 0000000..4043957 --- /dev/null +++ b/lib/libmqoi.c3l/manifest.json @@ -0,0 +1,9 @@ +{ + "provides" : "mqoi", + "targets" : { + "linux-x64" : { + "dependencies" : [], + "linked-libraries" : ["mqoi", "c"] + } + } +} diff --git a/lib/libmqoi.c3l/project.json b/lib/libmqoi.c3l/project.json new file mode 100644 index 0000000..78ab828 --- /dev/null +++ b/lib/libmqoi.c3l/project.json @@ -0,0 +1,31 @@ +{ + // Language version of C3. + "langrev": "1", + // Warnings used for all targets. + "warnings": [ "no-unused" ], + // Directories where C3 library files may be found. + "dependency-search-paths": [ ".." ], + // Libraries to use for all targets. + "dependencies": [ "mqoi" ], + // Authors, optionally with email. + "authors": [ "Alessandro Mauri <alemauri001@gmail.com" ], + // Version using semantic versioning. + "version": "0.1.0", + // Sources compiled for all targets. + "sources": [ ], + // C sources if the project also compiles C sources + // relative to the project file. + // "c-sources": [ "csource/**" ], + // Output location, relative to project file. + "output": "build", + // Architecture and OS target. + // You can use 'c3c --list-targets' to list all valid targets. + // "target": "windows-x64", + "features": [], + // Global settings. + // CPU name, used for optimizations in the LLVM backend. + "cpu": "generic", + // Optimization: "O0", "O1", "O2", "O3", "O4", "O5", "Os", "Oz". + "opt": "O0", + // See resources/examples/project_all_settings.json and 'c3c --list-project-properties' to see more properties. +} diff --git a/lib/libmqoi.c3l/thirdparty/mini-qoi b/lib/libmqoi.c3l/thirdparty/mini-qoi new file mode 160000 index 0000000..f8b5a8a --- /dev/null +++ b/lib/libmqoi.c3l/thirdparty/mini-qoi @@ -0,0 +1 @@ +Subproject commit f8b5a8a4d2eeee68b52e143060b7412ba78b5077 diff --git a/project.json b/project.json index 8922226..0101175 100644 --- a/project.json +++ b/project.json @@ -2,11 +2,11 @@ // Language version of C3. "langrev": "1", // Warnings used for all targets. - "warnings": [ "no-unused" ], + "warnings": ["no-unused"], // Directories where C3 library files may be found. - "dependency-search-paths": [ "lib" ], + "dependency-search-paths": ["lib"], // Libraries to use for all targets. - "dependencies": [ "raylib", "schrift", "grapheme" ], + "dependencies": ["raylib", "schrift", "grapheme", "mqoi"], "features": [ // See rcore.c3 //"SUPPORT_INTERNAL_MEMORY_MANAGEMENT", @@ -22,11 +22,11 @@ //"RAYGUI_CUSTOM_ICONS", ], // Authors, optionally with email. - "authors": [ "Alessandro Mauri <ale@shitposting.expert>" ], + "authors": ["Alessandro Mauri <ale@shitposting.expert>"], // Version using semantic versioning. "version": "0.1.0", // Sources compiled for all targets. - "sources": [ "src/**" ], + "sources": ["src/**"], // C sources if the project also compiles C sources // relative to the project file. // "c-sources": [ "csource/**" ], @@ -41,15 +41,15 @@ "targets": { "ugui": { // Executable or library. - "type": "executable", + "type": "executable" // Additional libraries, sources // and overrides of global settings here. - }, + } }, // Global settings. // CPU name, used for optimizations in the LLVM backend. "cpu": "generic", // Optimization: "O0", "O1", "O2", "O3", "O4", "O5", "Os", "Oz". - "opt": "O0", + "opt": "O0" // See resources/examples/project_all_settings.json and 'c3c --list-project-properties' to see more properties. } diff --git a/resources/tux.qoi b/resources/tux.qoi new file mode 100644 index 0000000000000000000000000000000000000000..57d6333e7db058c1037e408d0b58459c73dc8b2c GIT binary patch literal 4092 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 literal 0 HcmV?d00001 diff --git a/src/main.c3 b/src/main.c3 index 7a88495..a5cad6f 100644 --- a/src/main.c3 +++ b/src/main.c3 @@ -81,6 +81,8 @@ fn int main(String[] args) ugui::Ctx ui; ui.init()!!; ui.load_font("font1", "resources/hack-nerd.ttf", 16)!!; + ui.sprite_atlas_create("icons", AtlasType.ATLAS_RGBA32, 512, 512)!!; + ui.import_sprite_file_qoi("tux", "resources/tux.qoi")!!; short width = 800; short height = 450; @@ -99,8 +101,11 @@ fn int main(String[] args) // font stuff rl::Shader font_shader = rl::load_shader_from_memory(null, FONT_FS); rl::Image font_atlas; + rl::Image sprite_atlas; rl::Texture2D font_texture; + rl::Texture2D sprite_texture; ugui::Id font_id = ui.get_font_id("font1"); + ugui::Id sprite_id = ui.get_sprite_atlas_id("icons"); // Main loop while (!rl::window_should_close()) { @@ -164,7 +169,7 @@ fn int main(String[] args) if (ui.button("button2", ugui::Rect{0,0,30,30})!!.mouse_release) { io::printn("release button2"); } - + ui.layout_set_row()!!; ui.layout_next_row()!!; static float rf, gf, bf, af; @@ -179,6 +184,7 @@ fn int main(String[] args) ui.layout_next_column()!!; ui.button_label("Continua!")!!; |}; + ui.draw_sprite("tux")!!; ui.div_end()!!; ui.div_begin("second", ugui::DIV_FILL, scroll_x: true, scroll_y: true)!!; @@ -205,7 +211,7 @@ fn int main(String[] args) ui.slider_hor("hs2", ugui::Rect{0,0,100,30}, &f2)!!; |}; ui.div_end()!!; - + // Timings counter TimeStats dts = draw_times.get_stats(); TimeStats uts = ui_times.get_stats(); @@ -237,27 +243,48 @@ fn int main(String[] args) float roundness = r.w > r.h ? (2.1f*rad)/(float)r.h : (2.1f*rad)/(float)r.w; rl::draw_rectangle_rounded(cmd.rect.rect.conv(), roundness, 0, cmd.rect.color.conv()); case ugui::CmdType.CMD_UPDATE_ATLAS: - if (cmd.update_atlas.id != font_id) { break; } - //rl::unload_image(font_atlas); - font_atlas.data = cmd.update_atlas.raw_buffer; - font_atlas.width = cmd.update_atlas.width; - font_atlas.height = cmd.update_atlas.height; - font_atlas.mipmaps = 1; - //font_atlas.format = rl::PixelFormat.PIXELFORMAT_UNCOMPRESSED_GRAYSCALE; - font_atlas.format = 1; - if (rl::is_texture_ready(font_texture)) { - rl::unload_texture(font_texture); + if (cmd.update_atlas.id == font_id) { + //rl::unload_image(font_atlas); + font_atlas.data = cmd.update_atlas.raw_buffer; + font_atlas.width = cmd.update_atlas.width; + font_atlas.height = cmd.update_atlas.height; + font_atlas.mipmaps = 1; + //font_atlas.format = rl::PixelFormat.PIXELFORMAT_UNCOMPRESSED_GRAYSCALE; + font_atlas.format = 1; + if (rl::is_texture_ready(font_texture)) { + rl::unload_texture(font_texture); + } + font_texture = rl::load_texture_from_image(font_atlas); + } else if (cmd.update_atlas.id == sprite_id) { + sprite_atlas.data = cmd.update_atlas.raw_buffer; + sprite_atlas.width = cmd.update_atlas.width; + sprite_atlas.height = cmd.update_atlas.height; + sprite_atlas.mipmaps = 1; + //sprite_atlas.format = rl::PixelFormat.PIXELFORMAT_UNCOMPRESSED_R8G8B8A8; + sprite_atlas.format = 7; + if (rl::is_texture_ready(sprite_texture)) { + rl::unload_texture(sprite_texture); + } + sprite_texture = rl::load_texture_from_image(sprite_atlas); } - font_texture = rl::load_texture_from_image(font_atlas); case ugui::CmdType.CMD_SPRITE: - if (cmd.sprite.texture_id != font_id) { break; } - rl::Vector2 position = { - .x = cmd.sprite.rect.x, - .y = cmd.sprite.rect.y, - }; - rl::begin_shader_mode(font_shader); - rl::draw_texture_rec(font_texture, cmd.sprite.texture_rect.conv(), position, cmd.sprite.hue.conv()); - rl::end_shader_mode(); + if (cmd.sprite.texture_id == font_id) { + rl::Vector2 position = { + .x = cmd.sprite.rect.x, + .y = cmd.sprite.rect.y, + }; + rl::begin_shader_mode(font_shader); + rl::draw_texture_rec(font_texture, cmd.sprite.texture_rect.conv(), position, cmd.sprite.hue.conv()); + rl::end_shader_mode(); + } else if (cmd.sprite.texture_id == sprite_id) { + rl::Vector2 position = { + .x = cmd.sprite.rect.x, + .y = cmd.sprite.rect.y, + }; + rl::draw_texture_rec(sprite_texture, cmd.sprite.texture_rect.conv(), position, cmd.sprite.hue.conv()); + } else { + io::printfn("unknown texture id: %d", cmd.sprite.texture_id); + } case ugui::CmdType.CMD_SCISSOR: if (cmd.scissor.rect.w == 0 && cmd.scissor.rect.h == 0) { rl::end_scissor_mode(); @@ -272,7 +299,6 @@ fn int main(String[] args) //draw_times.print_stats(); rl::end_drawing(); /* End Drawing */ - } rl::close_window(); diff --git a/src/ugui_atlas.c3 b/src/ugui_atlas.c3 index 0f04c08..1976ec4 100644 --- a/src/ugui_atlas.c3 +++ b/src/ugui_atlas.c3 @@ -9,6 +9,7 @@ fault UgAtlasError { enum AtlasType { ATLAS_GRAYSCALE, + ATLAS_RGBA32, } // black and white atlas @@ -28,6 +29,7 @@ macro usz AtlasType.bpp(type) { switch (type) { case ATLAS_GRAYSCALE: return 1; + case ATLAS_RGBA32: return 4; } } @@ -46,10 +48,25 @@ fn void Atlas.free(&atlas) free(atlas.buffer); } + +/* + * pixels -> +--------------+-----+ + * | | | h + * | | | e + * | | | i + * | | | g + * | | | h + * | | | t + * +--------------+-----+ + * |<--- width -->| + * |<----- stride ----->| + * bytes per pixels are inferred and have to be the same + * as the atlas type + */ // place a rect inside the atlas // uses a row first algorithm // TODO: use a skyline algorithm https://jvernay.fr/en/blog/skyline-2d-packer/implementation/ -fn Point! Atlas.place(&atlas, char[] pixels, ushort w, ushort h) +fn Point! Atlas.place(&atlas, char[] pixels, ushort w, ushort h, ushort stride) { Point p; @@ -66,11 +83,11 @@ fn Point! Atlas.place(&atlas, char[] pixels, ushort w, ushort h) } } + usz bpp = atlas.type.bpp(); for (usz y = 0; y < h; y++) { for (usz x = 0; x < w; x++) { - char[] buf = atlas.buffer[(usz)(p.y+y)*atlas.width + (p.x+x) ..]; - char[] pix = pixels[y*w + x ..]; - usz bpp = atlas.type.bpp(); + char[] buf = atlas.buffer[(usz)(p.y+y)*atlas.width*bpp + (p.x+x)*bpp ..]; + char[] pix = pixels[(usz)y*stride*bpp + x*bpp ..]; buf[0..bpp-1] = pix[0..bpp-1]; } diff --git a/src/ugui_core.c3 b/src/ugui_core.c3 index 52f9fb4..e2b9a0e 100644 --- a/src/ugui_core.c3 +++ b/src/ugui_core.c3 @@ -95,6 +95,7 @@ struct Ctx { ushort width, height; Style style; Font font; + SpriteAtlas sprite_atlas; bool has_focus; struct input { @@ -220,6 +221,7 @@ fn void Ctx.free(&ctx) (void)ctx.cache.free(); (void)ctx.cmd_queue.free(); (void)ctx.font.free(); + (void)ctx.sprite_atlas.free(); } fn void! Ctx.frame_begin(&ctx) @@ -281,6 +283,10 @@ fn void! Ctx.frame_end(&ctx) ctx.push_update_atlas(&ctx.font.atlas)!; ctx.font.should_update = false; } + if (ctx.sprite_atlas.should_update) { + ctx.push_update_atlas(&ctx.sprite_atlas.atlas)!; + ctx.sprite_atlas.should_update = false; + } $if 1: // draw mouse position diff --git a/src/ugui_font.c3 b/src/ugui_font.c3 index 189fdef..546c7b9 100644 --- a/src/ugui_font.c3 +++ b/src/ugui_font.c3 @@ -144,7 +144,7 @@ fn Glyph*! Font.get_glyph(&font, Codepoint code) //io::printfn("code=%c, w=%d, h=%d, ox=%d, oy=%d, adv=%d", // glyph.code, glyph.w, glyph.h, glyph.ox, glyph.oy, glyph.adv); - Point uv = font.atlas.place(pixels, glyph.w, glyph.h)!; + Point uv = font.atlas.place(pixels, glyph.w, glyph.h, (ushort)img.width)!; glyph.u = uv.x; glyph.v = uv.y; @@ -159,6 +159,7 @@ fn Glyph*! Font.get_glyph(&font, Codepoint code) fn void Font.free(&font) { font.atlas.free(); + font.table.free(); schrift::freefont(font.sft.font); } diff --git a/src/ugui_sprite.c3 b/src/ugui_sprite.c3 new file mode 100644 index 0000000..aff0649 --- /dev/null +++ b/src/ugui_sprite.c3 @@ -0,0 +1,130 @@ +module ugui; + +import std::collections::map; +import std::io; +import mqoi; + +const usz SRITES_PER_ATLAS = 64; + +struct Sprite { + Id id; + ushort u, v; + ushort w, h; +} + +def SpriteMap = map::HashMap(<Id, Sprite>); + +struct SpriteAtlas { + Id id; + Atlas atlas; + SpriteMap sprites; + bool should_update; +} + + +// name: some examples are "icons" or "images" +fn void! SpriteAtlas.init(&this, String name, AtlasType type, ushort width, ushort height) +{ + // FIXME: for now only RGBA32 format is supported + if (type != ATLAS_RGBA32) { + return UgAtlasError.INVALID_TYPE?; + } + + this.id = name.hash(); + this.atlas.new(this.id, AtlasType.ATLAS_RGBA32, width, height)!; + this.sprites.new_init(capacity: SRITES_PER_ATLAS); + this.should_update = false; +} + +fn void! SpriteAtlas.free(&this) +{ + this.atlas.free(); + this.sprites.free(); +} + +// FIXME: this should throw an error when a different pixel format than the atlas' is used +fn Sprite*! SpriteAtlas.insert(&this, String name, char[] pixels, ushort w, ushort h, ushort stride) +{ + Sprite s; + s.id = name.hash(); + Point uv = this.atlas.place(pixels, w, h, stride)!; + s.w = w; + s.h = h; + s.u = uv.x; + s.v = uv.y; + this.sprites.set(s.id, s); + this.should_update = true; + return this.sprites.get_ref(s.id); +} + +fn Sprite*! SpriteAtlas.get(&this, String name) +{ + Id id = name.hash(); + return this.sprites.get_ref(id); +} + +fn Sprite*! SpriteAtlas.get_by_id(&this, Id id) +{ + return this.sprites.get_ref(id); +} + +fn void! Ctx.sprite_atlas_create(&ctx, String name, AtlasType type, ushort w, ushort h) +{ + ctx.sprite_atlas.init(name, type, w, h)!; +} + +fn Id Ctx.get_sprite_atlas_id(&ctx, String name) +{ + return name.hash(); +} + +fn void! Ctx.import_sprite_memory(&ctx, String name, char[] pixels, ushort w, ushort h, ushort stride) +{ + ctx.sprite_atlas.insert(name, pixels, w, h, stride)!; +} + +fn void! Ctx.import_sprite_file_qoi(&ctx, String name, String path) +{ + mqoi::Desc image_desc; + uint w, h; + + File file = file::open(path, "rb")!; + defer (void) file.close(); + + while (!mqoi::desc_done(&image_desc)) { + mqoi::desc_push(&image_desc, file.read_byte()!); + } + if (mqoi::desc_verify(&image_desc, &w, &h) != 0) { + return IoError.FILE_NOT_VALID?; + } + + mqoi::Dec dec; + mqoi::Rgba* px; + + usz idx; + char[] pixels = mem::new_array(char, (usz)w*h*4); + defer mem::free(pixels); + + mqoi::dec_init(&dec, w*h); + while (!mqoi::dec_done(&dec)) { + mqoi::dec_push(&dec, file.read_byte()!); + + while ((px = mqoi::dec_pop(&dec)) != null) { + pixels[idx..idx+3] = px.value; + idx += 4; + } + } + + ctx.sprite_atlas.insert(name, pixels, (ushort)w, (ushort)h, (ushort)w)!; +} + +// FIXME: test function, very different from every other function here +fn void! Ctx.draw_sprite(&ctx, String name) +{ + Sprite* sprite = ctx.sprite_atlas.get(name)!; + Rect bounds = { 100, 100, sprite.w, sprite.h }; + Rect uv = { sprite.u, sprite.v, sprite.w, sprite.h }; + Id tex_id = ctx.sprite_atlas.id; + + return ctx.push_sprite(bounds, uv, tex_id)!; +}