arsd/pixmappaint.d

627 lines
14 KiB
D
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/+
== pixmappaint ==
Copyright Elias Batek (0xEAB) 2024.
Distributed under the Boost Software License, Version 1.0.
+/
module arsd.pixmappaint;
import arsd.color;
import arsd.core;
import std.math : round;
alias Color = arsd.color.Color;
///
alias ColorF = arsd.color.ColorF;
///
alias Pixel = Color;
///
alias Point = arsd.color.Point;
///
alias Rectangle = arsd.color.Rectangle;
///
alias Size = arsd.color.Size;
// verify assumption(s)
static assert(Pixel.sizeof == uint.sizeof);
@safe pure nothrow @nogc {
///
Pixel rgba(ubyte r, ubyte g, ubyte b, ubyte a = 0xFF) {
return Pixel(r, g, b, a);
}
///
Pixel rgba(ubyte r, ubyte g, ubyte b, float aPct)
in (aPct >= 0 && aPct <= 1) {
return Pixel(r, g, b, castTo!ubyte(aPct * 255));
}
///
Pixel rgb(ubyte r, ubyte g, ubyte b) {
return rgba(r, g, b, 0xFF);
}
}
/++
Pixel data container
+/
struct Pixmap {
/// Pixel data
Pixel[] data;
/// Pixel per row
int width;
@safe pure nothrow:
///
this(Size size) {
this.size = size;
}
///
this(int width, int height)
in (width > 0)
in (height > 0) {
this(Size(width, height));
}
///
this(Pixel[] data, int width) @nogc
in (data.length % width == 0) {
this.data = data;
this.width = width;
}
/++
Creates a $(I deep clone) of the Pixmap
+/
Pixmap clone() const {
auto c = Pixmap();
c.width = this.width;
c.data = this.data.dup;
return c;
}
// undocumented: really shouldnt be used.
// carries the risks of `length` and `width` getting out of sync accidentally.
deprecated("Use `size` instead.")
void length(int value) {
data.length = value;
}
/++
Changes the size of the buffer
Reallocates the underlying pixel array.
+/
void size(Size value) {
data.length = value.area;
width = value.width;
}
/// ditto
void size(int totalPixels, int width)
in (totalPixels % width == 0) {
data.length = totalPixels;
this.width = width;
}
static {
/++
Creates a Pixmap wrapping the pixel data from the provided `TrueColorImage`.
Interoperability function: `arsd.color`
+/
Pixmap fromTrueColorImage(TrueColorImage source) @nogc {
return Pixmap(source.imageData.colors, source.width);
}
/++
Creates a Pixmap wrapping the pixel data from the provided `MemoryImage`.
Interoperability function: `arsd.color`
+/
Pixmap fromMemoryImage(MemoryImage source) {
return fromTrueColorImage(source.getAsTrueColorImage());
}
}
@safe pure nothrow @nogc:
/// Height of the buffer, i.e. the number of lines
int height() inout {
if (width == 0) {
return 0;
}
return castTo!int(data.length / width);
}
/// Rectangular size of the buffer
Size size() inout {
return Size(width, height);
}
/// Length of the buffer, i.e. the number of pixels
int length() inout {
return castTo!int(data.length);
}
/++
Number of bytes per line
Returns:
width × Pixel.sizeof
+/
int pitch() inout {
return (width * int(Pixel.sizeof));
}
/++
Retrieves a linear slice of the pixmap.
Returns:
`n` pixels starting at the top-left position `pos`.
+/
inout(Pixel)[] sliceAt(Point pos, int n) inout {
immutable size_t offset = linearOffset(width, pos);
immutable size_t end = (offset + n);
return data[offset .. end];
}
/// Clears the buffers contents (by setting each pixel to the same color)
void clear(Pixel value) {
data[] = value;
}
}
///
struct SpriteSheet {
private {
Pixmap _pixmap;
Size _spriteDimensions;
Size _layout; // pre-computed upon construction
}
@safe pure nothrow @nogc:
///
public this(Pixmap pixmap, Size spriteSize) {
_pixmap = pixmap;
_spriteDimensions = spriteSize;
_layout = Size(
_pixmap.width / _spriteDimensions.width,
_pixmap.height / _spriteDimensions.height,
);
}
///
inout(Pixmap) pixmap() inout {
return _pixmap;
}
///
Size spriteSize() inout {
return _spriteDimensions;
}
///
Size layout() inout {
return _layout;
}
///
Point getSpriteColumn(int index) inout {
immutable x = index % layout.width;
immutable y = (index - x) / layout.height;
return Point(x, y);
}
///
Point getSpritePixelOffset2D(int index) inout {
immutable col = this.getSpriteColumn(index);
return Point(
col.x * _spriteDimensions.width,
col.y * _spriteDimensions.height,
);
}
}
// Silly micro-optimization
private struct OriginRectangle {
Size size;
@safe pure nothrow @nogc:
int left() const => 0;
int top() const => 0;
int right() const => size.width;
int bottom() const => size.height;
bool intersect(const Rectangle b) const {
// dfmt off
return (
(b.right > 0 ) &&
(b.left < this.right ) &&
(b.bottom > 0 ) &&
(b.top < this.bottom)
);
// dfmt on
}
}
@safe pure nothrow @nogc:
// misc
private {
Point pos(Rectangle r) => r.upperLeft;
}
/++
Fast 8-bit percentage function
This function optimizes its runtime performance by substituting
the division by 255 with an approximation using bitshifts.
Nonetheless, the its result are as accurate as a floating point
division with 64-bit precision.
Params:
nPercentage = percentage as the number of 255ths (two hundred fifty-fifths)
value = base value (total)
Returns:
`round(value * nPercentage / 255.0)`
+/
ubyte n255thsOf(const ubyte nPercentage, const ubyte value) {
immutable factor = (nPercentage | (nPercentage << 8));
return (((value * factor) + 0x8080) >> 16);
}
@safe unittest {
// Accuracy verification
static ubyte n255thsOfFP64(const ubyte nPercentage, const ubyte value) {
return (value * nPercentage / 255.0).round().castTo!ubyte();
}
for (int value = ubyte.min; value <= ubyte.max; ++value) {
for (int percent = ubyte.min; percent <= ubyte.max; ++percent) {
immutable v = cast(ubyte) value;
immutable p = cast(ubyte) percent;
immutable approximated = n255thsOf(p, v);
immutable precise = n255thsOfFP64(p, v);
assert(approximated == precise);
}
}
}
/++
Sets the opacity of a [Pixmap].
This lossy operation updates the alpha-channel value of each pixel.
`alpha *= opacity`
See_Also:
Use [opacityF] with opacity values in percent (%).
+/
void opacity(ref Pixmap pixmap, const ubyte opacity) {
foreach (ref px; pixmap.data) {
px.a = opacity.n255thsOf(px.a);
}
}
/++
Sets the opacity of a [Pixmap].
This lossy operation updates the alpha-channel value of each pixel.
`alpha *= opacity`
See_Also:
Use [opacity] with 8-bit integer opacity values (in 255ths).
+/
void opacityF(ref Pixmap pixmap, const float opacity)
in (opacity >= 0)
in (opacity <= 1.0) {
immutable opacity255 = round(opacity * 255).castTo!ubyte;
pixmap.opacity = opacity255;
}
// ==== Alpha-blending functions ====
///
public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted
in (source.length == target.length) {
foreach (immutable idx, ref pxTarget; target) {
alphaBlend(pxTarget, source.ptr[idx]);
}
}
///
public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted {
pragma(inline, true);
immutable alphaSource = (pxSource.a | (pxSource.a << 8));
immutable alphaTarget = (0xFFFF - alphaSource);
foreach (immutable ib, ref px; pxTarget.components) {
immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16);
immutable s = cast(ubyte)(((pxSource.components.ptr[ib] * alphaSource) + 0x8080) >> 16);
px = cast(ubyte)(d + s);
}
}
///
public void alphaBlend(
ubyte function(const ubyte, const ubyte) blend
)(ref Pixel pxTarget, const Pixel pxSource) @trusted {
pragma(inline, true);
immutable alphaSource = (pxSource.a | (pxSource.a << 8));
immutable alphaTarget = (0xFFFF - alphaSource);
foreach (immutable ib, ref px; pxTarget.components) {
immutable b = blend(px, pxSource.components.ptr[ib]);
immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16);
immutable s = cast(ubyte)(((b * alphaSource) + 0x8080) >> 16);
px = cast(ubyte)(d + s);
}
}
// ==== Blending functions ====
enum BlendMode {
none = 0,
replace = none,
normal = 1,
alpha = normal,
multiply,
screen,
darken,
lighten,
}
///
alias Blend = BlendMode;
// undocumented
enum blendNormal = BlendMode.normal;
/++
Blends pixel `source` into pixel `target`.
+/
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == BlendMode.replace) {
target = source;
}
/// ditto
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.alpha) {
return alphaBlend(target, source);
}
/// ditto
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.multiply) {
return alphaBlend!((a, b) => n255thsOf(a, b))(target, source);
}
/// ditto
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.screen) {
assert(false, "TODO");
}
/// ditto
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.darken) {
assert(false, "TODO");
}
/// ditto
void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.lighten) {
assert(false, "TODO");
}
/++
Blends the pixel data of `source` into `target`.
`source` and `target` MUST have the same length.
+/
void blendPixels(BlendMode mode)(scope Pixel[] target, scope const Pixel[] source) @trusted
in (source.length == target.length) {
static if (mode == BlendMode.replace) {
target.ptr[0 .. target.length] = source.ptr[0 .. target.length];
} else {
// better error message in case its not implemented
static if (!is(typeof(blendPixel!mode))) {
pragma(msg, "Hint: Missing or bad `blendPixel!(" ~ mode.stringof ~ ")`.");
}
foreach (immutable idx, ref pxTarget; target) {
blendPixel!mode(pxTarget, source.ptr[idx]);
}
}
}
/// ditto
void blendPixels(scope Pixel[] target, scope const Pixel[] source, BlendMode mode) {
import std.meta : NoDuplicates;
import std.traits : EnumMembers;
final switch (mode) with (BlendMode) {
static foreach (m; NoDuplicates!(EnumMembers!BlendMode)) {
case m:
return blendPixels!m(target, source);
}
}
}
// ==== Drawing functions ====
/++
Draws a single pixel
+/
void drawPixel(Pixmap target, Point pos, Pixel color) {
immutable size_t offset = linearOffset(target.width, pos);
target.data[offset] = color;
}
/++
Draws a rectangle
+/
void drawRectangle(Pixmap target, Rectangle rectangle, Pixel color) {
alias r = rectangle;
immutable tRect = OriginRectangle(
Size(target.width, target.height),
);
// out of bounds?
if (!tRect.intersect(r)) {
return;
}
immutable drawingTarget = Point(
(r.pos.x >= 0) ? r.pos.x : 0,
(r.pos.y >= 0) ? r.pos.y : 0,
);
immutable drawingEnd = Point(
(r.right < tRect.right) ? r.right : tRect.right,
(r.bottom < tRect.bottom) ? r.bottom : tRect.bottom,
);
immutable int drawingWidth = drawingEnd.x - drawingTarget.x;
foreach (y; drawingTarget.y .. drawingEnd.y) {
target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = color;
}
}
/++
Draws a line
+/
void drawLine(Pixmap target, Point a, Point b, Pixel color) {
import std.math : sqrt;
// TODO: line width
// TODO: anti-aliasing (looks awful without it!)
float deltaX = b.x - a.x;
float deltaY = b.y - a.y;
int steps = sqrt(deltaX * deltaX + deltaY * deltaY).castTo!int;
float[2] step = [
(deltaX / steps),
(deltaY / steps),
];
foreach (i; 0 .. steps) {
// dfmt off
immutable Point p = a + Point(
round(step[0] * i).castTo!int,
round(step[1] * i).castTo!int,
);
// dfmt on
immutable offset = linearOffset(p, target.width);
target.data[offset] = color;
}
immutable offsetEnd = linearOffset(b, target.width);
target.data[offsetEnd] = color;
}
/++
Draws an image (a source pixmap) on a target pixmap
Params:
target = target pixmap to draw on
image = source pixmap
pos = top-left destination position (on the target pixmap)
+/
void drawPixmap(Pixmap target, Pixmap image, Point pos, Blend blend = blendNormal) {
alias source = image;
immutable tRect = OriginRectangle(
Size(target.width, target.height),
);
immutable sRect = Rectangle(pos, source.size);
// out of bounds?
if (!tRect.intersect(sRect)) {
return;
}
immutable drawingTarget = Point(
(pos.x >= 0) ? pos.x : 0,
(pos.y >= 0) ? pos.y : 0,
);
immutable drawingEnd = Point(
(sRect.right < tRect.right) ? sRect.right : tRect.right,
(sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom,
);
immutable drawingSource = Point(drawingTarget.x, 0) - Point(sRect.pos.x, sRect.pos.y);
immutable int drawingWidth = drawingEnd.x - drawingTarget.x;
foreach (y; drawingTarget.y .. drawingEnd.y) {
blendPixels(
target.sliceAt(Point(drawingTarget.x, y), drawingWidth),
source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth),
blend,
);
}
}
/++
Draws a sprite from a spritesheet
+/
void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos, Blend blend = blendNormal) {
immutable tRect = OriginRectangle(
Size(target.width, target.height),
);
immutable spriteOffset = sheet.getSpritePixelOffset2D(spriteIndex);
immutable sRect = Rectangle(pos, sheet.spriteSize);
// out of bounds?
if (!tRect.intersect(sRect)) {
return;
}
immutable drawingTarget = Point(
(pos.x >= 0) ? pos.x : 0,
(pos.y >= 0) ? pos.y : 0,
);
immutable drawingEnd = Point(
(sRect.right < tRect.right) ? sRect.right : tRect.right,
(sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom,
);
immutable drawingSource =
spriteOffset
+ Point(drawingTarget.x, 0)
- Point(sRect.pos.x, sRect.pos.y);
immutable int drawingWidth = drawingEnd.x - drawingTarget.x;
foreach (y; drawingTarget.y .. drawingEnd.y) {
blendPixels(
target.sliceAt(Point(drawingTarget.x, y), drawingWidth),
sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth),
blend,
);
}
}