@@ -21,6 +21,11 @@ public class ImageBrush : Brush
21
21
/// </summary>
22
22
private readonly RectangleF region ;
23
23
24
+ /// <summary>
25
+ /// The offet to apply to the source image while applying the imagebrush
26
+ /// </summary>
27
+ private readonly Point offset ;
28
+
24
29
/// <summary>
25
30
/// Initializes a new instance of the <see cref="ImageBrush"/> class.
26
31
/// </summary>
@@ -33,12 +38,44 @@ public ImageBrush(Image image)
33
38
/// <summary>
34
39
/// Initializes a new instance of the <see cref="ImageBrush"/> class.
35
40
/// </summary>
36
- /// <param name="image">The source image.</param>
37
- /// <param name="region">The region of interest within the source image to draw.</param>
41
+ /// <param name="image">The image.</param>
42
+ /// <param name="offset">
43
+ /// An offset to apply the to image image while drawing apply the texture.
44
+ /// </param>
45
+ public ImageBrush ( Image image , Point offset )
46
+ : this ( image , image . Bounds , offset )
47
+ {
48
+ }
49
+
50
+ /// <summary>
51
+ /// Initializes a new instance of the <see cref="ImageBrush"/> class.
52
+ /// </summary>
53
+ /// <param name="image">The image.</param>
54
+ /// <param name="region">
55
+ /// The region of interest.
56
+ /// This overrides any region used to initialize the brush applicator.
57
+ /// </param>
38
58
public ImageBrush ( Image image , RectangleF region )
59
+ : this ( image , region , Point . Empty )
60
+ {
61
+ }
62
+
63
+ /// <summary>
64
+ /// Initializes a new instance of the <see cref="ImageBrush"/> class.
65
+ /// </summary>
66
+ /// <param name="image">The image.</param>
67
+ /// <param name="region">
68
+ /// The region of interest.
69
+ /// This overrides any region used to initialize the brush applicator.
70
+ /// </param>
71
+ /// <param name="offset">
72
+ /// An offset to apply the to image image while drawing apply the texture.
73
+ /// </param>
74
+ public ImageBrush ( Image image , RectangleF region , Point offset )
39
75
{
40
76
this . image = image ;
41
77
this . region = RectangleF . Intersect ( image . Bounds , region ) ;
78
+ this . offset = offset ;
42
79
}
43
80
44
81
/// <inheritdoc />
@@ -64,11 +101,11 @@ public override BrushApplicator<TPixel> CreateApplicator<TPixel>(
64
101
{
65
102
if ( this . image is Image < TPixel > specificImage )
66
103
{
67
- return new ImageBrushApplicator < TPixel > ( configuration , options , source , specificImage , region , this . region , false ) ;
104
+ return new ImageBrushApplicator < TPixel > ( configuration , options , source , specificImage , region , this . region , this . offset , false ) ;
68
105
}
69
106
70
107
specificImage = this . image . CloneAs < TPixel > ( ) ;
71
- return new ImageBrushApplicator < TPixel > ( configuration , options , source , specificImage , region , this . region , true ) ;
108
+ return new ImageBrushApplicator < TPixel > ( configuration , options , source , specificImage , region , this . region , this . offset , true ) ;
72
109
}
73
110
74
111
/// <summary>
@@ -107,6 +144,7 @@ private class ImageBrushApplicator<TPixel> : BrushApplicator<TPixel>
107
144
/// <param name="image">The image.</param>
108
145
/// <param name="targetRegion">The region of the target image we will be drawing to.</param>
109
146
/// <param name="sourceRegion">The region of the source image we will be using to source pixels to draw from.</param>
147
+ /// <param name="offset">An offset to apply to the texture while drawing.</param>
110
148
/// <param name="shouldDisposeImage">Whether to dispose the image on disposal of the applicator.</param>
111
149
public ImageBrushApplicator (
112
150
Configuration configuration ,
@@ -115,6 +153,7 @@ public ImageBrushApplicator(
115
153
Image < TPixel > image ,
116
154
RectangleF targetRegion ,
117
155
RectangleF sourceRegion ,
156
+ Point offset ,
118
157
bool shouldDisposeImage )
119
158
: base ( configuration , options , target )
120
159
{
@@ -124,8 +163,8 @@ public ImageBrushApplicator(
124
163
125
164
this . sourceRegion = Rectangle . Intersect ( image . Bounds , ( Rectangle ) sourceRegion ) ;
126
165
127
- this . offsetY = ( int ) MathF . Max ( MathF . Floor ( targetRegion . Top ) , 0 ) ;
128
- this . offsetX = ( int ) MathF . Max ( MathF . Floor ( targetRegion . Left ) , 0 ) ;
166
+ this . offsetY = ( int ) MathF . Floor ( targetRegion . Top ) + offset . Y ;
167
+ this . offsetX = ( int ) MathF . Floor ( targetRegion . Left ) + offset . X ;
129
168
}
130
169
131
170
internal TPixel this [ int x , int y ]
@@ -166,14 +205,18 @@ public override void Apply(Span<float> scanline, int x, int y)
166
205
Span < TPixel > overlaySpan = overlay . Memory . Span ;
167
206
168
207
int offsetX = x - this . offsetX ;
169
- int sourceY = ( ( y - this . offsetY ) % this . sourceRegion . Height ) + this . sourceRegion . Y ;
208
+ int sourceY = ( ( ( ( y - this . offsetY ) % this . sourceRegion . Height ) // clamp the number between -height and +height
209
+ + this . sourceRegion . Height ) % this . sourceRegion . Height ) // clamp the number between 0 and +height
210
+ + this . sourceRegion . Y ;
170
211
Span < TPixel > sourceRow = this . sourceFrame . PixelBuffer . DangerousGetRowSpan ( sourceY ) ;
171
212
172
213
for ( int i = 0 ; i < scanline . Length ; i ++ )
173
214
{
174
215
amountSpan [ i ] = scanline [ i ] * this . Options . BlendPercentage ;
175
216
176
- int sourceX = ( ( i + offsetX ) % this . sourceRegion . Width ) + this . sourceRegion . X ;
217
+ int sourceX = ( ( ( ( i + offsetX ) % this . sourceRegion . Width ) // clamp the number between -width and +width
218
+ + this . sourceRegion . Width ) % this . sourceRegion . Width ) // clamp the number between 0 and +width
219
+ + this . sourceRegion . X ;
177
220
178
221
overlaySpan [ i ] = sourceRow [ sourceX ] ;
179
222
}
0 commit comments