Skip to content

Commit fda6302

Browse files
Merge pull request #309 from SixLabors/sw/image-brush-offsets
Ensure the negatively offset shapes are correctly offset the ImageBrush Texture during rendering.
2 parents 3fb38a4 + 184fe2e commit fda6302

File tree

11 files changed

+155
-15
lines changed

11 files changed

+155
-15
lines changed

src/ImageSharp.Drawing/Processing/ImageBrush.cs

+51-8
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,11 @@ public class ImageBrush : Brush
2121
/// </summary>
2222
private readonly RectangleF region;
2323

24+
/// <summary>
25+
/// The offet to apply to the source image while applying the imagebrush
26+
/// </summary>
27+
private readonly Point offset;
28+
2429
/// <summary>
2530
/// Initializes a new instance of the <see cref="ImageBrush"/> class.
2631
/// </summary>
@@ -33,12 +38,44 @@ public ImageBrush(Image image)
3338
/// <summary>
3439
/// Initializes a new instance of the <see cref="ImageBrush"/> class.
3540
/// </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>
3858
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)
3975
{
4076
this.image = image;
4177
this.region = RectangleF.Intersect(image.Bounds, region);
78+
this.offset = offset;
4279
}
4380

4481
/// <inheritdoc />
@@ -64,11 +101,11 @@ public override BrushApplicator<TPixel> CreateApplicator<TPixel>(
64101
{
65102
if (this.image is Image<TPixel> specificImage)
66103
{
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);
68105
}
69106

70107
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);
72109
}
73110

74111
/// <summary>
@@ -107,6 +144,7 @@ private class ImageBrushApplicator<TPixel> : BrushApplicator<TPixel>
107144
/// <param name="image">The image.</param>
108145
/// <param name="targetRegion">The region of the target image we will be drawing to.</param>
109146
/// <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>
110148
/// <param name="shouldDisposeImage">Whether to dispose the image on disposal of the applicator.</param>
111149
public ImageBrushApplicator(
112150
Configuration configuration,
@@ -115,6 +153,7 @@ public ImageBrushApplicator(
115153
Image<TPixel> image,
116154
RectangleF targetRegion,
117155
RectangleF sourceRegion,
156+
Point offset,
118157
bool shouldDisposeImage)
119158
: base(configuration, options, target)
120159
{
@@ -124,8 +163,8 @@ public ImageBrushApplicator(
124163

125164
this.sourceRegion = Rectangle.Intersect(image.Bounds, (Rectangle)sourceRegion);
126165

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;
129168
}
130169

131170
internal TPixel this[int x, int y]
@@ -166,14 +205,18 @@ public override void Apply(Span<float> scanline, int x, int y)
166205
Span<TPixel> overlaySpan = overlay.Memory.Span;
167206

168207
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;
170211
Span<TPixel> sourceRow = this.sourceFrame.PixelBuffer.DangerousGetRowSpan(sourceY);
171212

172213
for (int i = 0; i < scanline.Length; i++)
173214
{
174215
amountSpan[i] = scanline[i] * this.Options.BlendPercentage;
175216

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;
177220

178221
overlaySpan[i] = sourceRow[sourceX];
179222
}

src/ImageSharp.Drawing/Processing/Processors/Drawing/ClipPathProcessor{TPixel}.cs

+15-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,21 @@ public void Execute()
3939
// Use an image brush to apply cloned image as the source for filling the shape.
4040
// We pass explicit bounds to avoid the need to crop the clone;
4141
RectangleF bounds = this.definition.Region.Bounds;
42-
var brush = new ImageBrush(clone, bounds);
42+
43+
// add some clamping offsets to the brush to account for the target drawing location due to the cloned image not fill the image as expected
44+
var offsetX = 0;
45+
var offsetY = 0;
46+
if (bounds.X < 0)
47+
{
48+
offsetX = -(int)MathF.Floor(bounds.X);
49+
}
50+
51+
if (bounds.Y < 0)
52+
{
53+
offsetY = -(int)MathF.Floor(bounds.Y);
54+
}
55+
56+
var brush = new ImageBrush(clone, bounds, new Point(offsetX, offsetY));
4357

4458
// Grab hold of an image processor that can fill paths with a brush to allow it to do the hard pixel pushing for us
4559
var processor = new FillPathProcessor(this.definition.Options, brush, this.definition.Region);

src/ImageSharp.Drawing/Processing/Processors/Drawing/FillPathProcessor.cs

+1-3
Original file line numberDiff line numberDiff line change
@@ -56,11 +56,9 @@ public IImageProcessor<TPixel> CreatePixelSpecificProcessor<TPixel>(Configuratio
5656
var rect = (Rectangle)rectF;
5757
if (!this.Options.GraphicsOptions.Antialias || rectF == rect)
5858
{
59-
var interest = Rectangle.Intersect(sourceRectangle, rect);
60-
6159
// Cast as in and back are the same or we are using anti-aliasing
6260
return new FillProcessor(this.Options, this.Brush)
63-
.CreatePixelSpecificProcessor(configuration, source, interest);
61+
.CreatePixelSpecificProcessor(configuration, source, rect);
6462
}
6563
}
6664

src/ImageSharp.Drawing/Processing/Processors/Drawing/FillPathProcessor{TPixel}.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ protected override void OnFrameApply(ImageFrame<TPixel> source)
7474
subpixelCount = Math.Max(subpixelCount, graphicsOptions.AntialiasSubpixelDepth);
7575
}
7676

77-
using BrushApplicator<TPixel> applicator = brush.CreateApplicator(configuration, graphicsOptions, source, interest);
77+
using BrushApplicator<TPixel> applicator = brush.CreateApplicator(configuration, graphicsOptions, source, this.bounds);
7878
int scanlineWidth = interest.Width;
7979
MemoryAllocator allocator = this.Configuration.MemoryAllocator;
8080
bool scanlineDirty = true;

src/ImageSharp.Drawing/Processing/Processors/Drawing/FillProcessor{TPixel}.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ protected override void OnFrameApply(ImageFrame<TPixel> source)
5858
configuration,
5959
options,
6060
source,
61-
interest);
61+
this.SourceRectangle);
6262

6363
amount.Memory.Span.Fill(1F);
6464

tests/ImageSharp.Drawing.Tests/Drawing/ClipTests.cs

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ public class ClipTests
1414
[Theory]
1515
[WithBasicTestPatternImages(250, 350, PixelTypes.Rgba32, 0, 0, 0.5)]
1616
[WithBasicTestPatternImages(250, 350, PixelTypes.Rgba32, -20, -20, 0.5)]
17+
[WithBasicTestPatternImages(250, 350, PixelTypes.Rgba32, -20, -100, 0.5)]
1718
[WithBasicTestPatternImages(250, 350, PixelTypes.Rgba32, 20, 20, 0.5)]
1819
[WithBasicTestPatternImages(250, 350, PixelTypes.Rgba32, 40, 60, 0.2)]
1920
public void Clip<TPixel>(TestImageProvider<TPixel> provider, float dx, float dy, float sizeMult)

tests/ImageSharp.Drawing.Tests/Drawing/FillImageBrushTests.cs

+73-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
// Copyright (c) Six Labors.
22
// Licensed under the Six Labors Split License.
33

4+
using System.Drawing;
5+
using SixLabors.ImageSharp.Advanced;
46
using SixLabors.ImageSharp.Drawing.Processing;
57
using SixLabors.ImageSharp.PixelFormats;
68
using SixLabors.ImageSharp.Processing;
@@ -72,11 +74,81 @@ public void CanDrawPortraitImage<TPixel>(TestImageProvider<TPixel> provider)
7274

7375
overlay.Mutate(c => c.Crop(new Rectangle(0, 0, 90, 125)));
7476

75-
ImageBrush brush = new(overlay);
77+
var brush = new ImageBrush(overlay);
78+
background.Mutate(c => c.Fill(brush));
79+
80+
background.DebugSave(provider, appendSourceFileOrDescription: false);
81+
background.CompareToReferenceOutput(provider, appendSourceFileOrDescription: false);
82+
}
83+
84+
[Theory]
85+
[WithTestPatternImage(400, 400, PixelTypes.Rgba32)]
86+
public void CanOffsetImage<TPixel>(TestImageProvider<TPixel> provider)
87+
where TPixel : unmanaged, IPixel<TPixel>
88+
{
89+
byte[] data = TestFile.Create(TestImages.Png.Ducky).Bytes;
90+
using Image<TPixel> background = provider.GetImage();
91+
using Image overlay = Image.Load<Rgba32>(data);
92+
93+
var brush = new ImageBrush(overlay);
94+
background.Mutate(c => c.Fill(brush, new RectangularPolygon(0, 0, 400, 200)));
95+
background.Mutate(c => c.Fill(brush, new RectangularPolygon(-100, 200, 500, 200)));
96+
97+
background.DebugSave(provider, appendSourceFileOrDescription: false);
98+
background.CompareToReferenceOutput(provider, appendSourceFileOrDescription: false);
99+
}
100+
101+
[Theory]
102+
[WithTestPatternImage(400, 400, PixelTypes.Rgba32)]
103+
public void CanOffsetViaBrushImage<TPixel>(TestImageProvider<TPixel> provider)
104+
where TPixel : unmanaged, IPixel<TPixel>
105+
{
106+
byte[] data = TestFile.Create(TestImages.Png.Ducky).Bytes;
107+
using Image<TPixel> background = provider.GetImage();
108+
using Image overlay = Image.Load<Rgba32>(data);
109+
110+
var brush = new ImageBrush(overlay);
111+
var brushOffset = new ImageBrush(overlay, new Point(100, 0));
112+
background.Mutate(c => c.Fill(brush, new RectangularPolygon(0, 0, 400, 200)));
113+
background.Mutate(c => c.Fill(brushOffset, new RectangularPolygon(0, 200, 400, 200)));
114+
115+
background.DebugSave(provider, appendSourceFileOrDescription: false);
116+
background.CompareToReferenceOutput(provider, appendSourceFileOrDescription: false);
117+
}
118+
119+
[Theory]
120+
[WithSolidFilledImages(1000, 1000, "White", PixelTypes.Rgba32)]
121+
public void CanDrawOffsetImage<TPixel>(TestImageProvider<TPixel> provider)
122+
where TPixel : unmanaged, IPixel<TPixel>
123+
{
124+
byte[] data = TestFile.Create(TestImages.Png.Ducky).Bytes;
125+
using Image<TPixel> background = provider.GetImage();
126+
127+
using Image templateImage = Image.Load<Rgba32>(data);
128+
using Image finalTexture = BuildMultiRowTexture(templateImage);
129+
130+
finalTexture.Mutate(c => c.Resize(100, 200));
131+
132+
ImageBrush brush = new(finalTexture);
76133
background.Mutate(c => c.Fill(brush));
77134

78135
background.DebugSave(provider, appendSourceFileOrDescription: false);
79136
background.CompareToReferenceOutput(provider, appendSourceFileOrDescription: false);
137+
138+
Image BuildMultiRowTexture(Image sourceTexture)
139+
{
140+
int halfWidth = sourceTexture.Width / 2;
141+
142+
Image final = sourceTexture.Clone(x => x.Resize(new ResizeOptions
143+
{
144+
Size = new Size(templateImage.Width, templateImage.Height * 2),
145+
Position = AnchorPositionMode.TopLeft,
146+
Mode = ResizeMode.Pad,
147+
})
148+
.DrawImage(templateImage, new Point(halfWidth, sourceTexture.Height), new Rectangle(0, 0, halfWidth, sourceTexture.Height), 1)
149+
.DrawImage(templateImage, new Point(0, templateImage.Height), new Rectangle(halfWidth, 0, halfWidth, sourceTexture.Height), 1));
150+
return final;
151+
}
80152
}
81153

82154
[Theory]
Loading
Loading

0 commit comments

Comments
 (0)