mirror of https://github.com/SixLabors/ImageSharp
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
177 lines
6.7 KiB
177 lines
6.7 KiB
// Copyright (c) Six Labors and contributors.
|
|
// Licensed under the Apache License, Version 2.0.
|
|
|
|
using System;
|
|
using System.Buffers;
|
|
using System.Numerics;
|
|
using SixLabors.ImageSharp.Advanced;
|
|
using SixLabors.ImageSharp.PixelFormats;
|
|
using SixLabors.ImageSharp.Primitives;
|
|
using SixLabors.Memory;
|
|
using SixLabors.Primitives;
|
|
|
|
namespace SixLabors.ImageSharp.Processing
|
|
{
|
|
/// <summary>
|
|
/// Provides an implementation of a pattern brush for painting patterns.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// The patterns that are used to create a custom pattern brush are made up of a repeating matrix of flags,
|
|
/// where each flag denotes whether to draw the foreground color or the background color.
|
|
/// so to create a new bool[,] with your flags
|
|
/// <para>
|
|
/// For example if you wanted to create a diagonal line that repeat every 4 pixels you would use a pattern like so
|
|
/// 1000
|
|
/// 0100
|
|
/// 0010
|
|
/// 0001
|
|
/// </para>
|
|
/// <para>
|
|
/// or you want a horizontal stripe which is 3 pixels apart you would use a pattern like
|
|
/// 1
|
|
/// 0
|
|
/// 0
|
|
/// </para>
|
|
/// </remarks>
|
|
/// <typeparam name="TPixel">The pixel format.</typeparam>
|
|
public class PatternBrush<TPixel> : IBrush<TPixel>
|
|
where TPixel : struct, IPixel<TPixel>
|
|
{
|
|
/// <summary>
|
|
/// The pattern.
|
|
/// </summary>
|
|
private readonly DenseMatrix<TPixel> pattern;
|
|
private readonly DenseMatrix<Vector4> patternVector;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="PatternBrush{TPixel}"/> class.
|
|
/// </summary>
|
|
/// <param name="foreColor">Color of the fore.</param>
|
|
/// <param name="backColor">Color of the back.</param>
|
|
/// <param name="pattern">The pattern.</param>
|
|
public PatternBrush(TPixel foreColor, TPixel backColor, bool[,] pattern)
|
|
: this(foreColor, backColor, new DenseMatrix<bool>(pattern))
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="PatternBrush{TPixel}"/> class.
|
|
/// </summary>
|
|
/// <param name="foreColor">Color of the fore.</param>
|
|
/// <param name="backColor">Color of the back.</param>
|
|
/// <param name="pattern">The pattern.</param>
|
|
internal PatternBrush(TPixel foreColor, TPixel backColor, DenseMatrix<bool> pattern)
|
|
{
|
|
var foreColorVector = foreColor.ToVector4();
|
|
var backColorVector = backColor.ToVector4();
|
|
this.pattern = new DenseMatrix<TPixel>(pattern.Columns, pattern.Rows);
|
|
this.patternVector = new DenseMatrix<Vector4>(pattern.Columns, pattern.Rows);
|
|
for (int i = 0; i < pattern.Data.Length; i++)
|
|
{
|
|
if (pattern.Data[i])
|
|
{
|
|
this.pattern.Data[i] = foreColor;
|
|
this.patternVector.Data[i] = foreColorVector;
|
|
}
|
|
else
|
|
{
|
|
this.pattern.Data[i] = backColor;
|
|
this.patternVector.Data[i] = backColorVector;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="PatternBrush{TPixel}"/> class.
|
|
/// </summary>
|
|
/// <param name="brush">The brush.</param>
|
|
internal PatternBrush(PatternBrush<TPixel> brush)
|
|
{
|
|
this.pattern = brush.pattern;
|
|
this.patternVector = brush.patternVector;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public BrushApplicator<TPixel> CreateApplicator(ImageFrame<TPixel> source, RectangleF region, GraphicsOptions options)
|
|
{
|
|
return new PatternBrushApplicator(source, this.pattern, this.patternVector, options);
|
|
}
|
|
|
|
/// <summary>
|
|
/// The pattern brush applicator.
|
|
/// </summary>
|
|
private class PatternBrushApplicator : BrushApplicator<TPixel>
|
|
{
|
|
/// <summary>
|
|
/// The pattern.
|
|
/// </summary>
|
|
private readonly DenseMatrix<TPixel> pattern;
|
|
private readonly DenseMatrix<Vector4> patternVector;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="PatternBrushApplicator" /> class.
|
|
/// </summary>
|
|
/// <param name="source">The source image.</param>
|
|
/// <param name="pattern">The pattern.</param>
|
|
/// <param name="patternVector">The patternVector.</param>
|
|
/// <param name="options">The options</param>
|
|
public PatternBrushApplicator(ImageFrame<TPixel> source, DenseMatrix<TPixel> pattern, DenseMatrix<Vector4> patternVector, GraphicsOptions options)
|
|
: base(source, options)
|
|
{
|
|
this.pattern = pattern;
|
|
this.patternVector = patternVector;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the color for a single pixel.
|
|
/// </summary>#
|
|
/// <param name="x">The x.</param>
|
|
/// <param name="y">The y.</param>
|
|
/// <returns>
|
|
/// The Color.
|
|
/// </returns>
|
|
internal override TPixel this[int x, int y]
|
|
{
|
|
get
|
|
{
|
|
x = x % this.pattern.Columns;
|
|
y = y % this.pattern.Rows;
|
|
|
|
// 2d array index at row/column
|
|
return this.pattern[y, x];
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override void Dispose()
|
|
{
|
|
// noop
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
internal override void Apply(Span<float> scanline, int x, int y)
|
|
{
|
|
int patternY = y % this.pattern.Rows;
|
|
MemoryAllocator memoryAllocator = this.Target.MemoryAllocator;
|
|
|
|
using (IMemoryOwner<float> amountBuffer = memoryAllocator.Allocate<float>(scanline.Length))
|
|
using (IMemoryOwner<TPixel> overlay = memoryAllocator.Allocate<TPixel>(scanline.Length))
|
|
{
|
|
Span<float> amountSpan = amountBuffer.GetSpan();
|
|
Span<TPixel> overlaySpan = overlay.GetSpan();
|
|
|
|
for (int i = 0; i < scanline.Length; i++)
|
|
{
|
|
amountSpan[i] = (scanline[i] * this.Options.BlendPercentage).Clamp(0, 1);
|
|
|
|
int patternX = (x + i) % this.pattern.Columns;
|
|
overlaySpan[i] = this.pattern[patternY, patternX];
|
|
}
|
|
|
|
Span<TPixel> destinationRow = this.Target.GetPixelRowSpan(y).Slice(x, scanline.Length);
|
|
this.Blender.Blend(memoryAllocator, destinationRow, destinationRow, overlaySpan, amountSpan);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|