#include <SDL3_image/SDL_image.h>
#ifdef __EMSCRIPTEN__
#include <emscripten/emscripten.h>
#endif
#ifndef SDL_PROP_SURFACE_FLIP_NUMBER
#define SDL_PROP_SURFACE_FLIP_NUMBER "SDL.surface.flip"
#endif
#if defined(SDL_BUILD_MAJOR_VERSION)
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MAJOR_VERSION,
SDL_IMAGE_MAJOR_VERSION == SDL_BUILD_MAJOR_VERSION);
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MINOR_VERSION,
SDL_IMAGE_MINOR_VERSION == SDL_BUILD_MINOR_VERSION);
SDL_COMPILE_TIME_ASSERT(SDL_BUILD_MICRO_VERSION,
SDL_IMAGE_MICRO_VERSION == SDL_BUILD_MICRO_VERSION);
#endif
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MAJOR_VERSION_min, SDL_IMAGE_MAJOR_VERSION >= 0);
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MAJOR_VERSION_max, SDL_IMAGE_MAJOR_VERSION <= 10);
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MINOR_VERSION_min, SDL_IMAGE_MINOR_VERSION >= 0);
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MINOR_VERSION_max, SDL_IMAGE_MINOR_VERSION <= 999);
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MICRO_VERSION_min, SDL_IMAGE_MICRO_VERSION >= 0);
SDL_COMPILE_TIME_ASSERT(SDL_IMAGE_MICRO_VERSION_max, SDL_IMAGE_MICRO_VERSION <= 999);
static struct {
const char *type;
bool (SDLCALL *is)(SDL_IOStream *src);
SDL_Surface *(SDLCALL *load)(SDL_IOStream *src);
} supported[] = {
{ "TGA", NULL, IMG_LoadTGA_IO },
{ "AVIF",IMG_isAVIF,IMG_LoadAVIF_IO },
{ "CUR", IMG_isCUR, IMG_LoadCUR_IO },
{ "ICO", IMG_isICO, IMG_LoadICO_IO },
{ "BMP", IMG_isBMP, IMG_LoadBMP_IO },
{ "GIF", IMG_isGIF, IMG_LoadGIF_IO },
{ "JPG", IMG_isJPG, IMG_LoadJPG_IO },
{ "JXL", IMG_isJXL, IMG_LoadJXL_IO },
{ "LBM", IMG_isLBM, IMG_LoadLBM_IO },
{ "PCX", IMG_isPCX, IMG_LoadPCX_IO },
{ "PNG", IMG_isPNG, IMG_LoadPNG_IO },
{ "PNM", IMG_isPNM, IMG_LoadPNM_IO },
{ "SVG", IMG_isSVG, IMG_LoadSVG_IO },
{ "TIF", IMG_isTIF, IMG_LoadTIF_IO },
{ "XCF", IMG_isXCF, IMG_LoadXCF_IO },
{ "XPM", IMG_isXPM, IMG_LoadXPM_IO },
{ "XV", IMG_isXV, IMG_LoadXV_IO },
{ "WEBP", IMG_isWEBP, IMG_LoadWEBP_IO },
{ "QOI", IMG_isQOI, IMG_LoadQOI_IO },
};
static struct {
const char *type;
bool (SDLCALL *is)(SDL_IOStream *src);
IMG_Animation *(SDLCALL *load)(SDL_IOStream *src);
} supported_anims[] = {
{ "GIF", IMG_isGIF, IMG_LoadGIFAnimation_IO },
{ "WEBP", IMG_isWEBP, IMG_LoadWEBPAnimation_IO },
{ "APNG", IMG_isPNG, IMG_LoadAPNGAnimation_IO },
{ "AVIFS", IMG_isAVIF, IMG_LoadAVIFAnimation_IO },
{ "ANI", IMG_isANI, IMG_LoadANIAnimation_IO },
};
int IMG_Version(void)
{
return SDL_IMAGE_VERSION;
}
#if !defined(__APPLE__) || defined(SDL_IMAGE_USE_COMMON_BACKEND)
SDL_Surface *IMG_Load(const char *file)
{
#ifdef __EMSCRIPTEN__
int w, h;
char *data;
SDL_Surface *surf;
data = emscripten_get_preloaded_image_data(file, &w, &h);
if (data != NULL) {
surf = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_RGBA32);
if (surf != NULL) {
SDL_memcpy(surf->pixels, data, w * h * 4);
}
free(data);
return surf;
}
#endif
SDL_IOStream *src = SDL_IOFromFile(file, "rb");
if (!src) {
return NULL;
}
const char *ext = SDL_strrchr(file, '.');
if (ext) {
ext++;
}
return IMG_LoadTyped_IO(src, true, ext);
}
#endif
SDL_Surface *IMG_Load_IO(SDL_IOStream *src, bool closeio)
{
return IMG_LoadTyped_IO(src, closeio, NULL);
}
SDL_Surface *IMG_LoadTyped_IO(SDL_IOStream *src, bool closeio, const char *type)
{
size_t i;
SDL_Surface *image;
if (!src) {
SDL_InvalidParamError("src");
return NULL;
}
if (SDL_SeekIO(src, 0, SDL_IO_SEEK_CUR) < 0) {
SDL_SetError("Can't seek in this data source");
if (closeio) {
SDL_CloseIO(src);
}
return NULL;
}
#ifdef __EMSCRIPTEN__
FILE *fp = (FILE *)SDL_GetPointerProperty(SDL_GetIOProperties(src), SDL_PROP_IOSTREAM_STDIO_FILE_POINTER, NULL);
if (fp) {
int w, h;
char *data;
data = emscripten_get_preloaded_image_data_from_FILE(fp, &w, &h);
if (data) {
image = SDL_CreateSurface(w, h, SDL_PIXELFORMAT_RGBA32);
if (image != NULL) {
SDL_memcpy(image->pixels, data, w * h * 4);
}
free(data);
if (closeio) {
SDL_CloseIO(src);
}
return image;
}
}
#endif
for (i = 0; i < SDL_arraysize(supported); ++i) {
if (supported[i].is) {
if (!supported[i].is(src)) {
continue;
}
} else {
if (!type || SDL_strcasecmp(type, supported[i].type) != 0) {
continue;
}
}
#ifdef DEBUG_IMGLIB
SDL_Log("IMGLIB: Loading image as %s\n", supported[i].type);
#endif
image = supported[i].load(src);
if (closeio) {
SDL_CloseIO(src);
}
return image;
}
if (closeio) {
SDL_CloseIO(src);
}
SDL_SetError("Unsupported image format");
return NULL;
}
SDL_Texture *IMG_LoadTexture(SDL_Renderer *renderer, const char *file)
{
SDL_Texture *texture = NULL;
SDL_Surface *surface = IMG_Load(file);
if (surface) {
texture = SDL_CreateTextureFromSurface(renderer, surface);
SDL_DestroySurface(surface);
}
return texture;
}
SDL_Texture *IMG_LoadTexture_IO(SDL_Renderer *renderer, SDL_IOStream *src, bool closeio)
{
SDL_Texture *texture = NULL;
SDL_Surface *surface = IMG_Load_IO(src, closeio);
if (surface) {
texture = SDL_CreateTextureFromSurface(renderer, surface);
SDL_DestroySurface(surface);
}
return texture;
}
SDL_Texture *IMG_LoadTextureTyped_IO(SDL_Renderer *renderer, SDL_IOStream *src, bool closeio, const char *type)
{
SDL_Texture *texture = NULL;
SDL_Surface *surface = IMG_LoadTyped_IO(src, closeio, type);
if (surface) {
texture = SDL_CreateTextureFromSurface(renderer, surface);
SDL_DestroySurface(surface);
}
return texture;
}
IMG_Animation *IMG_LoadAnimation(const char *file)
{
SDL_IOStream *src = SDL_IOFromFile(file, "rb");
const char *ext = SDL_strrchr(file, '.');
if (ext) {
ext++;
}
if (!src) {
return NULL;
}
return IMG_LoadAnimationTyped_IO(src, true, ext);
}
IMG_Animation *IMG_LoadAnimation_IO(SDL_IOStream *src, bool closeio)
{
return IMG_LoadAnimationTyped_IO(src, closeio, NULL);
}
IMG_Animation *IMG_LoadAnimationTyped_IO(SDL_IOStream *src, bool closeio, const char *type)
{
size_t i;
IMG_Animation *anim;
SDL_Surface *image;
if (!src) {
SDL_InvalidParamError("src");
return NULL;
}
if (SDL_SeekIO(src, 0, SDL_IO_SEEK_CUR) < 0) {
SDL_SetError("Can't seek in this data source");
if (closeio) {
SDL_CloseIO(src);
}
return NULL;
}
for (i = 0; i < SDL_arraysize(supported_anims); ++i) {
if (supported_anims[i].is) {
if (!supported_anims[i].is(src)) {
continue;
}
} else {
if (!type || SDL_strcasecmp(type, supported_anims[i].type) != 0) {
continue;
}
}
#ifdef DEBUG_IMGLIB
SDL_Log("IMGLIB: Loading image as %s\n", supported_anims[i].type);
#endif
anim = supported_anims[i].load(src);
if (closeio) {
SDL_CloseIO(src);
}
return anim;
}
image = IMG_LoadTyped_IO(src, closeio, type);
if (image) {
anim = (IMG_Animation *)SDL_malloc(sizeof(*anim));
if (anim) {
anim->w = image->w;
anim->h = image->h;
anim->count = 1;
anim->frames = (SDL_Surface **)SDL_calloc(anim->count, sizeof(*anim->frames));
anim->delays = (int *)SDL_calloc(anim->count, sizeof(*anim->delays));
if (anim->frames && anim->delays) {
anim->frames[0] = image;
return anim;
}
IMG_FreeAnimation(anim);
}
SDL_DestroySurface(image);
}
return NULL;
}
bool IMG_VerifyCanSaveSurface(SDL_Surface *surface)
{
if (!surface) {
return SDL_InvalidParamError("surface");
}
if (SDL_ISPIXELFORMAT_INDEXED(surface->format) && !SDL_GetSurfacePalette(surface)) {
return SDL_SetError("Indexed surfaces must have a palette");
}
return true;
}
bool IMG_Save(SDL_Surface *surface, const char *file)
{
if (!IMG_VerifyCanSaveSurface(surface)) {
return false;
}
if (!file || !*file) {
return SDL_InvalidParamError("file");
}
const char *type = SDL_strrchr(file, '.');
if (type) {
++type;
} else {
return SDL_SetError("Couldn't determine file type");
}
SDL_IOStream *dst = SDL_IOFromFile(file, "wb");
if (!dst) {
return false;
}
return IMG_SaveTyped_IO(surface, dst, true, type);
}
bool IMG_SaveTyped_IO(SDL_Surface *surface, SDL_IOStream *dst, bool closeio, const char *type)
{
bool result = false;
if (!surface) {
SDL_InvalidParamError("surface");
goto done;
}
if (!dst) {
SDL_InvalidParamError("dst");
goto done;
}
if (!type || !*type) {
SDL_InvalidParamError("type");
goto done;
}
if (SDL_strcasecmp(type, "avif") == 0) {
result = IMG_SaveAVIF_IO(surface, dst, false, 90);
} else if (SDL_strcasecmp(type, "bmp") == 0) {
result = IMG_SaveBMP_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "cur") == 0) {
result = IMG_SaveCUR_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "gif") == 0) {
result = IMG_SaveGIF_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "ico") == 0) {
result = IMG_SaveICO_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "jpg") == 0 ||
SDL_strcasecmp(type, "jpeg") == 0) {
result = IMG_SaveJPG_IO(surface, dst, false, 90);
} else if (SDL_strcasecmp(type, "png") == 0) {
result = IMG_SavePNG_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "tga") == 0) {
result = IMG_SaveTGA_IO(surface, dst, false);
} else if (SDL_strcasecmp(type, "webp") == 0) {
result = IMG_SaveWEBP_IO(surface, dst, false, 90.0f);
} else {
result = SDL_SetError("Unsupported image format");
}
done:
if (dst && closeio) {
result &= SDL_CloseIO(dst);
}
return result;
}
bool IMG_SaveAnimation(IMG_Animation *anim, const char *file)
{
if (!anim) {
return SDL_InvalidParamError("anim");
}
if (!file || !*file) {
return SDL_InvalidParamError("file");
}
const char *type = SDL_strrchr(file, '.');
if (type) {
++type;
} else {
return SDL_SetError("Couldn't determine file type");
}
SDL_IOStream *dst = SDL_IOFromFile(file, "wb");
if (!dst) {
return false;
}
return IMG_SaveAnimationTyped_IO(anim, dst, true, type);
}
bool IMG_SaveAnimationTyped_IO(IMG_Animation *anim, SDL_IOStream *dst, bool closeio, const char *type)
{
bool result = false;
if (!anim) {
SDL_InvalidParamError("anim");
goto done;
}
if (!dst) {
SDL_InvalidParamError("dst");
goto done;
}
if (!type || !*type) {
SDL_InvalidParamError("type");
goto done;
}
if (SDL_strcasecmp(type, "ani") == 0) {
result = IMG_SaveANIAnimation_IO(anim, dst, false);
} else if (SDL_strcasecmp(type, "apng") == 0 || SDL_strcasecmp(type, "png") == 0) {
result = IMG_SaveAPNGAnimation_IO(anim, dst, false);
} else if (SDL_strcasecmp(type, "avif") == 0) {
result = IMG_SaveAVIFAnimation_IO(anim, dst, false, 90);
} else if (SDL_strcasecmp(type, "gif") == 0) {
result = IMG_SaveGIFAnimation_IO(anim, dst, false);
} else if (SDL_strcasecmp(type, "webp") == 0) {
result = IMG_SaveWEBPAnimation_IO(anim, dst, false, 90);
} else {
result = SDL_SetError("Unsupported image format");
}
done:
if (dst && closeio) {
result &= SDL_CloseIO(dst);
}
return result;
}
SDL_Surface *IMG_GetClipboardImage(void)
{
SDL_Surface *surface = NULL;
char **mime_types = SDL_GetClipboardMimeTypes(NULL);
if (mime_types) {
for (int i = 0; !surface && mime_types[i]; ++i) {
if (SDL_strncmp(mime_types[i], "image/", 6) == 0) {
size_t size = 0;
void *data = SDL_GetClipboardData(mime_types[i], &size);
if (data) {
SDL_IOStream *src = SDL_IOFromConstMem(data, size);
if (src) {
surface = IMG_Load_IO(src, true);
}
SDL_free(data);
}
}
}
}
if (!surface) {
SDL_SetError("No clipboard image available");
}
return surface;
}
SDL_Cursor *IMG_CreateAnimatedCursor(IMG_Animation *anim, int hot_x, int hot_y)
{
int i;
SDL_CursorFrameInfo *frames;
SDL_Cursor *cursor;
if (!anim) {
SDL_InvalidParamError("anim");
return NULL;
}
frames = (SDL_CursorFrameInfo *)SDL_calloc(anim->count, sizeof(*frames));
if (!frames) {
return NULL;
}
for (i = 0; i < anim->count; ++i) {
frames[i].surface = anim->frames[i];
frames[i].duration = (Uint32)anim->delays[i];
}
cursor = SDL_CreateAnimatedCursor(frames, anim->count, hot_x, hot_y);
SDL_free(frames);
return cursor;
}
void IMG_FreeAnimation(IMG_Animation *anim)
{
if (anim) {
if (anim->frames) {
int i;
for (i = 0; i < anim->count; ++i) {
if (anim->frames[i]) {
SDL_DestroySurface(anim->frames[i]);
}
}
SDL_free(anim->frames);
}
if (anim->delays) {
SDL_free(anim->delays);
}
SDL_free(anim);
}
}
Uint64 IMG_TimebaseDuration(Uint64 pts, Uint64 duration, Uint64 src_numerator, Uint64 src_denominator, Uint64 dst_numerator, Uint64 dst_denominator)
{
Uint64 a = ( ( ( ( pts + duration ) * 2 ) + 1 ) * src_numerator * dst_denominator ) / ( 2 * src_denominator * dst_numerator );
Uint64 b = ( ( ( pts * 2 ) + 1 ) * src_numerator * dst_denominator ) / ( 2 * src_denominator * dst_numerator );
return (a - b);
}
SDL_Surface *IMG_ApplyOrientation(SDL_Surface *surface, int orientation)
{
float rotation = 0.0f;
SDL_FlipMode flip = SDL_FLIP_NONE;
switch (orientation) {
case 1:
break;
case 2:
flip = SDL_FLIP_HORIZONTAL;
break;
case 3:
rotation = 180.0f;
break;
case 4:
flip = SDL_FLIP_VERTICAL;
break;
case 5:
flip = SDL_FLIP_HORIZONTAL;
rotation = 270.0f;
break;
case 6:
rotation = 90.0f;
break;
case 7:
flip = SDL_FLIP_HORIZONTAL;
rotation = 90.0f;
break;
case 8:
rotation = 270.0f;
break;
default:
break;
}
#ifdef ORIENTATION_USES_PROPERTIES
if (flip != SDL_FLIP_NONE) {
SDL_PropertiesID props = SDL_GetSurfaceProperties(surface);
SDL_SetNumberProperty(props, SDL_PROP_SURFACE_FLIP_NUMBER, flip);
}
if (rotation != 0.0f) {
SDL_PropertiesID props = SDL_GetSurfaceProperties(surface);
SDL_SetFloatProperty(props, SDL_PROP_SURFACE_ROTATION_FLOAT, rotation);
}
#else
if (flip != SDL_FLIP_NONE) {
if (!SDL_FlipSurface(surface, flip)) {
SDL_DestroySurface(surface);
return NULL;
}
}
if (rotation != 0.0f) {
SDL_Surface *tmp = SDL_RotateSurface(surface, rotation);
SDL_DestroySurface(surface);
if (!tmp) {
return NULL;
}
surface = tmp;
}
#endif
return surface;
}