1
0
mirror of https://github.com/DarthAffe/RGB.NET.git synced 2025-12-13 10:08:31 +00:00

Simplified textures

This commit is contained in:
Darth Affe 2021-02-24 13:05:15 +01:00
parent 8ddf0bc734
commit cc7abb63f0
7 changed files with 85 additions and 129 deletions

View File

@ -15,13 +15,14 @@ namespace RGB.NET.Core
#region Properties & Fields #region Properties & Fields
private readonly int _dataPerColor; private readonly int _dataPerPixel;
private readonly int _stride;
protected ISampler<T> Sampler { get; set; }
protected T[] Data { get; set; }
public ISampler<T> Sampler { get; set; }
public Size Size { get; } public Size Size { get; }
protected abstract ReadOnlySpan<T> Data { get; }
public virtual Color this[in Point point] public virtual Color this[in Point point]
{ {
get get
@ -30,7 +31,7 @@ namespace RGB.NET.Core
int x = (int)Math.Round(Size.Width * point.X.Clamp(0, 1)); int x = (int)Math.Round(Size.Width * point.X.Clamp(0, 1));
int y = (int)Math.Round(Size.Height * point.Y.Clamp(0, 1)); int y = (int)Math.Round(Size.Height * point.Y.Clamp(0, 1));
return GetColor(x, y); return GetColor(GetPixelData(x, y));
} }
} }
@ -45,22 +46,30 @@ namespace RGB.NET.Core
int width = (int)Math.Round(Size.Width * rectangle.Size.Width.Clamp(0, 1)); int width = (int)Math.Round(Size.Width * rectangle.Size.Width.Clamp(0, 1));
int height = (int)Math.Round(Size.Height * rectangle.Size.Height.Clamp(0, 1)); int height = (int)Math.Round(Size.Height * rectangle.Size.Height.Clamp(0, 1));
int bufferSize = width * height * _dataPerColor; int bufferSize = width * height * _dataPerPixel;
if (bufferSize <= STACK_ALLOC_LIMIT) if (bufferSize <= STACK_ALLOC_LIMIT)
{ {
Span<T> buffer = stackalloc T[bufferSize]; Span<T> buffer = stackalloc T[bufferSize];
GetRegionData(x, y, width, height, buffer); GetRegionData(x, y, width, height, buffer);
return Sampler.SampleColor(new SamplerInfo<T>(width, height, buffer));
Span<T> pixelData = stackalloc T[_dataPerPixel];
Sampler.SampleColor(new SamplerInfo<T>(width, height, buffer), pixelData);
return GetColor(pixelData);
} }
else else
{ {
T[] rent = ArrayPool<T>.Shared.Rent(bufferSize); T[] rent = ArrayPool<T>.Shared.Rent(bufferSize);
Span<T> buffer = new Span<T>(rent).Slice(0, bufferSize); Span<T> buffer = new Span<T>(rent).Slice(0, bufferSize);
GetRegionData(x, y, width, height, buffer); GetRegionData(x, y, width, height, buffer);
Color color = Sampler.SampleColor(new SamplerInfo<T>(width, height, buffer));
Span<T> pixelData = stackalloc T[_dataPerPixel];
Sampler.SampleColor(new SamplerInfo<T>(width, height, buffer), pixelData);
ArrayPool<T>.Shared.Return(rent); ArrayPool<T>.Shared.Return(rent);
return color; return GetColor(pixelData);
} }
} }
} }
@ -69,10 +78,10 @@ namespace RGB.NET.Core
#region Constructors #region Constructors
public PixelTexture(int with, int height, T[] data, int dataPerColor, ISampler<T> sampler) public PixelTexture(int with, int height, int dataPerPixel, ISampler<T> sampler)
{ {
this.Data = data; this._stride = with;
this._dataPerColor = dataPerColor; this._dataPerPixel = dataPerPixel;
this.Sampler = sampler; this.Sampler = sampler;
Size = new Size(with, height); Size = new Size(with, height);
@ -82,8 +91,22 @@ namespace RGB.NET.Core
#region Methods #region Methods
protected abstract Color GetColor(int x, int y); protected abstract Color GetColor(ReadOnlySpan<T> pixel);
protected abstract void GetRegionData(int x, int y, int width, int height, in Span<T> buffer);
[MethodImpl(MethodImplOptions.AggressiveInlining)]
protected virtual ReadOnlySpan<T> GetPixelData(int x, int y) => Data.Slice((y * _stride) + x, _dataPerPixel);
protected virtual void GetRegionData(int x, int y, int width, int height, Span<T> buffer)
{
int dataWidth = width * _dataPerPixel;
ReadOnlySpan<T> data = Data;
for (int i = 0; i < height; i++)
{
ReadOnlySpan<T> dataSlice = data.Slice((((y + i) * _stride) + x) * _dataPerPixel, dataWidth);
Span<T> destination = buffer.Slice(i * dataWidth, dataWidth);
dataSlice.CopyTo(destination);
}
}
#endregion #endregion
} }
@ -92,7 +115,9 @@ namespace RGB.NET.Core
{ {
#region Properties & Fields #region Properties & Fields
private readonly int _stride; private readonly Color[] _data;
protected override ReadOnlySpan<Color> Data => _data;
#endregion #endregion
@ -103,9 +128,9 @@ namespace RGB.NET.Core
{ } { }
public PixelTexture(int with, int height, Color[] data, ISampler<Color> sampler) public PixelTexture(int with, int height, Color[] data, ISampler<Color> sampler)
: base(with, height, data, 1, sampler) : base(with, height, 1, sampler)
{ {
this._stride = with; this._data = data;
if (Data.Length != (with * height)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} ({with * height})."); if (Data.Length != (with * height)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} ({with * height}).");
} }
@ -114,19 +139,7 @@ namespace RGB.NET.Core
#region Methods #region Methods
[MethodImpl(MethodImplOptions.AggressiveInlining)] protected override Color GetColor(ReadOnlySpan<Color> pixel) => pixel[0];
protected override Color GetColor(int x, int y) => Data[(y * _stride) + x];
protected override void GetRegionData(int x, int y, int width, int height, in Span<Color> buffer)
{
ReadOnlySpan<Color> data = Data.AsSpan();
for (int i = 0; i < height; i++)
{
ReadOnlySpan<Color> dataSlice = data.Slice(((y + i) * _stride) + x, width);
Span<Color> destination = buffer.Slice(i * width, width);
dataSlice.CopyTo(destination);
}
}
#endregion #endregion
} }

View File

@ -1,19 +1,15 @@
namespace RGB.NET.Core using System;
namespace RGB.NET.Core
{ {
public class AverageColorSampler : ISampler<Color> public class AverageColorSampler : ISampler<Color>
{ {
#region Properties & Fields
public bool SampleAlpha { get; set; }
#endregion
#region Methods #region Methods
public Color SampleColor(SamplerInfo<Color> info) public void SampleColor(in SamplerInfo<Color> info, Span<Color> pixelData)
{ {
int count = info.Width * info.Height; int count = info.Width * info.Height;
if (count == 0) return Color.Transparent; if (count == 0) return;
float a = 0, r = 0, g = 0, b = 0; float a = 0, r = 0, g = 0, b = 0;
foreach (Color color in info.Data) foreach (Color color in info.Data)
@ -24,9 +20,7 @@
b += color.B; b += color.B;
} }
return SampleAlpha pixelData[0] = new Color(a / count, r / count, g / count, b / count);
? new Color(a / count, r / count, g / count, b / count)
: new Color(r / count, g / count, b / count);
} }
#endregion #endregion

View File

@ -1,7 +1,9 @@
namespace RGB.NET.Core using System;
namespace RGB.NET.Core
{ {
public interface ISampler<T> public interface ISampler<T>
{ {
Color SampleColor(SamplerInfo<T> info); void SampleColor(in SamplerInfo<T> info, Span<T> pixelData);
} }
} }

View File

@ -8,22 +8,11 @@ namespace RGB.NET.Presets.Textures
{ {
#region Properties & Fields #region Properties & Fields
private readonly int _stride; private readonly byte[] _data;
protected override ReadOnlySpan<byte> Data => _data;
public ColorFormat ColorFormat { get; } public ColorFormat ColorFormat { get; }
public override Color this[in Rectangle rectangle]
{
get
{
Color color = base[rectangle];
if (ColorFormat == ColorFormat.BGR)
return new Color(color.A, color.B, color.G, color.R);
return color;
}
}
#endregion #endregion
#region Constructors #region Constructors
@ -33,9 +22,9 @@ namespace RGB.NET.Presets.Textures
{ } { }
public BytePixelTexture(int with, int height, byte[] data, ISampler<byte> sampler, ColorFormat colorFormat = ColorFormat.RGB) public BytePixelTexture(int with, int height, byte[] data, ISampler<byte> sampler, ColorFormat colorFormat = ColorFormat.RGB)
: base(with, height, data, 3, sampler) : base(with, height, 3, sampler)
{ {
this._stride = with; this._data = data;
this.ColorFormat = colorFormat; this.ColorFormat = colorFormat;
if (Data.Length != ((with * height) * 3)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} * 3 bytes ({with * height * 3})."); if (Data.Length != ((with * height) * 3)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} * 3 bytes ({with * height * 3}).");
@ -45,26 +34,12 @@ namespace RGB.NET.Presets.Textures
#region Methods #region Methods
protected override Color GetColor(int x, int y) protected override Color GetColor(ReadOnlySpan<byte> pixel)
{ {
int offset = ((y * _stride) + x) * 3;
if (ColorFormat == ColorFormat.BGR) if (ColorFormat == ColorFormat.BGR)
return new Color(Data[offset + 2], Data[offset + 1], Data[offset + 1]); return new Color(pixel[2], pixel[1], pixel[0]);
return new Color(Data[offset], Data[offset + 1], Data[offset + 2]); return new Color(pixel[0], pixel[1], pixel[2]);
}
protected override void GetRegionData(int x, int y, int width, int height, in Span<byte> buffer)
{
int width3 = width * 3;
ReadOnlySpan<byte> data = Data.AsSpan();
for (int i = 0; i < height; i++)
{
ReadOnlySpan<byte> dataSlice = data.Slice((((y + i) * _stride) + x) * 3, width3);
Span<byte> destination = buffer.Slice(i * width3, width3);
dataSlice.CopyTo(destination);
}
} }
#endregion #endregion

View File

@ -8,22 +8,11 @@ namespace RGB.NET.Presets.Textures
{ {
#region Properties & Fields #region Properties & Fields
private readonly int _stride; private readonly float[] _data;
protected override ReadOnlySpan<float> Data => _data;
public ColorFormat ColorFormat { get; } public ColorFormat ColorFormat { get; }
public override Color this[in Rectangle rectangle]
{
get
{
Color color = base[rectangle];
if (ColorFormat == ColorFormat.BGR)
return new Color(color.A, color.B, color.G, color.R);
return color;
}
}
#endregion #endregion
#region Constructors #region Constructors
@ -33,9 +22,9 @@ namespace RGB.NET.Presets.Textures
{ } { }
public FloatPixelTexture(int with, int height, float[] data, ISampler<float> sampler, ColorFormat colorFormat = ColorFormat.RGB) public FloatPixelTexture(int with, int height, float[] data, ISampler<float> sampler, ColorFormat colorFormat = ColorFormat.RGB)
: base(with, height, data, 3, sampler) : base(with, height, 3, sampler)
{ {
this._stride = with; this._data = data;
this.ColorFormat = colorFormat; this.ColorFormat = colorFormat;
if (Data.Length != ((with * height) * 3)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} * 3 bytes ({with * height * 3})."); if (Data.Length != ((with * height) * 3)) throw new ArgumentException($"Data-Length {Data.Length} differs from the given size {with}x{height} * 3 bytes ({with * height * 3}).");
@ -45,26 +34,12 @@ namespace RGB.NET.Presets.Textures
#region Methods #region Methods
protected override Color GetColor(int x, int y) protected override Color GetColor(ReadOnlySpan<float> pixel)
{ {
int offset = ((y * _stride) + x) * 3;
if (ColorFormat == ColorFormat.BGR) if (ColorFormat == ColorFormat.BGR)
return new Color(Data[offset + 2], Data[offset + 1], Data[offset + 1]); return new Color(pixel[2], pixel[1], pixel[0]);
return new Color(Data[offset], Data[offset + 1], Data[offset + 2]); return new Color(pixel[0], pixel[1], pixel[2]);
}
protected override void GetRegionData(int x, int y, int width, int height, in Span<float> buffer)
{
int width3 = width * 3;
ReadOnlySpan<float> data = Data.AsSpan();
for (int i = 0; i < height; i++)
{
ReadOnlySpan<float> dataSlice = data.Slice((((y + i) * _stride) + x) * 3, width3);
Span<float> destination = buffer.Slice(i * width3, width3);
dataSlice.CopyTo(destination);
}
} }
#endregion #endregion

View File

@ -7,23 +7,21 @@ namespace RGB.NET.Presets.Textures.Sampler
{ {
#region Methods #region Methods
public Color SampleColor(SamplerInfo<byte> info) public void SampleColor(in SamplerInfo<byte> info, Span<byte> pixelData)
{ {
int count = info.Width * info.Height; int count = info.Width * info.Height;
if (count == 0) return Color.Transparent; if (count == 0) return;
ReadOnlySpan<byte> data = info.Data; ReadOnlySpan<byte> data = info.Data;
uint r = 0, g = 0, b = 0; int dataLength = pixelData.Length;
for (int i = 0; i < data.Length; i += 3) Span<uint> sums = stackalloc uint[dataLength];
{ for (int i = 0; i < data.Length; i += dataLength)
r += data[i]; for (int j = 0; j < sums.Length; j++)
g += data[i + 1]; sums[j] += data[i + j];
b += data[i + 2];
}
float divisor = count * byte.MaxValue; for (int i = 0; i < pixelData.Length; i++)
return new Color(r / divisor, g / divisor, b / divisor); pixelData[i] = (byte)MathF.Round(sums[i] / (float)count);
} }
#endregion #endregion

View File

@ -7,22 +7,21 @@ namespace RGB.NET.Presets.Textures.Sampler
{ {
#region Methods #region Methods
public Color SampleColor(SamplerInfo<float> info) public void SampleColor(in SamplerInfo<float> info, Span<float> pixelData)
{ {
int count = info.Width * info.Height; int count = info.Width * info.Height;
if (count == 0) return Color.Transparent; if (count == 0) return;
ReadOnlySpan<float> data = info.Data; ReadOnlySpan<float> data = info.Data;
float r = 0, g = 0, b = 0; int dataLength = pixelData.Length;
for (int i = 0; i < data.Length; i += 3) Span<float> sums = stackalloc float[dataLength];
{ for (int i = 0; i < data.Length; i += dataLength)
r += data[i]; for (int j = 0; j < sums.Length; j++)
g += data[i + 1]; sums[j] += data[i + j];
b += data[i + 2];
}
return new Color(r / count, g / count, b / count); for (int i = 0; i < pixelData.Length; i++)
pixelData[i] = sums[i] / count;
} }
#endregion #endregion