2025-11-13 07:15:41 -06:00
|
|
|
#include "pxl8_tilesheet.h"
|
|
|
|
|
|
2025-09-24 00:39:44 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
2025-10-01 12:56:13 -05:00
|
|
|
#include "pxl8_ase.h"
|
2025-11-28 14:41:35 -06:00
|
|
|
#include "pxl8_color.h"
|
2025-10-01 12:56:13 -05:00
|
|
|
#include "pxl8_macros.h"
|
|
|
|
|
#include "pxl8_tilemap.h"
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
struct pxl8_tilesheet {
|
|
|
|
|
u8* data;
|
|
|
|
|
bool* tile_valid;
|
|
|
|
|
|
|
|
|
|
u32 height;
|
|
|
|
|
u32 tile_size;
|
|
|
|
|
u32 tiles_per_row;
|
|
|
|
|
u32 total_tiles;
|
|
|
|
|
u32 width;
|
|
|
|
|
|
2025-11-28 23:42:57 -06:00
|
|
|
pxl8_pixel_mode pixel_mode;
|
2025-10-04 04:13:48 -05:00
|
|
|
u32 ref_count;
|
|
|
|
|
|
|
|
|
|
pxl8_tile_animation* animations;
|
|
|
|
|
u32 animation_count;
|
|
|
|
|
|
|
|
|
|
pxl8_tile_properties* properties;
|
|
|
|
|
|
|
|
|
|
pxl8_autotile_rule** autotile_rules;
|
|
|
|
|
u32* autotile_rule_counts;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
pxl8_tilesheet* pxl8_tilesheet_create(u32 tile_size) {
|
|
|
|
|
pxl8_tilesheet* tilesheet = calloc(1, sizeof(pxl8_tilesheet));
|
|
|
|
|
if (!tilesheet) return NULL;
|
|
|
|
|
|
|
|
|
|
tilesheet->tile_size = tile_size;
|
|
|
|
|
tilesheet->ref_count = 1;
|
|
|
|
|
|
|
|
|
|
return tilesheet;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pxl8_tilesheet_destroy(pxl8_tilesheet* tilesheet) {
|
2025-09-24 00:39:44 -05:00
|
|
|
if (!tilesheet) return;
|
|
|
|
|
|
|
|
|
|
if (tilesheet->data) {
|
|
|
|
|
free(tilesheet->data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tilesheet->tile_valid) {
|
|
|
|
|
free(tilesheet->tile_valid);
|
|
|
|
|
}
|
2025-09-27 11:03:36 -05:00
|
|
|
|
|
|
|
|
if (tilesheet->animations) {
|
|
|
|
|
for (u32 i = 0; i < tilesheet->animation_count; i++) {
|
|
|
|
|
if (tilesheet->animations[i].frames) {
|
|
|
|
|
free(tilesheet->animations[i].frames);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(tilesheet->animations);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tilesheet->properties) {
|
|
|
|
|
free(tilesheet->properties);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tilesheet->autotile_rules) {
|
|
|
|
|
for (u32 i = 0; i <= tilesheet->total_tiles; i++) {
|
|
|
|
|
if (tilesheet->autotile_rules[i]) {
|
|
|
|
|
free(tilesheet->autotile_rules[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(tilesheet->autotile_rules);
|
|
|
|
|
free(tilesheet->autotile_rule_counts);
|
|
|
|
|
}
|
2025-10-04 04:13:48 -05:00
|
|
|
|
|
|
|
|
free(tilesheet);
|
2025-09-24 00:39:44 -05:00
|
|
|
}
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
pxl8_result pxl8_tilesheet_load(pxl8_tilesheet* tilesheet, const char* filepath, pxl8_gfx* gfx) {
|
2025-09-24 00:39:44 -05:00
|
|
|
if (!tilesheet || !filepath || !gfx) return PXL8_ERROR_NULL_POINTER;
|
|
|
|
|
|
|
|
|
|
pxl8_ase_file ase_file;
|
|
|
|
|
pxl8_result result = pxl8_ase_load(filepath, &ase_file);
|
|
|
|
|
if (result != PXL8_OK) {
|
|
|
|
|
pxl8_error("Failed to load tilesheet: %s", filepath);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tilesheet->data) {
|
|
|
|
|
free(tilesheet->data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u32 width = ase_file.header.width;
|
|
|
|
|
u32 height = ase_file.header.height;
|
|
|
|
|
|
|
|
|
|
if (ase_file.header.grid_width > 0 && ase_file.header.grid_height > 0) {
|
|
|
|
|
tilesheet->tile_size = ase_file.header.grid_width;
|
|
|
|
|
pxl8_info("Using Aseprite grid size: %dx%d",
|
|
|
|
|
ase_file.header.grid_width, ase_file.header.grid_height);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tilesheet->width = width;
|
|
|
|
|
tilesheet->height = height;
|
|
|
|
|
tilesheet->tiles_per_row = width / tilesheet->tile_size;
|
|
|
|
|
tilesheet->total_tiles = (width / tilesheet->tile_size) * (height / tilesheet->tile_size);
|
2025-11-28 23:42:57 -06:00
|
|
|
tilesheet->pixel_mode = pxl8_gfx_get_pixel_mode(gfx);
|
2025-09-24 00:39:44 -05:00
|
|
|
|
2025-11-28 14:41:35 -06:00
|
|
|
u32 pixel_count = width * height;
|
|
|
|
|
u16 ase_depth = ase_file.header.color_depth;
|
2025-11-28 23:42:57 -06:00
|
|
|
bool gfx_hicolor = (tilesheet->pixel_mode == PXL8_PIXEL_HICOLOR);
|
2025-09-24 00:39:44 -05:00
|
|
|
|
2025-11-28 23:42:57 -06:00
|
|
|
size_t data_size = pixel_count * pxl8_bytes_per_pixel(tilesheet->pixel_mode);
|
2025-09-24 00:39:44 -05:00
|
|
|
tilesheet->data = malloc(data_size);
|
|
|
|
|
if (!tilesheet->data) {
|
2025-10-04 04:13:48 -05:00
|
|
|
pxl8_ase_destroy(&ase_file);
|
2025-09-24 00:39:44 -05:00
|
|
|
return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ase_file.frame_count > 0 && ase_file.frames[0].pixels) {
|
2025-11-28 14:41:35 -06:00
|
|
|
const u8* src = ase_file.frames[0].pixels;
|
|
|
|
|
|
|
|
|
|
if (ase_depth == 8 && !gfx_hicolor) {
|
|
|
|
|
memcpy(tilesheet->data, src, pixel_count);
|
|
|
|
|
} else if (ase_depth == 32 && gfx_hicolor) {
|
|
|
|
|
u16* dst = (u16*)tilesheet->data;
|
|
|
|
|
const u32* rgba = (const u32*)src;
|
|
|
|
|
for (u32 i = 0; i < pixel_count; i++) {
|
|
|
|
|
u32 c = rgba[i];
|
|
|
|
|
u8 a = (c >> 24) & 0xFF;
|
|
|
|
|
if (a == 0) {
|
|
|
|
|
dst[i] = 0;
|
|
|
|
|
} else {
|
|
|
|
|
dst[i] = pxl8_rgba32_to_rgb565(c);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (ase_depth == 8 && gfx_hicolor) {
|
|
|
|
|
pxl8_warn("Indexed ASE with hicolor gfx - storing as indexed");
|
2025-11-28 23:42:57 -06:00
|
|
|
tilesheet->pixel_mode = PXL8_PIXEL_INDEXED;
|
2025-11-28 14:41:35 -06:00
|
|
|
u8* new_data = realloc(tilesheet->data, pixel_count);
|
|
|
|
|
if (!new_data) {
|
|
|
|
|
free(tilesheet->data);
|
|
|
|
|
tilesheet->data = NULL;
|
|
|
|
|
pxl8_ase_destroy(&ase_file);
|
|
|
|
|
return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
tilesheet->data = new_data;
|
|
|
|
|
memcpy(tilesheet->data, src, pixel_count);
|
|
|
|
|
} else {
|
|
|
|
|
pxl8_error("Unsupported ASE color depth %d for gfx mode", ase_depth);
|
|
|
|
|
free(tilesheet->data);
|
|
|
|
|
tilesheet->data = NULL;
|
|
|
|
|
pxl8_ase_destroy(&ase_file);
|
|
|
|
|
return PXL8_ERROR_INVALID_ARGUMENT;
|
|
|
|
|
}
|
2025-09-24 00:39:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tilesheet->tile_valid = calloc(tilesheet->total_tiles + 1, sizeof(bool));
|
|
|
|
|
if (!tilesheet->tile_valid) {
|
|
|
|
|
free(tilesheet->data);
|
|
|
|
|
tilesheet->data = NULL;
|
2025-10-04 04:13:48 -05:00
|
|
|
pxl8_ase_destroy(&ase_file);
|
2025-09-24 00:39:44 -05:00
|
|
|
return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u32 valid_tiles = 0;
|
2025-11-28 23:42:57 -06:00
|
|
|
bool is_hicolor = (tilesheet->pixel_mode == PXL8_PIXEL_HICOLOR);
|
2025-09-24 00:39:44 -05:00
|
|
|
|
|
|
|
|
for (u32 tile_id = 1; tile_id <= tilesheet->total_tiles; tile_id++) {
|
|
|
|
|
u32 tile_x = ((tile_id - 1) % tilesheet->tiles_per_row) * tilesheet->tile_size;
|
|
|
|
|
u32 tile_y = ((tile_id - 1) / tilesheet->tiles_per_row) * tilesheet->tile_size;
|
|
|
|
|
|
|
|
|
|
bool has_content = false;
|
|
|
|
|
for (u32 py = 0; py < tilesheet->tile_size && !has_content; py++) {
|
|
|
|
|
for (u32 px = 0; px < tilesheet->tile_size; px++) {
|
2025-11-28 14:41:35 -06:00
|
|
|
u32 idx = (tile_y + py) * width + (tile_x + px);
|
2025-09-24 00:39:44 -05:00
|
|
|
if (is_hicolor) {
|
2025-11-28 14:41:35 -06:00
|
|
|
if (((u16*)tilesheet->data)[idx] != 0) {
|
2025-09-24 00:39:44 -05:00
|
|
|
has_content = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (tilesheet->data[idx] != 0) {
|
|
|
|
|
has_content = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (has_content) {
|
|
|
|
|
tilesheet->tile_valid[tile_id] = true;
|
|
|
|
|
valid_tiles++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pxl8_info("Loaded tilesheet %s: %dx%d, %d valid tiles out of %d slots (%dx%d each)",
|
|
|
|
|
filepath, width, height, valid_tiles, tilesheet->total_tiles,
|
|
|
|
|
tilesheet->tile_size, tilesheet->tile_size);
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
pxl8_ase_destroy(&ase_file);
|
2025-09-24 00:39:44 -05:00
|
|
|
return PXL8_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
void pxl8_tilesheet_render_tile(const pxl8_tilesheet* tilesheet, pxl8_gfx* gfx,
|
2025-09-24 00:39:44 -05:00
|
|
|
u16 tile_id, i32 x, i32 y, u8 flags) {
|
|
|
|
|
if (!tilesheet || !gfx || !tilesheet->data) return;
|
|
|
|
|
if (tile_id == 0 || tile_id > tilesheet->total_tiles) return;
|
|
|
|
|
if (tilesheet->tile_valid && !tilesheet->tile_valid[tile_id]) return;
|
|
|
|
|
|
|
|
|
|
u32 tile_x = ((tile_id - 1) % tilesheet->tiles_per_row) * tilesheet->tile_size;
|
|
|
|
|
u32 tile_y = ((tile_id - 1) / tilesheet->tiles_per_row) * tilesheet->tile_size;
|
|
|
|
|
|
|
|
|
|
for (u32 py = 0; py < tilesheet->tile_size; py++) {
|
|
|
|
|
for (u32 px = 0; px < tilesheet->tile_size; px++) {
|
|
|
|
|
u32 src_x = tile_x + px;
|
|
|
|
|
u32 src_y = tile_y + py;
|
|
|
|
|
|
|
|
|
|
if (flags & PXL8_TILE_FLIP_X) src_x = tile_x + (tilesheet->tile_size - 1 - px);
|
|
|
|
|
if (flags & PXL8_TILE_FLIP_Y) src_y = tile_y + (tilesheet->tile_size - 1 - py);
|
|
|
|
|
|
|
|
|
|
u32 src_idx = src_y * tilesheet->width + src_x;
|
|
|
|
|
u8 color_idx = tilesheet->data[src_idx];
|
|
|
|
|
|
|
|
|
|
if (color_idx != 0) {
|
|
|
|
|
i32 screen_x = x + px;
|
|
|
|
|
i32 screen_y = y + py;
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
if (screen_x >= 0 && screen_x < pxl8_gfx_get_width(gfx) &&
|
|
|
|
|
screen_y >= 0 && screen_y < pxl8_gfx_get_height(gfx)) {
|
2025-09-24 00:39:44 -05:00
|
|
|
pxl8_pixel(gfx, screen_x, screen_y, color_idx);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool pxl8_tilesheet_is_tile_valid(const pxl8_tilesheet* tilesheet, u16 tile_id) {
|
|
|
|
|
if (!tilesheet || tile_id == 0 || tile_id > tilesheet->total_tiles) return false;
|
|
|
|
|
return tilesheet->tile_valid ? tilesheet->tile_valid[tile_id] : true;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-27 11:03:36 -05:00
|
|
|
void pxl8_tilesheet_ref(pxl8_tilesheet* tilesheet) {
|
|
|
|
|
if (!tilesheet) return;
|
|
|
|
|
tilesheet->ref_count++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pxl8_tilesheet_unref(pxl8_tilesheet* tilesheet) {
|
|
|
|
|
if (!tilesheet) return;
|
|
|
|
|
|
|
|
|
|
if (--tilesheet->ref_count == 0) {
|
2025-10-04 04:13:48 -05:00
|
|
|
pxl8_tilesheet_destroy(tilesheet);
|
2025-09-27 11:03:36 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pxl8_result pxl8_tilesheet_add_animation(pxl8_tilesheet* tilesheet, u16 base_tile_id,
|
|
|
|
|
const u16* frames, u16 frame_count, f32 frame_duration) {
|
|
|
|
|
if (!tilesheet || !frames || frame_count == 0) return PXL8_ERROR_INVALID_ARGUMENT;
|
|
|
|
|
if (base_tile_id == 0 || base_tile_id > tilesheet->total_tiles) return PXL8_ERROR_INVALID_ARGUMENT;
|
|
|
|
|
|
|
|
|
|
if (!tilesheet->animations) {
|
|
|
|
|
tilesheet->animations = calloc(tilesheet->total_tiles + 1, sizeof(pxl8_tile_animation));
|
|
|
|
|
if (!tilesheet->animations) return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pxl8_tile_animation* anim = &tilesheet->animations[base_tile_id];
|
|
|
|
|
if (anim->frames) free(anim->frames);
|
|
|
|
|
|
|
|
|
|
anim->frames = malloc(frame_count * sizeof(u16));
|
|
|
|
|
if (!anim->frames) return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
|
|
memcpy(anim->frames, frames, frame_count * sizeof(u16));
|
|
|
|
|
anim->frame_count = frame_count;
|
|
|
|
|
anim->current_frame = 0;
|
|
|
|
|
anim->frame_duration = frame_duration;
|
|
|
|
|
anim->time_accumulator = 0;
|
|
|
|
|
|
|
|
|
|
tilesheet->animation_count++;
|
|
|
|
|
return PXL8_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pxl8_tilesheet_update_animations(pxl8_tilesheet* tilesheet, f32 delta_time) {
|
|
|
|
|
if (!tilesheet || !tilesheet->animations) return;
|
|
|
|
|
|
|
|
|
|
for (u32 i = 1; i <= tilesheet->total_tiles; i++) {
|
|
|
|
|
pxl8_tile_animation* anim = &tilesheet->animations[i];
|
|
|
|
|
if (!anim->frames || anim->frame_count == 0) continue;
|
|
|
|
|
|
|
|
|
|
anim->time_accumulator += delta_time;
|
|
|
|
|
while (anim->time_accumulator >= anim->frame_duration) {
|
|
|
|
|
anim->time_accumulator -= anim->frame_duration;
|
|
|
|
|
anim->current_frame = (anim->current_frame + 1) % anim->frame_count;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u16 pxl8_tilesheet_get_animated_frame(const pxl8_tilesheet* tilesheet, u16 tile_id) {
|
|
|
|
|
if (!tilesheet || !tilesheet->animations || tile_id == 0 || tile_id > tilesheet->total_tiles) {
|
|
|
|
|
return tile_id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pxl8_tile_animation* anim = &tilesheet->animations[tile_id];
|
|
|
|
|
if (!anim->frames || anim->frame_count == 0) return tile_id;
|
|
|
|
|
|
|
|
|
|
return anim->frames[anim->current_frame];
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-01 12:39:59 -05:00
|
|
|
pxl8_result pxl8_tilesheet_set_tile_pixels(pxl8_tilesheet* tilesheet, u16 tile_id, const u8* pixels) {
|
|
|
|
|
if (!tilesheet || !pixels || tile_id == 0 || tile_id > tilesheet->total_tiles) return PXL8_ERROR_INVALID_ARGUMENT;
|
|
|
|
|
if (!tilesheet->data) return PXL8_ERROR_NULL_POINTER;
|
|
|
|
|
|
|
|
|
|
u32 tile_x = (tile_id - 1) % tilesheet->tiles_per_row;
|
|
|
|
|
u32 tile_y = (tile_id - 1) / tilesheet->tiles_per_row;
|
2025-11-28 23:42:57 -06:00
|
|
|
u32 bytes_per_pixel = pxl8_bytes_per_pixel(tilesheet->pixel_mode);
|
2025-11-01 12:39:59 -05:00
|
|
|
|
|
|
|
|
for (u32 py = 0; py < tilesheet->tile_size; py++) {
|
|
|
|
|
for (u32 px = 0; px < tilesheet->tile_size; px++) {
|
|
|
|
|
u32 src_idx = py * tilesheet->tile_size + px;
|
|
|
|
|
u32 dst_x = tile_x * tilesheet->tile_size + px;
|
|
|
|
|
u32 dst_y = tile_y * tilesheet->tile_size + py;
|
2025-11-28 14:41:35 -06:00
|
|
|
u32 dst_idx = dst_y * tilesheet->width + dst_x;
|
2025-11-01 12:39:59 -05:00
|
|
|
|
2025-11-28 14:41:35 -06:00
|
|
|
if (bytes_per_pixel == 2) {
|
|
|
|
|
((u16*)tilesheet->data)[dst_idx] = ((const u16*)pixels)[src_idx];
|
2025-11-01 12:39:59 -05:00
|
|
|
} else {
|
|
|
|
|
tilesheet->data[dst_idx] = pixels[src_idx];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tilesheet->tile_valid) {
|
|
|
|
|
tilesheet->tile_valid[tile_id] = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return PXL8_OK;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-27 11:03:36 -05:00
|
|
|
void pxl8_tilesheet_set_tile_property(pxl8_tilesheet* tilesheet, u16 tile_id,
|
|
|
|
|
const pxl8_tile_properties* props) {
|
|
|
|
|
if (!tilesheet || !props || tile_id == 0 || tile_id > tilesheet->total_tiles) return;
|
|
|
|
|
|
|
|
|
|
if (!tilesheet->properties) {
|
|
|
|
|
tilesheet->properties = calloc(tilesheet->total_tiles + 1, sizeof(pxl8_tile_properties));
|
|
|
|
|
if (!tilesheet->properties) return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tilesheet->properties[tile_id] = *props;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pxl8_tile_properties* pxl8_tilesheet_get_tile_property(const pxl8_tilesheet* tilesheet, u16 tile_id) {
|
|
|
|
|
if (!tilesheet || !tilesheet->properties || tile_id == 0 || tile_id > tilesheet->total_tiles) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
return &tilesheet->properties[tile_id];
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-04 04:13:48 -05:00
|
|
|
u32 pxl8_tilesheet_get_tile_size(const pxl8_tilesheet* tilesheet) {
|
|
|
|
|
return tilesheet ? tilesheet->tile_size : 0;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-27 11:03:36 -05:00
|
|
|
pxl8_result pxl8_tilesheet_add_autotile_rule(pxl8_tilesheet* tilesheet, u16 base_tile_id,
|
|
|
|
|
u8 neighbor_mask, u16 result_tile_id) {
|
|
|
|
|
if (!tilesheet || base_tile_id == 0 || base_tile_id > tilesheet->total_tiles) {
|
|
|
|
|
return PXL8_ERROR_INVALID_ARGUMENT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!tilesheet->autotile_rules) {
|
|
|
|
|
tilesheet->autotile_rules = calloc(tilesheet->total_tiles + 1, sizeof(pxl8_autotile_rule*));
|
|
|
|
|
tilesheet->autotile_rule_counts = calloc(tilesheet->total_tiles + 1, sizeof(u32));
|
|
|
|
|
if (!tilesheet->autotile_rules || !tilesheet->autotile_rule_counts) {
|
|
|
|
|
return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u32 count = tilesheet->autotile_rule_counts[base_tile_id];
|
|
|
|
|
pxl8_autotile_rule* new_rules = realloc(tilesheet->autotile_rules[base_tile_id],
|
|
|
|
|
(count + 1) * sizeof(pxl8_autotile_rule));
|
|
|
|
|
if (!new_rules) return PXL8_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
|
|
new_rules[count].neighbor_mask = neighbor_mask;
|
|
|
|
|
new_rules[count].tile_id = result_tile_id;
|
|
|
|
|
|
|
|
|
|
tilesheet->autotile_rules[base_tile_id] = new_rules;
|
|
|
|
|
tilesheet->autotile_rule_counts[base_tile_id]++;
|
|
|
|
|
|
|
|
|
|
return PXL8_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u16 pxl8_tilesheet_apply_autotile(const pxl8_tilesheet* tilesheet, u16 base_tile_id, u8 neighbors) {
|
|
|
|
|
if (!tilesheet || !tilesheet->autotile_rules || base_tile_id == 0 ||
|
|
|
|
|
base_tile_id > tilesheet->total_tiles) {
|
|
|
|
|
return base_tile_id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pxl8_autotile_rule* rules = tilesheet->autotile_rules[base_tile_id];
|
|
|
|
|
u32 rule_count = tilesheet->autotile_rule_counts[base_tile_id];
|
|
|
|
|
|
|
|
|
|
for (u32 i = 0; i < rule_count; i++) {
|
|
|
|
|
if (rules[i].neighbor_mask == neighbors) {
|
|
|
|
|
return rules[i].tile_id;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return base_tile_id;
|
|
|
|
|
}
|