Implement stbi_write here.
This commit is contained in:
parent
e0e57b53f4
commit
e7d7f5f826
@ -1,9 +1,9 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net6.0</TargetFramework>
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
<Nullable>disable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<LangVersion>7.3</LangVersion>
|
<LangVersion>latest</LangVersion>
|
||||||
<AllowUnsafeBlocks>True</AllowUnsafeBlocks>
|
<AllowUnsafeBlocks>True</AllowUnsafeBlocks>
|
||||||
<RuntimeIdentifiers>linux-arm;linux-arm64;linux-x64;win-x86;win-x64;osx-arm64;osx-x64</RuntimeIdentifiers>
|
<RuntimeIdentifiers>linux-arm;linux-arm64;linux-x64;win-x86;win-x64;osx-arm64;osx-x64</RuntimeIdentifiers>
|
||||||
<RootNamespace>ReFuel.Stb</RootNamespace>
|
<RootNamespace>ReFuel.Stb</RootNamespace>
|
||||||
|
364
StbImage.cs
364
StbImage.cs
@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
@ -50,6 +51,71 @@ namespace ReFuel.Stb
|
|||||||
Dispose(false);
|
Dispose(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get a safe span to the image pointer.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">The pixel type.</typeparam>
|
||||||
|
/// <returns>A span to the image data.</returns>
|
||||||
|
/// <exception cref="Exception">The image uses an unexpected image format.</exception>
|
||||||
|
public ReadOnlySpan<T> AsSpan<T>() where T : unmanaged
|
||||||
|
{
|
||||||
|
int sz = Format switch
|
||||||
|
{
|
||||||
|
StbiImageFormat.Grey => 1,
|
||||||
|
StbiImageFormat.GreyAlpha => 2,
|
||||||
|
StbiImageFormat.Rgb => 3,
|
||||||
|
StbiImageFormat.Rgba => 4,
|
||||||
|
_ => throw new Exception("unkown image format")
|
||||||
|
} * (IsFloat ? sizeof(float) : sizeof(byte));
|
||||||
|
|
||||||
|
return new ReadOnlySpan<T>((T*)ImagePointer, Width * Height * sz / sizeof(T));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="dest">The destination stream.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// Incurs a conversion cost if the image format is not a byte format. Affected by non-thread safe global options.
|
||||||
|
/// </remarks>
|
||||||
|
public void WritePng(Stream dest) => WritePng(AsSpan<byte>(), Width, Height, Format, dest, isFloat: IsFloat);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write image to a BMP file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="dest">The destination stream.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// Incurs a conversion cost if the image format is not a byte format. Affected by non-thread safe global options.
|
||||||
|
/// </remarks>
|
||||||
|
public void WriteBmp(Stream dest) => WriteBmp(AsSpan<byte>(), Width, Height, Format, dest, isFloat: IsFloat);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write image to a TGA file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="dest">The destination stream.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// Incurs a conversion cost if the image format is not a byte format. Ignores alpha channel. Affected by non-thread safe global options.
|
||||||
|
/// </remarks>
|
||||||
|
public void WriteTga(Stream dest) => WriteTga(AsSpan<byte>(), Width, Height, Format, dest, isFloat: IsFloat);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="dest">The destination stream.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// Incurs a conversion cost if the image format is not a float format. Affected by non-thread safe global options.
|
||||||
|
/// </remarks>
|
||||||
|
public void WriteHdr(Stream dest) => WriteHdr(AsSpan<byte>(), Width, Height, Format, dest, isFloat: IsFloat);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="dest">The destination stream.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// Incurs a conversion cost if the image format is not a byte format. Ignores alpha channel. Affected by non-thread safe global options.
|
||||||
|
/// </remarks>
|
||||||
|
public void WriteJpg(Stream dest, int quality = 90) => WriteJpg(AsSpan<byte>(), Width, Height, Format, dest, quality: quality, isFloat: IsFloat);
|
||||||
|
|
||||||
public void Dispose() => Dispose(true);
|
public void Dispose() => Dispose(true);
|
||||||
|
|
||||||
private void Dispose(bool disposing)
|
private void Dispose(bool disposing)
|
||||||
@ -68,7 +134,12 @@ namespace ReFuel.Stb
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Set to flip the y-axis of loaded images on load.
|
/// Set to flip the y-axis of loaded images on load.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static bool FlipVerticallyOnLoad { set => Stbi.set_flip_vertically_on_load(1); }
|
public static bool FlipVerticallyOnLoad { set => Stbi.set_flip_vertically_on_load(value ? 1 : 0); }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Set to flip the y-axis of saved images.
|
||||||
|
/// </summary>
|
||||||
|
public static bool FlipVerticallyOnSave { set => Stbi.flip_vertically_on_write(value ? 1 : 0); }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Set to unpremultiply images on load.
|
/// Set to unpremultiply images on load.
|
||||||
@ -79,6 +150,50 @@ namespace ReFuel.Stb
|
|||||||
/// </remarks>
|
/// </remarks>
|
||||||
public static bool UnpremultiplyOnLoad { set => Stbi.set_unpremultiply_on_load(1); }
|
public static bool UnpremultiplyOnLoad { set => Stbi.set_unpremultiply_on_load(1); }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Force a filter on PNG filter when saving.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// -1 for auto, 0 through 5 to pick a filter. Higher is more. Not thread safe.
|
||||||
|
/// </remarks>
|
||||||
|
public int WriteForcePngFilter
|
||||||
|
{
|
||||||
|
get => Stbi.write_force_png_filter;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (value < -1 || value > 5)
|
||||||
|
{
|
||||||
|
throw new ArgumentOutOfRangeException(nameof(value), "The PNG filter must be in the range 0 to 5, or -1 for auto.");
|
||||||
|
}
|
||||||
|
|
||||||
|
Stbi.write_force_png_filter = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Change the PNG compression level on save.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Higher is more. Defaults to 8. Not thread safe.
|
||||||
|
/// </remarks>
|
||||||
|
public int WritePngCompressionLevel
|
||||||
|
{
|
||||||
|
get => Stbi.write_png_compression_level;
|
||||||
|
set => Stbi.write_png_compression_level = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Enable run length encoding on TGA images on save.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Not thread safe.
|
||||||
|
/// </remarks>
|
||||||
|
public bool WriteTgaEnableRLE
|
||||||
|
{
|
||||||
|
get => Stbi.write_tga_with_rle != 0;
|
||||||
|
set => Stbi.write_tga_with_rle = value ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Try loading an image, without raising exceptions.
|
/// Try loading an image, without raising exceptions.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -86,7 +201,7 @@ namespace ReFuel.Stb
|
|||||||
/// <param name="stream">Source stream.</param>
|
/// <param name="stream">Source stream.</param>
|
||||||
/// <param name="format">The desired image format.</param>
|
/// <param name="format">The desired image format.</param>
|
||||||
/// <returns>True on success.</returns>
|
/// <returns>True on success.</returns>
|
||||||
public static bool TryLoad(out StbImage image, Stream stream, StbiImageFormat format = StbiImageFormat.Default, bool isFloat = false)
|
public static bool TryLoad([NotNullWhen(true)] out StbImage? image, Stream stream, StbiImageFormat format = StbiImageFormat.Default, bool asFloat = false)
|
||||||
{
|
{
|
||||||
int x, y, iFormat;
|
int x, y, iFormat;
|
||||||
StbiStreamWrapper wrapper = new StbiStreamWrapper(stream, true);
|
StbiStreamWrapper wrapper = new StbiStreamWrapper(stream, true);
|
||||||
@ -94,7 +209,7 @@ namespace ReFuel.Stb
|
|||||||
|
|
||||||
stream.Position = 0;
|
stream.Position = 0;
|
||||||
IntPtr imagePtr;
|
IntPtr imagePtr;
|
||||||
if (isFloat)
|
if (asFloat)
|
||||||
{
|
{
|
||||||
imagePtr = (IntPtr)Stbi.loadf_from_callbacks(&cb, null, &x, &y, &iFormat, (int)format);
|
imagePtr = (IntPtr)Stbi.loadf_from_callbacks(&cb, null, &x, &y, &iFormat, (int)format);
|
||||||
}
|
}
|
||||||
@ -105,7 +220,7 @@ namespace ReFuel.Stb
|
|||||||
|
|
||||||
if (imagePtr != IntPtr.Zero)
|
if (imagePtr != IntPtr.Zero)
|
||||||
{
|
{
|
||||||
image = new StbImage(imagePtr, x, y, (StbiImageFormat)iFormat, isFloat);
|
image = new StbImage(imagePtr, x, y, (StbiImageFormat)iFormat, asFloat);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -122,25 +237,24 @@ namespace ReFuel.Stb
|
|||||||
/// <param name="span">Source memory span.</param>
|
/// <param name="span">Source memory span.</param>
|
||||||
/// <param name="format">The desired image format.</param>
|
/// <param name="format">The desired image format.</param>
|
||||||
/// <returns>True on success.</returns>
|
/// <returns>True on success.</returns>
|
||||||
public static bool TryLoad<T>(out StbImage image, ReadOnlySpan<T> span, StbiImageFormat format = StbiImageFormat.Default, bool isFloat = false)
|
public static bool TryLoad([NotNullWhen(true)] out StbImage? image, ReadOnlySpan<byte> span, StbiImageFormat format = StbiImageFormat.Default, bool asFloat = false)
|
||||||
where T : unmanaged
|
|
||||||
{
|
{
|
||||||
IntPtr imagePtr = IntPtr.Zero;
|
IntPtr imagePtr = IntPtr.Zero;
|
||||||
int x, y, iFormat;
|
int x, y, iFormat;
|
||||||
fixed (byte *ptr = MemoryMarshal.AsBytes(span))
|
fixed (byte *ptr = MemoryMarshal.AsBytes(span))
|
||||||
{
|
{
|
||||||
if (isFloat)
|
if (asFloat)
|
||||||
{
|
{
|
||||||
imagePtr = (IntPtr)Stbi.loadf_from_memory(ptr, span.Length * sizeof(T), &x, &y, &iFormat, (int)format);
|
imagePtr = (IntPtr)Stbi.loadf_from_memory(ptr, span.Length, &x, &y, &iFormat, (int)format);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
imagePtr = (IntPtr)Stbi.load_from_memory(ptr, span.Length * sizeof(T), &x, &y, &iFormat, (int)format);
|
imagePtr = (IntPtr)Stbi.load_from_memory(ptr, span.Length, &x, &y, &iFormat, (int)format);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (imagePtr != IntPtr.Zero)
|
if (imagePtr != IntPtr.Zero)
|
||||||
{
|
{
|
||||||
image = new StbImage(imagePtr, x, y, (StbiImageFormat)iFormat, isFloat);
|
image = new StbImage(imagePtr, x, y, (StbiImageFormat)iFormat, asFloat);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -157,14 +271,14 @@ namespace ReFuel.Stb
|
|||||||
/// <param name="stream">The stream to load from.</param>
|
/// <param name="stream">The stream to load from.</param>
|
||||||
/// <param name="format">The desired image format.</param>
|
/// <param name="format">The desired image format.</param>
|
||||||
/// <returns>The image object.</returns>
|
/// <returns>The image object.</returns>
|
||||||
public static StbImage Load(Stream stream, StbiImageFormat format = StbiImageFormat.Default, bool isFloat = false)
|
public static StbImage Load(Stream stream, StbiImageFormat format = StbiImageFormat.Default, bool asFloat = false)
|
||||||
{
|
{
|
||||||
if (TryLoad(out StbImage image, stream, format, isFloat))
|
if (TryLoad(out StbImage? image, stream, format, asFloat))
|
||||||
{
|
{
|
||||||
return image;
|
return image;
|
||||||
}
|
}
|
||||||
|
|
||||||
string reason = Marshal.PtrToStringUTF8((IntPtr)Stbi.failure_reason());
|
string reason = Marshal.PtrToStringUTF8((IntPtr)Stbi.failure_reason())!;
|
||||||
throw new Exception($"Failed to load image: {reason}");
|
throw new Exception($"Failed to load image: {reason}");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -174,15 +288,14 @@ namespace ReFuel.Stb
|
|||||||
/// <param name="span">The span of memory to load from.</param>
|
/// <param name="span">The span of memory to load from.</param>
|
||||||
/// <param name="format">The desired image format.</param>
|
/// <param name="format">The desired image format.</param>
|
||||||
/// <returns>The image object.</returns>
|
/// <returns>The image object.</returns>
|
||||||
public static StbImage Load<T>(ReadOnlySpan<T> span, StbiImageFormat format = StbiImageFormat.Default, bool isFloat = false)
|
public static StbImage Load(ReadOnlySpan<byte> span, StbiImageFormat format = StbiImageFormat.Default, bool asFloat = false)
|
||||||
where T : unmanaged
|
|
||||||
{
|
{
|
||||||
if (TryLoad(out StbImage image, span, format, isFloat))
|
if (TryLoad(out StbImage? image, span, format, asFloat))
|
||||||
{
|
{
|
||||||
return image;
|
return image;
|
||||||
}
|
}
|
||||||
|
|
||||||
string reason = Marshal.PtrToStringUTF8((IntPtr)Stbi.failure_reason());
|
string reason = Marshal.PtrToStringUTF8((IntPtr)Stbi.failure_reason())!;
|
||||||
throw new Exception($"Failed to load image: {reason}");
|
throw new Exception($"Failed to load image: {reason}");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -230,5 +343,222 @@ namespace ReFuel.Stb
|
|||||||
return result != 0;
|
return result != 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static int Components(StbiImageFormat format) => format switch
|
||||||
|
{
|
||||||
|
StbiImageFormat.Grey => 1,
|
||||||
|
StbiImageFormat.GreyAlpha => 2,
|
||||||
|
StbiImageFormat.Rgb => 3,
|
||||||
|
StbiImageFormat.Rgba => 4,
|
||||||
|
_ => throw new ArgumentException("Expected a fully qualified format.")
|
||||||
|
};
|
||||||
|
|
||||||
|
private static byte[] ConvertFloatToByte<T>(ReadOnlySpan<T> source, int width, int height, int components)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
byte[] conversion = new byte[width * height * components];
|
||||||
|
ReadOnlySpan<float> dataAsFloat = MemoryMarshal.Cast<T, float>(source);
|
||||||
|
|
||||||
|
for (int i = 0; i<conversion.Length; i++)
|
||||||
|
{
|
||||||
|
conversion[i] = (byte) Math.Clamp(MathF.Round(dataAsFloat[i]* 255), 0, 255);
|
||||||
|
}
|
||||||
|
|
||||||
|
return conversion;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static float[] ConvertByteToFloat<T>(ReadOnlySpan<T> source, int width, int height, int components)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
float[] conversion = new float[width * height * components];
|
||||||
|
ReadOnlySpan<byte> dataAsByte = MemoryMarshal.Cast<T, byte>(source);
|
||||||
|
|
||||||
|
for (int i = 0; i < conversion.Length; i++)
|
||||||
|
{
|
||||||
|
conversion[i] = Math.Clamp(dataAsByte[i]/255f, 0f, 1f);
|
||||||
|
}
|
||||||
|
|
||||||
|
return conversion;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write any image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">Any packed byte or float array structure.</typeparam>
|
||||||
|
/// <param name="data">Span of pixel data.</param>
|
||||||
|
/// <param name="width">Width of the pixel data in pixels.</param>
|
||||||
|
/// <param name="height">Height of the pixel data in pixels.</param>
|
||||||
|
/// <param name="format">Color format of the pixel data. Must not be <see cref="StbiImageFormat.StbiImageFormat"/>.</param>
|
||||||
|
/// <param name="destination">The destination stream.</param>
|
||||||
|
/// <param name="isFloat">True if the pixel format in data is a floating point format.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// This will incur a conversion cost if the pixel format is not a byte format. Affected by global non-thread safe options.
|
||||||
|
/// </remarks>
|
||||||
|
public static void WritePng<T>(ReadOnlySpan<T> data, int width, int height, StbiImageFormat format, Stream destination, bool isFloat = false)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
int components = Components(format);
|
||||||
|
ReadOnlySpan<byte> source;
|
||||||
|
byte[]? conversion;
|
||||||
|
|
||||||
|
if (isFloat)
|
||||||
|
{
|
||||||
|
conversion = ConvertFloatToByte(data, width, height, components);
|
||||||
|
source = conversion;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
source = MemoryMarshal.AsBytes(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
StbiWriteStreamWrapper wrapper = new StbiWriteStreamWrapper(destination);
|
||||||
|
|
||||||
|
fixed (byte *ptr = source)
|
||||||
|
Stbi.write_png_to_func(wrapper, null, width, height, components, ptr, width * components);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write any image to a BMP file.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">Any packed byte or float array structure.</typeparam>
|
||||||
|
/// <param name="data">Span of pixel data.</param>
|
||||||
|
/// <param name="width">Width of the pixel data in pixels.</param>
|
||||||
|
/// <param name="height">Height of the pixel data in pixels.</param>
|
||||||
|
/// <param name="format">Color format of the pixel data. Must not be <see cref="StbiImageFormat.StbiImageFormat"/>.</param>
|
||||||
|
/// <param name="destination">The destination stream.</param>
|
||||||
|
/// <param name="isFloat">True if the pixel format in data is a floating point format.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// This will incur a conversion cost if the pixel format is not a byte format. Ignores the alpha channel. Affected by global non-thread safe options.
|
||||||
|
/// </remarks>
|
||||||
|
public static void WriteBmp<T>(ReadOnlySpan<T> data, int width, int height, StbiImageFormat format, Stream destination, bool isFloat = false)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
int components = Components(format);
|
||||||
|
ReadOnlySpan<byte> source;
|
||||||
|
byte[]? conversion;
|
||||||
|
|
||||||
|
if (isFloat)
|
||||||
|
{
|
||||||
|
conversion = ConvertFloatToByte(data, width, height, components);
|
||||||
|
source = conversion;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
source = MemoryMarshal.AsBytes(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
StbiWriteStreamWrapper wrapper = new StbiWriteStreamWrapper(destination);
|
||||||
|
|
||||||
|
fixed (byte* ptr = source)
|
||||||
|
Stbi.write_bmp_to_func(wrapper, null, width, height, components, ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write any image to a TGA file.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">Any packed byte or float array structure.</typeparam>
|
||||||
|
/// <param name="data">Span of pixel data.</param>
|
||||||
|
/// <param name="width">Width of the pixel data in pixels.</param>
|
||||||
|
/// <param name="height">Height of the pixel data in pixels.</param>
|
||||||
|
/// <param name="format">Color format of the pixel data. Must not be <see cref="StbiImageFormat.StbiImageFormat"/>.</param>
|
||||||
|
/// <param name="destination">The destination stream.</param>
|
||||||
|
/// <param name="isFloat">True if the pixel format in data is a floating point format.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// This will incur a conversion cost if the pixel format is not a byte format. Affected by global non-thread safe options.
|
||||||
|
/// </remarks>
|
||||||
|
public static void WriteTga<T>(ReadOnlySpan<T> data, int width, int height, StbiImageFormat format, Stream destination, bool isFloat = false)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
int components = Components(format);
|
||||||
|
ReadOnlySpan<byte> source;
|
||||||
|
byte[]? conversion;
|
||||||
|
|
||||||
|
if (isFloat)
|
||||||
|
{
|
||||||
|
conversion = ConvertFloatToByte(data, width, height, components);
|
||||||
|
source = conversion;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
source = MemoryMarshal.AsBytes(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
StbiWriteStreamWrapper wrapper = new StbiWriteStreamWrapper(destination);
|
||||||
|
|
||||||
|
fixed (byte* ptr = source)
|
||||||
|
Stbi.write_tga_to_func(wrapper, null, width, height, components, ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write any image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">Any packed byte or float array structure.</typeparam>
|
||||||
|
/// <param name="data">Span of pixel data.</param>
|
||||||
|
/// <param name="width">Width of the pixel data in pixels.</param>
|
||||||
|
/// <param name="height">Height of the pixel data in pixels.</param>
|
||||||
|
/// <param name="format">Color format of the pixel data. Must not be <see cref="StbiImageFormat.StbiImageFormat"/>.</param>
|
||||||
|
/// <param name="destination">The destination stream.</param>
|
||||||
|
/// <param name="isFloat">True if the pixel format in data is a floating point format.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// This will incur a conversion cost if the pixel format is not a float format. Affected by global non-thread safe options.
|
||||||
|
/// </remarks>
|
||||||
|
public static void WriteHdr<T>(ReadOnlySpan<T> data, int width, int height, StbiImageFormat format, Stream destination, bool isFloat = false)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
int components = Components(format);
|
||||||
|
ReadOnlySpan<float> source;
|
||||||
|
float[]? conversion;
|
||||||
|
|
||||||
|
if (!isFloat)
|
||||||
|
{
|
||||||
|
conversion = ConvertByteToFloat(data, width, height, components);
|
||||||
|
source = conversion;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
source = MemoryMarshal.Cast<T, float>(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
StbiWriteStreamWrapper wrapper = new StbiWriteStreamWrapper(destination);
|
||||||
|
|
||||||
|
fixed (float* ptr = source)
|
||||||
|
Stbi.write_hdr_to_func(wrapper, null, width, height, components, ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Write any image to a PNG file.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">Any packed byte or float array structure.</typeparam>
|
||||||
|
/// <param name="data">Span of pixel data.</param>
|
||||||
|
/// <param name="width">Width of the pixel data in pixels.</param>
|
||||||
|
/// <param name="height">Height of the pixel data in pixels.</param>
|
||||||
|
/// <param name="format">Color format of the pixel data. Must not be <see cref="StbiImageFormat.StbiImageFormat"/>.</param>
|
||||||
|
/// <param name="destination">The destination stream.</param>
|
||||||
|
/// <param name="isFloat">True if the pixel format in data is a floating point format.</param>
|
||||||
|
/// <remarks>
|
||||||
|
/// This will incur a conversion cost if the pixel format is not a byte format. Ignores the alpha channel. Affected by global non-thread safe options.
|
||||||
|
/// </remarks>
|
||||||
|
public static void WriteJpg<T>(ReadOnlySpan<T> data, int width, int height, StbiImageFormat format, Stream destination, int quality = 90, bool isFloat = false)
|
||||||
|
where T : unmanaged
|
||||||
|
{
|
||||||
|
int components = Components(format);
|
||||||
|
ReadOnlySpan<byte> source;
|
||||||
|
byte[]? conversion;
|
||||||
|
|
||||||
|
if (isFloat)
|
||||||
|
{
|
||||||
|
conversion = ConvertFloatToByte(data, width, height, components);
|
||||||
|
source = conversion;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
source = MemoryMarshal.AsBytes(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
StbiWriteStreamWrapper wrapper = new StbiWriteStreamWrapper(destination);
|
||||||
|
|
||||||
|
fixed (byte* ptr = source)
|
||||||
|
Stbi.write_jpg_to_func(wrapper, null, width, height, components, ptr, quality);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -10,14 +10,17 @@ namespace ReFuel.Stb
|
|||||||
|
|
||||||
public unsafe class StbiStreamWrapper : IDisposable
|
public unsafe class StbiStreamWrapper : IDisposable
|
||||||
{
|
{
|
||||||
private Stream _stream;
|
private readonly stbi_io_callbacks _callbacks;
|
||||||
private bool _keepOpen;
|
private readonly Stream _stream;
|
||||||
|
private readonly bool _keepOpen;
|
||||||
private bool _isDisposed;
|
private bool _isDisposed;
|
||||||
|
|
||||||
private StbiReadProc _readCb;
|
private StbiReadProc _readCb;
|
||||||
private StbiSkipProc _skipCb;
|
private StbiSkipProc _skipCb;
|
||||||
private StbiEofProc _eofCb;
|
private StbiEofProc _eofCb;
|
||||||
|
|
||||||
|
public ref readonly stbi_io_callbacks Callbacks => ref _callbacks;
|
||||||
|
|
||||||
public StbiStreamWrapper(Stream stream, bool keepOpen = false)
|
public StbiStreamWrapper(Stream stream, bool keepOpen = false)
|
||||||
{
|
{
|
||||||
if (stream == null) throw new ArgumentNullException(nameof(stream));
|
if (stream == null) throw new ArgumentNullException(nameof(stream));
|
||||||
@ -28,6 +31,11 @@ namespace ReFuel.Stb
|
|||||||
_readCb = ReadCb;
|
_readCb = ReadCb;
|
||||||
_skipCb = SkipCb;
|
_skipCb = SkipCb;
|
||||||
_eofCb = EofCb;
|
_eofCb = EofCb;
|
||||||
|
|
||||||
|
_callbacks = default;
|
||||||
|
_callbacks.read = Marshal.GetFunctionPointerForDelegate<StbiReadProc>(_readCb);
|
||||||
|
_callbacks.skip = Marshal.GetFunctionPointerForDelegate<StbiSkipProc>(_skipCb);
|
||||||
|
_callbacks.eof = Marshal.GetFunctionPointerForDelegate<StbiEofProc>(_eofCb);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void CreateCallbacks(out stbi_io_callbacks cb)
|
public void CreateCallbacks(out stbi_io_callbacks cb)
|
||||||
@ -65,4 +73,28 @@ namespace ReFuel.Stb
|
|||||||
_isDisposed = true;
|
_isDisposed = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal struct StbiWriteStreamWrapper
|
||||||
|
{
|
||||||
|
private readonly Stream _stream;
|
||||||
|
private readonly StbiWriteProc _cb;
|
||||||
|
|
||||||
|
public IntPtr Callback => Marshal.GetFunctionPointerForDelegate(_cb);
|
||||||
|
|
||||||
|
public StbiWriteStreamWrapper(Stream stream)
|
||||||
|
{
|
||||||
|
_stream = stream;
|
||||||
|
unsafe
|
||||||
|
{
|
||||||
|
_cb = WriteCb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private unsafe void WriteCb(void *context, void *data, int size)
|
||||||
|
{
|
||||||
|
_stream.Write(new ReadOnlySpan<byte>((byte*)data, size));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator IntPtr(in StbiWriteStreamWrapper wrapper) => wrapper.Callback;
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user