mirror of https://github.com/SixLabors/ImageSharp
committed by
GitHub
25 changed files with 935 additions and 211 deletions
@ -0,0 +1,25 @@ |
|||
// Copyright (c) Six Labors.
|
|||
// Licensed under the Apache License, Version 2.0.
|
|||
|
|||
namespace SixLabors.ImageSharp.Processing.Processors.Convolution |
|||
{ |
|||
/// <summary>
|
|||
/// Wrapping mode for the border pixels in convolution processing.
|
|||
/// </summary>
|
|||
public enum BorderWrappingMode : byte |
|||
{ |
|||
/// <summary>Repeat the border pixel value: aaaaaa|abcdefgh|hhhhhhh</summary>
|
|||
Repeat = 0, |
|||
|
|||
/// <summary>Take values from the opposite edge: cdefgh|abcdefgh|abcdefg</summary>
|
|||
Wrap = 1, |
|||
|
|||
/// <summary>Mirror the last few border values: fedcba|abcdefgh|hgfedcb</summary>
|
|||
/// <remarks>This Mode is similar to <see cref="Bounce"/>, but here the very border pixel is repeated.</remarks>
|
|||
Mirror = 2, |
|||
|
|||
/// <summary>Bounce off the border: fedcb|abcdefgh|gfedcb</summary>
|
|||
/// <remarks>This Mode is similar to <see cref="Mirror"/>, but here the very border pixel is not repeated.</remarks>
|
|||
Bounce = 3 |
|||
} |
|||
} |
|||
@ -0,0 +1,421 @@ |
|||
// Copyright (c) Six Labors.
|
|||
// Licensed under the Apache License, Version 2.0.
|
|||
|
|||
using SixLabors.ImageSharp.Processing.Processors.Convolution; |
|||
using Xunit; |
|||
|
|||
namespace SixLabors.ImageSharp.Tests.Processing.Convolution |
|||
{ |
|||
[Trait("Category", "Processors")] |
|||
public class KernelSamplingMapTest |
|||
{ |
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7RepeatBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Repeat; |
|||
int[] expected = |
|||
{ |
|||
0, 0, 0, 1, 2, |
|||
0, 0, 1, 2, 3, |
|||
0, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 6, |
|||
4, 5, 6, 6, 6, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7BounceBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Bounce; |
|||
int[] expected = |
|||
{ |
|||
2, 1, 0, 1, 2, |
|||
1, 0, 1, 2, 3, |
|||
0, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 5, |
|||
4, 5, 6, 5, 4, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7MirrorBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Mirror; |
|||
int[] expected = |
|||
{ |
|||
1, 0, 0, 1, 2, |
|||
0, 0, 1, 2, 3, |
|||
0, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 6, |
|||
4, 5, 6, 6, 5, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7WrapBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] expected = |
|||
{ |
|||
5, 6, 0, 1, 2, |
|||
6, 0, 1, 2, 3, |
|||
0, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 0, |
|||
4, 5, 6, 0, 1, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image9x9BounceBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(1, 1, 9, 9); |
|||
var mode = BorderWrappingMode.Bounce; |
|||
int[] expected = |
|||
{ |
|||
3, 2, 1, 2, 3, |
|||
2, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 9, |
|||
6, 7, 8, 9, 8, |
|||
7, 8, 9, 8, 7, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image9x9MirrorBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(1, 1, 9, 9); |
|||
var mode = BorderWrappingMode.Mirror; |
|||
int[] expected = |
|||
{ |
|||
2, 1, 1, 2, 3, |
|||
1, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 9, |
|||
6, 7, 8, 9, 9, |
|||
7, 8, 9, 9, 8, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image9x9WrapBorder() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(1, 1, 9, 9); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] expected = |
|||
{ |
|||
8, 9, 1, 2, 3, |
|||
9, 1, 2, 3, 4, |
|||
1, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 9, |
|||
6, 7, 8, 9, 1, |
|||
7, 8, 9, 1, 2, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7RepeatBorderTile() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Repeat; |
|||
int[] expected = |
|||
{ |
|||
2, 2, 2, 3, 4, |
|||
2, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 8, |
|||
6, 7, 8, 8, 8, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7BounceBorderTile() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Bounce; |
|||
int[] expected = |
|||
{ |
|||
4, 3, 2, 3, 4, |
|||
3, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 7, |
|||
6, 7, 8, 7, 6, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7MirrorBorderTile() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Mirror; |
|||
int[] expected = |
|||
{ |
|||
3, 2, 2, 3, 4, |
|||
2, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 8, |
|||
6, 7, 8, 8, 7, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel5Image7x7WrapBorderTile() |
|||
{ |
|||
var kernelSize = new Size(5, 5); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] expected = |
|||
{ |
|||
7, 8, 2, 3, 4, |
|||
8, 2, 3, 4, 5, |
|||
2, 3, 4, 5, 6, |
|||
3, 4, 5, 6, 7, |
|||
4, 5, 6, 7, 8, |
|||
5, 6, 7, 8, 2, |
|||
6, 7, 8, 2, 3, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7RepeatBorder() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Repeat; |
|||
int[] expected = |
|||
{ |
|||
0, 0, 1, |
|||
0, 1, 2, |
|||
1, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 6, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7BounceBorder() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Bounce; |
|||
int[] expected = |
|||
{ |
|||
1, 0, 1, |
|||
0, 1, 2, |
|||
1, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 5, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7MirrorBorder() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Mirror; |
|||
int[] expected = |
|||
{ |
|||
0, 0, 1, |
|||
0, 1, 2, |
|||
1, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 6, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7WrapBorder() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(0, 0, 7, 7); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] expected = |
|||
{ |
|||
6, 0, 1, |
|||
0, 1, 2, |
|||
1, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 0, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7RepeatBorderTile() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Repeat; |
|||
int[] expected = |
|||
{ |
|||
2, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 7, |
|||
6, 7, 8, |
|||
7, 8, 8, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7BounceBorderTile() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Bounce; |
|||
int[] expected = |
|||
{ |
|||
3, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 7, |
|||
6, 7, 8, |
|||
7, 8, 7, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7MirrorBorderTile() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Mirror; |
|||
int[] expected = |
|||
{ |
|||
2, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 7, |
|||
6, 7, 8, |
|||
7, 8, 8, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x7WrapBorderTile() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(2, 2, 7, 7); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] expected = |
|||
{ |
|||
8, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 7, |
|||
6, 7, 8, |
|||
7, 8, 2, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, expected, expected); |
|||
} |
|||
|
|||
[Fact] |
|||
public void KernalSamplingMap_Kernel3Image7x5WrapBorderTile() |
|||
{ |
|||
var kernelSize = new Size(3, 3); |
|||
var bounds = new Rectangle(2, 2, 7, 5); |
|||
var mode = BorderWrappingMode.Wrap; |
|||
int[] xExpected = |
|||
{ |
|||
8, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 7, |
|||
6, 7, 8, |
|||
7, 8, 2, |
|||
}; |
|||
int[] yExpected = |
|||
{ |
|||
6, 2, 3, |
|||
2, 3, 4, |
|||
3, 4, 5, |
|||
4, 5, 6, |
|||
5, 6, 2, |
|||
}; |
|||
this.AssertOffsets(kernelSize, bounds, mode, mode, xExpected, yExpected); |
|||
} |
|||
|
|||
private void AssertOffsets(Size kernelSize, Rectangle bounds, BorderWrappingMode xBorderMode, BorderWrappingMode yBorderMode, int[] xExpected, int[] yExpected) |
|||
{ |
|||
// Arrange
|
|||
var map = new KernelSamplingMap(Configuration.Default.MemoryAllocator); |
|||
|
|||
// Act
|
|||
map.BuildSamplingOffsetMap(kernelSize.Height, kernelSize.Width, bounds, xBorderMode, yBorderMode); |
|||
|
|||
// Assert
|
|||
var xOffsets = map.GetColumnOffsetSpan().ToArray(); |
|||
Assert.Equal(xExpected, xOffsets); |
|||
var yOffsets = map.GetRowOffsetSpan().ToArray(); |
|||
Assert.Equal(yExpected, yOffsets); |
|||
} |
|||
} |
|||
} |
|||
Loading…
Reference in new issue