// FYI: There used to be image resize code in here directly, but I moved it to `imageresize.d`. /++ This file imports all available image decoders in the arsd library, and provides convenient functions to load image regardless of it's format. Main functions: [loadImageFromFile] and [loadImageFromMemory]. $(WARNING This module is exempt from my usual build-compatibility guarantees. I may add new built-time dependency modules to it at any time without notice. You should either copy the `image.d` module and the pieces you use to your own project, or always use it along with the rest of the repo and `dmd -i`, or the dub `arsd-official:image_files` subpackage, which both will include new files automatically and avoid breaking your build. ) History: The image resize code used to live directly in here, but has now moved to a new module, [arsd.imageresize]. It is public imported here for compatibility, but the build has changed as of December 25, 2020. +/ module arsd.image; public import arsd.color; public import arsd.png; public import arsd.jpeg; public import arsd.bmp; public import arsd.targa; public import arsd.pcx; public import arsd.dds; public import arsd.svg; public import arsd.imageresize; import core.memory; static if (__traits(compiles, { import iv.vfs; })) enum ArsdImageHasIVVFS = true; else enum ArsdImageHasIVVFS = false; MemoryImage readSvg(string filename) { import std.file; return readSvg(cast(const(ubyte)[]) readText(filename)); } MemoryImage readSvg(const(ubyte)[] rawData) { // Load NSVG* image = nsvgParse(cast(const(char)[]) rawData); if(image is null) return null; int w = cast(int) image.width + 1; int h = cast(int) image.height + 1; NSVGrasterizer rast = nsvgCreateRasterizer(); auto img = new TrueColorImage(w, h); rasterize(rast, image, 0, 0, 1, img.imageData.bytes.ptr, w, h, w*4); image.kill(); return img; } private bool strEquCI (const(char)[] s0, const(char)[] s1) pure nothrow @trusted @nogc { if (s0.length != s1.length) return false; foreach (immutable idx, char ch; s0) { if (ch >= 'A' && ch <= 'Z') ch += 32; // poor man's tolower() char c1 = s1.ptr[idx]; if (c1 >= 'A' && c1 <= 'Z') c1 += 32; // poor man's tolower() if (ch != c1) return false; } return true; } /// Image formats `arsd.image` can load (except `Unknown`, of course). enum ImageFileFormat { Unknown, /// Png, /// Bmp, /// Jpeg, /// Tga, /// Gif, /// we can't load it yet, but we can at least detect it Pcx, /// can load 8BPP and 24BPP pcx images Dds, /// can load ARGB8888, DXT1, DXT3, DXT5 dds images (without mipmaps) Svg, /// will rasterize simple svg images } /// Try to guess image format from file extension. public ImageFileFormat guessImageFormatFromExtension (const(char)[] filename) { if (filename.length < 2) return ImageFileFormat.Unknown; size_t extpos = filename.length; version(Windows) { while (extpos > 0 && filename.ptr[extpos-1] != '.' && filename.ptr[extpos-1] != '/' && filename.ptr[extpos-1] != '\\' && filename.ptr[extpos-1] != ':') --extpos; } else { while (extpos > 0 && filename.ptr[extpos-1] != '.' && filename.ptr[extpos-1] != '/') --extpos; } if (extpos == 0 || filename.ptr[extpos-1] != '.') return ImageFileFormat.Unknown; auto ext = filename[extpos..$]; if (strEquCI(ext, "png")) return ImageFileFormat.Png; if (strEquCI(ext, "bmp")) return ImageFileFormat.Bmp; if (strEquCI(ext, "jpg") || strEquCI(ext, "jpeg")) return ImageFileFormat.Jpeg; if (strEquCI(ext, "gif")) return ImageFileFormat.Gif; if (strEquCI(ext, "tga")) return ImageFileFormat.Tga; if (strEquCI(ext, "pcx")) return ImageFileFormat.Pcx; if (strEquCI(ext, "dds")) return ImageFileFormat.Dds; if (strEquCI(ext, "svg")) return ImageFileFormat.Svg; return ImageFileFormat.Unknown; } /// Try to guess image format by first data bytes. public ImageFileFormat guessImageFormatFromMemory (const(void)[] membuf) { enum TargaSign = "TRUEVISION-XFILE.\x00"; auto buf = cast(const(ubyte)[])membuf; if (buf.length == 0) return ImageFileFormat.Unknown; // detect file format // png if (buf.length > 7 && buf.ptr[0] == 0x89 && buf.ptr[1] == 0x50 && buf.ptr[2] == 0x4E && buf.ptr[3] == 0x47 && buf.ptr[4] == 0x0D && buf.ptr[5] == 0x0A && buf.ptr[6] == 0x1A) { return ImageFileFormat.Png; } // bmp if (buf.length > 6 && buf.ptr[0] == 'B' && buf.ptr[1] == 'M') { uint datasize = buf.ptr[2]|(buf.ptr[3]<<8)|(buf.ptr[4]<<16)|(buf.ptr[5]<<24); if (datasize > 6 && datasize <= buf.length) return ImageFileFormat.Bmp; } // gif if (buf.length > 5 && buf.ptr[0] == 'G' && buf.ptr[1] == 'I' && buf.ptr[2] == 'F' && buf.ptr[3] == '8' && (buf.ptr[4] == '7' || buf.ptr[4] == '9')) { return ImageFileFormat.Gif; } // dds if (ddsDetect(membuf)) return ImageFileFormat.Dds; // jpg try { int width, height, components; if (detect_jpeg_image_from_memory(buf, width, height, components)) return ImageFileFormat.Jpeg; } catch (Exception e) {} // sorry // tga (sorry, targas without footer, i don't love you) if (buf.length > TargaSign.length+4*2 && cast(const(char)[])(buf[$-TargaSign.length..$]) == TargaSign) { // more guesswork switch (buf.ptr[2]) { case 1: case 2: case 3: case 9: case 10: case 11: return ImageFileFormat.Tga; default: } } // ok, try to guess targa by validating some header fields bool guessTarga () nothrow @trusted @nogc { if (buf.length < 45) return false; // minimal 1x1 tga immutable ubyte idlength = buf.ptr[0]; immutable ubyte bColorMapType = buf.ptr[1]; immutable ubyte type = buf.ptr[2]; immutable ushort wColorMapFirstEntryIndex = cast(ushort)(buf.ptr[3]|(buf.ptr[4]<<8)); immutable ushort wColorMapLength = cast(ushort)(buf.ptr[5]|(buf.ptr[6]<<8)); immutable ubyte bColorMapEntrySize = buf.ptr[7]; immutable ushort wOriginX = cast(ushort)(buf.ptr[8]|(buf.ptr[9]<<8)); immutable ushort wOriginY = cast(ushort)(buf.ptr[10]|(buf.ptr[11]<<8)); immutable ushort wImageWidth = cast(ushort)(buf.ptr[12]|(buf.ptr[13]<<8)); immutable ushort wImageHeight = cast(ushort)(buf.ptr[14]|(buf.ptr[15]<<8)); immutable ubyte bPixelDepth = buf.ptr[16]; immutable ubyte bImageDescriptor = buf.ptr[17]; if (wImageWidth < 1 || wImageHeight < 1 || wImageWidth > 32000 || wImageHeight > 32000) return false; // arbitrary limit immutable uint pixelsize = (bPixelDepth>>3); switch (type) { case 2: // truecolor, raw case 10: // truecolor, rle switch (pixelsize) { case 2: case 3: case 4: break; default: return false; } break; case 1: // paletted, raw case 9: // paletted, rle if (pixelsize != 1) return false; break; case 3: // b/w, raw case 11: // b/w, rle if (pixelsize != 1 && pixelsize != 2) return false; break; default: // invalid type return false; } // check for valid colormap switch (bColorMapType) { case 0: if (wColorMapFirstEntryIndex != 0 || wColorMapLength != 0) return 0; break; case 1: if (bColorMapEntrySize != 15 && bColorMapEntrySize != 16 && bColorMapEntrySize != 24 && bColorMapEntrySize != 32) return false; if (wColorMapLength == 0) return false; break; default: // invalid colormap type return false; } if (((bImageDescriptor>>6)&3) != 0) return false; // this *looks* like a tga return true; } if (guessTarga()) return ImageFileFormat.Tga; bool guessPcx() nothrow @trusted @nogc { if (buf.length < 129) return false; // we should have at least header ubyte manufacturer = buf.ptr[0]; ubyte ver = buf.ptr[1]; ubyte encoding = buf.ptr[2]; ubyte bitsperpixel = buf.ptr[3]; ushort xmin = cast(ushort)(buf.ptr[4]+256*buf.ptr[5]); ushort ymin = cast(ushort)(buf.ptr[6]+256*buf.ptr[7]); ushort xmax = cast(ushort)(buf.ptr[8]+256*buf.ptr[9]); ushort ymax = cast(ushort)(buf.ptr[10]+256*buf.ptr[11]); ubyte reserved = buf.ptr[64]; ubyte colorplanes = buf.ptr[65]; ushort bytesperline = cast(ushort)(buf.ptr[66]+256*buf.ptr[67]); //ushort palettetype = cast(ushort)(buf.ptr[68]+256*buf.ptr[69]); // check some header fields if (manufacturer != 0x0a) return false; if (/*ver != 0 && ver != 2 && ver != 3 &&*/ ver != 5) return false; if (encoding != 0 && encoding != 1) return false; int wdt = xmax-xmin+1; int hgt = ymax-ymin+1; // arbitrary size limits if (wdt < 1 || wdt > 32000) return false; if (hgt < 1 || hgt > 32000) return false; if (bytesperline < wdt) return false; // if it's not a 256-color PCX file, and not 24-bit PCX file, gtfo bool bpp24 = false; if (colorplanes == 1) { if (bitsperpixel != 8 && bitsperpixel != 24 && bitsperpixel != 32) return false; bpp24 = (bitsperpixel == 24); } else if (colorplanes == 3 || colorplanes == 4) { if (bitsperpixel != 8) return false; bpp24 = true; } // additional checks if (reserved != 0) return false; // 8bpp files MUST have palette if (!bpp24 && buf.length < 129+769) return false; // it can be pcx return true; } if (guessPcx()) return ImageFileFormat.Pcx; // kinda lame svg detection but don't want to parse too much of it here if (buf.length > 6 && buf.ptr[0] == '<') { return ImageFileFormat.Svg; } // dunno return ImageFileFormat.Unknown; } /// Try to guess image format from file name and load that image. public MemoryImage loadImageFromFile(T:const(char)[]) (T filename) { static if (is(T == typeof(null))) { throw new Exception("cannot load image from unnamed file"); } else { final switch (guessImageFormatFromExtension(filename)) { case ImageFileFormat.Unknown: //throw new Exception("cannot determine file format from extension"); static if (ArsdImageHasIVVFS) { auto fl = VFile(filename); } else { import std.stdio; static if (is(T == string)) { auto fl = File(filename); } else { auto fl = File(filename.idup); } } auto fsz = fl.size-fl.tell; if (fsz < 4) throw new Exception("cannot determine file format"); if (fsz > int.max/8) throw new Exception("image data too big"); auto data = new ubyte[](cast(uint)fsz); scope(exit) { import core.memory : GC; GC.free(data.ptr); } // this should be safe, as image will copy data to it's internal storage fl.rawReadExact(data); return loadImageFromMemory(data); case ImageFileFormat.Png: static if (is(T == string)) return readPng(filename); else return readPng(filename.idup); case ImageFileFormat.Bmp: static if (is(T == string)) return readBmp(filename); else return readBmp(filename.idup); case ImageFileFormat.Jpeg: return readJpeg(filename); case ImageFileFormat.Gif: throw new Exception("arsd has no GIF loader yet"); case ImageFileFormat.Tga: return loadTga(filename); case ImageFileFormat.Pcx: return loadPcx(filename); case ImageFileFormat.Svg: static if (is(T == string)) return readSvg(filename); else return readSvg(filename.idup); case ImageFileFormat.Dds: static if (ArsdImageHasIVVFS) { auto fl = VFile(filename); } else { import std.stdio; static if (is(T == string)) { auto fl = File(filename); } else { auto fl = File(filename.idup); } } return ddsLoadFromFile(fl); } } } /// Try to guess image format from data and load that image. public MemoryImage loadImageFromMemory (const(void)[] membuf) { final switch (guessImageFormatFromMemory(membuf)) { case ImageFileFormat.Unknown: throw new Exception("cannot determine file format"); case ImageFileFormat.Png: return imageFromPng(readPng(cast(const(ubyte)[])membuf)); case ImageFileFormat.Bmp: return readBmp(cast(const(ubyte)[])membuf); case ImageFileFormat.Jpeg: return readJpegFromMemory(cast(const(ubyte)[])membuf); case ImageFileFormat.Gif: throw new Exception("arsd has no GIF loader yet"); case ImageFileFormat.Tga: return loadTgaMem(membuf); case ImageFileFormat.Pcx: return loadPcxMem(membuf); case ImageFileFormat.Svg: return readSvg(cast(const(ubyte)[]) membuf); case ImageFileFormat.Dds: return ddsLoadFromMemory(membuf); } } static if (ArsdImageHasIVVFS) { import iv.vfs; public MemoryImage loadImageFromFile (VFile fl) { auto fsz = fl.size-fl.tell; if (fsz < 4) throw new Exception("cannot determine file format"); if (fsz > int.max/8) throw new Exception("image data too big"); auto data = new ubyte[](cast(uint)fsz); scope(exit) { import core.memory : GC; GC.free(data.ptr); } // this should be safe, as image will copy data to it's internal storage fl.rawReadExact(data); return loadImageFromMemory(data); } } // FYI: There used to be image resize code in here directly, but I moved it to `imageresize.d`.