mirror of
https://github.com/DarthAffe/HPPH.git
synced 2025-12-12 21:38:59 +00:00
210 lines
6.2 KiB
C#
210 lines
6.2 KiB
C#
using System.Collections;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace HPPH;
|
|
|
|
/// <inheritdoc />
|
|
[SkipLocalsInit]
|
|
public sealed class Image<T> : IImage<T>
|
|
where T : struct, IColor
|
|
{
|
|
#region Properties & Fields
|
|
|
|
private readonly byte[] _buffer;
|
|
|
|
private readonly int _x;
|
|
private readonly int _y;
|
|
private readonly int _stride;
|
|
|
|
/// <inheritdoc />
|
|
public IColorFormat ColorFormat
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => T.ColorFormat;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public int Width { get; }
|
|
|
|
/// <inheritdoc />
|
|
public int Height { get; }
|
|
|
|
/// <inheritdoc />
|
|
public int SizeInBytes => Width * Height * ColorFormat.BytesPerPixel;
|
|
|
|
#endregion
|
|
|
|
#region Indexer
|
|
|
|
IColor IImage.this[int x, int y] => this[x, y];
|
|
|
|
/// <inheritdoc />
|
|
public ref readonly T this[int x, int y]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
if ((x < 0) || (y < 0) || (x >= Width) || (y >= Height)) throw new IndexOutOfRangeException();
|
|
|
|
return ref Unsafe.Add(ref Unsafe.As<byte, T>(ref Unsafe.Add(ref MemoryMarshal.GetReference(_buffer.AsSpan()), (nint)(uint)((_y + y) * _stride))), (nint)(uint)(_x + x));
|
|
}
|
|
}
|
|
|
|
|
|
IImage IImage.this[int x, int y, int width, int height]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
if ((x < 0) || (y < 0) || (width <= 0) || (height <= 0) || ((x + width) > Width) || ((y + height) > Height)) throw new IndexOutOfRangeException();
|
|
|
|
return new Image<T>(_buffer, _x + x, _y + y, width, height, _stride);
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public RefImage<T> this[int x, int y, int width, int height]
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get
|
|
{
|
|
if ((x < 0) || (y < 0) || (width <= 0) || (height <= 0) || ((x + width) > Width) || ((y + height) > Height)) throw new IndexOutOfRangeException();
|
|
|
|
return new RefImage<T>(_buffer, _x + x, _y + y, width, height, _stride);
|
|
}
|
|
}
|
|
|
|
IImageRows IImage.Rows => new IColorImageRows<T>(_buffer, _x, _y, Width, Height, _stride);
|
|
|
|
/// <inheritdoc />
|
|
public ImageRows<T> Rows
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => new(_buffer, _x, _y, Width, Height, _stride);
|
|
}
|
|
|
|
IImageColumns IImage.Columns => new IColorImageColumns<T>(_buffer, _x, _y, Width, Height, _stride);
|
|
/// <inheritdoc />
|
|
public ImageColumns<T> Columns
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => new(_buffer, _x, _y, Width, Height, _stride);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Constructors
|
|
|
|
private Image(byte[] buffer, int x, int y, int width, int height, int stride)
|
|
{
|
|
this._buffer = buffer;
|
|
this._x = x;
|
|
this._y = y;
|
|
this.Width = width;
|
|
this.Height = height;
|
|
this._stride = stride;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Methods
|
|
|
|
public static Image<T> Create(ReadOnlySpan<T> buffer, int width, int height)
|
|
=> Create(MemoryMarshal.AsBytes(buffer), width, height, width * T.ColorFormat.BytesPerPixel);
|
|
|
|
public static Image<T> Create(ReadOnlySpan<byte> buffer, int width, int height, int stride)
|
|
{
|
|
if (stride < width) throw new ArgumentException("Stride can't be smaller than width.");
|
|
if (buffer.Length < (height * stride)) throw new ArgumentException("Not enough data in the buffer.");
|
|
|
|
byte[] data = new byte[buffer.Length];
|
|
buffer.CopyTo(data);
|
|
return new Image<T>(data, 0, 0, width, height, stride);
|
|
}
|
|
|
|
public void CopyTo(Span<T> destination) => CopyTo(MemoryMarshal.AsBytes(destination));
|
|
|
|
/// <inheritdoc />
|
|
public void CopyTo(Span<byte> destination)
|
|
{
|
|
if (destination == null) throw new ArgumentNullException(nameof(destination));
|
|
if (destination.Length < SizeInBytes) throw new ArgumentException("The destination is too small to fit this image.", nameof(destination));
|
|
|
|
int targetStride = Width * ColorFormat.BytesPerPixel;
|
|
ImageRows<T> rows = Rows;
|
|
Span<byte> target = destination;
|
|
foreach (ImageRow<T> row in rows)
|
|
{
|
|
row.CopyTo(target);
|
|
target = target[targetStride..];
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public byte[] ToRawArray()
|
|
{
|
|
byte[] array = new byte[SizeInBytes];
|
|
CopyTo(array);
|
|
return array;
|
|
}
|
|
|
|
public T[] ToArray()
|
|
{
|
|
T[] colors = new T[Width * Height];
|
|
CopyTo(colors);
|
|
return colors;
|
|
}
|
|
|
|
//TODO DarthAffe 11.07.2024: This has some potential for optimization
|
|
IColor[] IImage.ToArray()
|
|
{
|
|
IColor[] colors = new IColor[Width * Height];
|
|
|
|
int counter = 0;
|
|
foreach (ImageRow<T> row in Rows)
|
|
foreach (T color in row)
|
|
colors[counter++] = color;
|
|
|
|
return colors;
|
|
}
|
|
|
|
public RefImage<T> AsRefImage() => new(_buffer, _x, _y, Width, Height, _stride);
|
|
|
|
public RefImage<TColor> AsRefImage<TColor>()
|
|
where TColor : struct, IColor
|
|
{
|
|
if (typeof(TColor) != typeof(T)) throw new ArgumentException("The requested color format does not fit this image.", nameof(TColor));
|
|
|
|
return new RefImage<TColor>(_buffer, _x, _y, Width, Height, _stride);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a reference to the first element of this image inside the full image buffer.
|
|
/// </summary>
|
|
public ref readonly byte GetPinnableReference()
|
|
{
|
|
if (_buffer.Length == 0)
|
|
return ref Unsafe.NullRef<byte>();
|
|
|
|
return ref MemoryMarshal.GetReference(new ReadOnlySpan<byte>(_buffer)[((_y * _stride) + (_x * ColorFormat.BytesPerPixel))..]);
|
|
}
|
|
|
|
public IEnumerator<T> GetEnumerator()
|
|
{
|
|
for (int y = 0; y < Height; y++)
|
|
for (int x = 0; x < Width; x++)
|
|
yield return this[x, y];
|
|
}
|
|
|
|
IEnumerator<IColor> IEnumerable<IColor>.GetEnumerator()
|
|
{
|
|
for (int y = 0; y < Height; y++)
|
|
for (int x = 0; x < Width; x++)
|
|
yield return this[x, y];
|
|
}
|
|
|
|
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
|
|
#endregion
|
|
} |