pixie/src/pixie/paths.nim
2022-06-29 17:17:53 -05:00

2563 lines
77 KiB
Nim

import blends, bumpy, chroma, common, images, internal, masks, paints, std/fenv,
std/strutils, vmath
when defined(amd64) and allowSimd:
import nimsimd/sse2
type
WindingRule* = enum
## Winding rules.
NonZero
EvenOdd
LineCap* = enum
## Line cap type for strokes.
ButtCap, RoundCap, SquareCap
LineJoin* = enum
## Line join type for strokes.
MiterJoin, RoundJoin, BevelJoin
PathCommandKind = enum
## Type of path commands
Close,
Move, Line, HLine, VLine, Cubic, SCubic, Quad, TQuad, Arc,
RMove, RLine, RHLine, RVLine, RCubic, RSCubic, RQuad, RTQuad, RArc
PathCommand = object
## Binary version of an SVG command.
kind: PathCommandKind
numbers: seq[float32]
Path* = ref object
## Used to hold paths and create paths.
commands: seq[PathCommand]
start, at: Vec2 # Maintained by moveTo, lineTo, etc. Used by arcTo.
SomePath* = Path | string
PartitionEntry = object
segment: Segment
m, b: float32
winding: int16
Partition = object
entries: seq[PartitionEntry]
requiresAntiAliasing, twoNonintersectingSpanningSegments: bool
top, bottom: int
Fixed32 = int32 ## 24.8 fixed point
const
epsilon: float32 = 0.0001 * PI ## Tiny value used for some computations.
pixelErrorMargin: float32 = 0.2
defaultMiterLimit*: float32 = 4
when defined(release):
{.push checks: off.}
proc newPath*(): Path {.raises: [].} =
## Create a new Path.
Path()
proc pixelScale(transform: Mat3): float32 =
## What is the largest scale factor of this transform?
max(
vec2(transform[0, 0], transform[0, 1]).length,
vec2(transform[1, 0], transform[1, 1]).length
)
proc isRelative(kind: PathCommandKind): bool {.inline.} =
kind in {
RMove, RLine, TQuad, RTQuad, RHLine, RVLine, RCubic, RSCubic, RQuad, RArc
}
proc parameterCount(kind: PathCommandKind): int =
## Returns number of parameters a path command has.
case kind:
of Close: 0
of Move, Line, RMove, RLine, TQuad, RTQuad: 2
of HLine, VLine, RHLine, RVLine: 1
of Cubic, RCubic: 6
of SCubic, RSCubic, Quad, RQuad: 4
of Arc, RArc: 7
proc `$`*(path: Path): string {.raises: [].} =
## Turn path int into a string.
for i, command in path.commands:
case command.kind
of Move: result.add "M"
of Line: result.add "L"
of HLine: result.add "H"
of VLine: result.add "V"
of Cubic: result.add "C"
of SCubic: result.add "S"
of Quad: result.add "Q"
of TQuad: result.add "T"
of Arc: result.add "A"
of RMove: result.add "m"
of RLine: result.add "l"
of RHLine: result.add "h"
of RVLine: result.add "v"
of RCubic: result.add "c"
of RSCubic: result.add "s"
of RQuad: result.add "q"
of RTQuad: result.add "t"
of RArc: result.add "a"
of Close: result.add "Z"
for j, number in command.numbers:
if floor(number) == number:
result.add $number.int
else:
result.add $number
if i != path.commands.len - 1 or j != command.numbers.len - 1:
result.add " "
proc parsePath*(path: string): Path {.raises: [PixieError].} =
## Converts a SVG style path string into seq of commands.
result = newPath()
if path.len == 0:
return
var
p, numberStart: int
armed, hitDecimal: bool
kind: PathCommandKind
numbers: seq[float32]
proc finishNumber() =
if numberStart > 0:
try:
numbers.add(parseFloat(path[numberStart ..< p]))
except ValueError:
raise newException(PixieError, "Invalid path, parsing parameter failed")
numberStart = 0
hitDecimal = false
proc finishCommand(result: Path) =
finishNumber()
if armed: # The first finishCommand() arms
let paramCount = parameterCount(kind)
if paramCount == 0:
if numbers.len != 0:
raise newException(PixieError, "Invalid path, unexpected parameters")
result.commands.add(PathCommand(kind: kind))
else:
if numbers.len mod paramCount != 0:
raise newException(
PixieError,
"Invalid path, wrong number of parameters"
)
for batch in 0 ..< numbers.len div paramCount:
if batch > 0:
if kind == Move:
kind = Line
elif kind == RMove:
kind = RLine
result.commands.add(PathCommand(
kind: kind,
numbers: numbers[batch * paramCount ..< (batch + 1) * paramCount]
))
numbers.setLen(0)
armed = true
template expectsArcFlag(): bool =
kind in {Arc, RArc} and numbers.len mod 7 in {3, 4}
while p < path.len:
case path[p]:
# Relative
of 'm':
finishCommand(result)
kind = RMove
of 'l':
finishCommand(result)
kind = RLine
of 'h':
finishCommand(result)
kind = RHLine
of 'v':
finishCommand(result)
kind = RVLine
of 'c':
finishCommand(result)
kind = RCubic
of 's':
finishCommand(result)
kind = RSCubic
of 'q':
finishCommand(result)
kind = RQuad
of 't':
finishCommand(result)
kind = RTQuad
of 'a':
finishCommand(result)
kind = RArc
of 'z':
finishCommand(result)
kind = Close
# Absolute
of 'M':
finishCommand(result)
kind = Move
of 'L':
finishCommand(result)
kind = Line
of 'H':
finishCommand(result)
kind = HLine
of 'V':
finishCommand(result)
kind = VLine
of 'C':
finishCommand(result)
kind = Cubic
of 'S':
finishCommand(result)
kind = SCubic
of 'Q':
finishCommand(result)
kind = Quad
of 'T':
finishCommand(result)
kind = TQuad
of 'A':
finishCommand(result)
kind = Arc
of 'Z':
finishCommand(result)
kind = Close
of '-', '+':
if numberStart > 0 and path[p - 1] in {'e', 'E'}:
discard
else:
finishNumber()
numberStart = p
of '.':
if hitDecimal or expectsArcFlag():
finishNumber()
hitDecimal = true
if numberStart == 0:
numberStart = p
of ' ', ',', '\r', '\n', '\t':
finishNumber()
else:
if numberStart > 0 and expectsArcFlag():
finishNumber()
if p - 1 == numberStart and path[p - 1] == '0':
# If the number starts with 0 and we've hit another digit, finish the 0
# .. 01.3.. -> [..0, 1.3..]
finishNumber()
if numberStart == 0:
numberStart = p
inc p
finishCommand(result)
proc transform*(path: Path, mat: Mat3) {.raises: [].} =
## Apply a matrix transform to a path.
if mat == mat3():
return
if path.commands.len > 0 and path.commands[0].kind == RMove:
path.commands[0].kind = Move
for command in path.commands.mitems:
var mat = mat
if command.kind.isRelative():
mat.pos = vec2(0)
case command.kind:
of Close:
discard
of Move, Line, RMove, RLine, TQuad, RTQuad:
var pos = vec2(command.numbers[0], command.numbers[1])
pos = mat * pos
command.numbers[0] = pos.x
command.numbers[1] = pos.y
of HLine, RHLine:
var pos = vec2(command.numbers[0], 0)
pos = mat * pos
command.numbers[0] = pos.x
of VLine, RVLine:
var pos = vec2(0, command.numbers[0])
pos = mat * pos
command.numbers[0] = pos.y
of Cubic, RCubic:
var
ctrl1 = vec2(command.numbers[0], command.numbers[1])
ctrl2 = vec2(command.numbers[2], command.numbers[3])
to = vec2(command.numbers[4], command.numbers[5])
ctrl1 = mat * ctrl1
ctrl2 = mat * ctrl2
to = mat * to
command.numbers[0] = ctrl1.x
command.numbers[1] = ctrl1.y
command.numbers[2] = ctrl2.x
command.numbers[3] = ctrl2.y
command.numbers[4] = to.x
command.numbers[5] = to.y
of SCubic, RSCubic, Quad, RQuad:
var
ctrl = vec2(command.numbers[0], command.numbers[1])
to = vec2(command.numbers[2], command.numbers[3])
ctrl = mat * ctrl
to = mat * to
command.numbers[0] = ctrl.x
command.numbers[1] = ctrl.y
command.numbers[2] = to.x
command.numbers[3] = to.y
of Arc, RArc:
var
radii = vec2(command.numbers[0], command.numbers[1])
to = vec2(command.numbers[5], command.numbers[6])
# Extract the scale from the matrix and only apply that to the radii
radii = scale(vec2(mat[0, 0], mat[1, 1])) * radii
to = mat * to
command.numbers[0] = radii.x
command.numbers[1] = radii.y
command.numbers[5] = to.x
command.numbers[6] = to.y
proc addPath*(path: Path, other: Path) {.raises: [].} =
## Adds a path to the current path.
path.commands.add(other.commands)
proc closePath*(path: Path) {.raises: [].} =
## Attempts to add a straight line from the current point to the start of
## the current sub-path. If the shape has already been closed or has only
## one point, this function does nothing.
path.commands.add(PathCommand(kind: Close))
path.at = path.start
proc moveTo*(path: Path, x, y: float32) {.raises: [].} =
## Begins a new sub-path at the point (x, y).
path.commands.add(PathCommand(kind: Move, numbers: @[x, y]))
path.start = vec2(x, y)
path.at = path.start
proc moveTo*(path: Path, v: Vec2) {.inline, raises: [].} =
## Begins a new sub-path at the point (x, y).
path.moveTo(v.x, v.y)
proc lineTo*(path: Path, x, y: float32) {.raises: [].} =
## Adds a straight line to the current sub-path by connecting the sub-path's
## last point to the specified (x, y) coordinates.
path.commands.add(PathCommand(kind: Line, numbers: @[x, y]))
path.at = vec2(x, y)
proc lineTo*(path: Path, v: Vec2) {.inline, raises: [].} =
## Adds a straight line to the current sub-path by connecting the sub-path's
## last point to the specified (x, y) coordinates.
path.lineTo(v.x, v.y)
proc bezierCurveTo*(path: Path, x1, y1, x2, y2, x3, y3: float32) {.raises: [].} =
## Adds a cubic Bézier curve to the current sub-path. It requires three
## points: the first two are control points and the third one is the end
## point. The starting point is the latest point in the current path,
## which can be changed using moveTo() before creating the Bézier curve.
path.commands.add(PathCommand(
kind: Cubic,
numbers: @[x1, y1, x2, y2, x3, y3]
))
path.at = vec2(x3, y3)
proc bezierCurveTo*(path: Path, ctrl1, ctrl2, to: Vec2) {.inline, raises: [].} =
## Adds a cubic Bézier curve to the current sub-path. It requires three
## points: the first two are control points and the third one is the end
## point. The starting point is the latest point in the current path,
## which can be changed using moveTo() before creating the Bézier curve.
path.bezierCurveTo(ctrl1.x, ctrl1.y, ctrl2.x, ctrl2.y, to.x, to.y)
proc quadraticCurveTo*(path: Path, x1, y1, x2, y2: float32) {.raises: [].} =
## Adds a quadratic Bézier curve to the current sub-path. It requires two
## points: the first one is a control point and the second one is the end
## point. The starting point is the latest point in the current path,
## which can be changed using moveTo() before creating the quadratic
## Bézier curve.
path.commands.add(PathCommand(
kind: Quad,
numbers: @[x1, y1, x2, y2]
))
path.at = vec2(x2, y2)
proc quadraticCurveTo*(path: Path, ctrl, to: Vec2) {.inline, raises: [].} =
## Adds a quadratic Bézier curve to the current sub-path. It requires two
## points: the first one is a control point and the second one is the end
## point. The starting point is the latest point in the current path,
## which can be changed using moveTo() before creating the quadratic
## Bézier curve.
path.quadraticCurveTo(ctrl.x, ctrl.y, to.x, to.y)
proc ellipticalArcTo*(
path: Path,
rx, ry: float32,
xAxisRotation: float32,
largeArcFlag, sweepFlag: bool,
x, y: float32
) {.raises: [].} =
## Adds an elliptical arc to the current sub-path, using the given radius
## ratios, sweep flags, and end position.
path.commands.add(PathCommand(
kind: Arc,
numbers: @[
rx, ry, xAxisRotation, largeArcFlag.float32, sweepFlag.float32, x, y
]
))
path.at = vec2(x, y)
proc arc*(
path: Path, x, y, r, a0, a1: float32, ccw: bool = false
) {.raises: [PixieError].} =
## Adds a circular arc to the current sub-path.
if r == 0: # When radius is zero, do nothing.
return
if r < 0: # When radius is negative, error.
raise newException(PixieError, "Invalid arc, negative radius: " & $r)
let
dx = r * cos(a0)
dy = r * sin(a0)
x0 = x + dx
y0 = y + dy
cw = not ccw
if path.commands.len == 0: # Is this path empty? Move to (x0, y0).
path.moveTo(x0, y0)
elif abs(path.at.x - x0) > epsilon or abs(path.at.y - y0) > epsilon:
path.lineTo(x0, y0)
var angle =
if ccw: a0 - a1
else: a1 - a0
if angle < 0:
# When the angle goes the wrong way, flip the direction.
angle = angle mod TAU + TAU
if angle > TAU - epsilon:
# Angle describes a complete circle. Draw it in two arcs.
path.ellipticalArcTo(r, r, 0, true, cw, x - dx, y - dy)
path.at.x = x0
path.at.y = y0
path.ellipticalArcTo(r, r, 0, true, cw, path.at.x, path.at.y)
elif angle > epsilon:
path.at.x = x + r * cos(a1)
path.at.y = y + r * sin(a1)
path.ellipticalArcTo(r, r, 0, angle >= PI, cw, path.at.x, path.at.y)
proc arc*(
path: Path, pos: Vec2, r: float32, a: Vec2, ccw: bool = false
) {.inline, raises: [PixieError].} =
## Adds a circular arc to the current sub-path.
path.arc(pos.x, pos.y, r, a.x, a.y, ccw)
proc arcTo*(path: Path, x1, y1, x2, y2, r: float32) {.raises: [PixieError].} =
## Adds a circular arc using the given control points and radius.
## Commonly used for making rounded corners.
if r < 0: # When radius is negative, error.
raise newException(PixieError, "Invalid arc, negative radius: " & $r)
let
x0 = path.at.x
y0 = path.at.y
x21 = x2 - x1
y21 = y2 - y1
x01 = x0 - x1
y01 = y0 - y1
l01_2 = x01 * x01 + y01 * y01
if path.commands.len == 0: # Is this path empty? Move to (x0, y0).
path.moveTo(x0, y0)
elif not(l01_2 > epsilon): # Is (x1, y1) coincident with (x0, y0)? Do nothing.
discard
elif not(abs(y01 * x21 - y21 * x01) > epsilon) or r == 0: # Just a line?
path.lineTo(x1, y1)
else:
let
x20 = x2 - x0
y20 = y2 - y0
l21_2 = x21 * x21 + y21 * y21
l20_2 = x20 * x20 + y20 * y20
l21 = sqrt(l21_2)
l01 = sqrt(l01_2)
l = r * tan((PI - arccos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2)
t01 = l / l01
t21 = l / l21
# If the start tangent is not coincident with (x0, y0), line to.
if abs(t01 - 1) > epsilon:
path.lineTo(x1 + t01 * x01, y1 + t01 * y01)
path.at.x = x1 + t21 * x21
path.at.y = y1 + t21 * y21
path.ellipticalArcTo(r, r, 0, false, y01 * x20 > x01 * y20, path.at.x, path.at.y)
proc arcTo*(path: Path, a, b: Vec2, r: float32) {.inline, raises: [PixieError].} =
## Adds a circular arc using the given control points and radius.
path.arcTo(a.x, a.y, b.x, b.y, r)
proc rect*(path: Path, x, y, w, h: float32, clockwise = true) {.raises: [].} =
## Adds a rectangle.
## Clockwise param can be used to subtract a rect from a path when using
## even-odd winding rule.
if clockwise:
path.moveTo(x, y)
path.lineTo(x + w, y)
path.lineTo(x + w, y + h)
path.lineTo(x, y + h)
path.closePath()
else:
path.moveTo(x, y)
path.lineTo(x, y + h)
path.lineTo(x + w, y + h)
path.lineTo(x + w, y)
path.closePath()
proc rect*(path: Path, rect: Rect, clockwise = true) {.inline, raises: [].} =
## Adds a rectangle.
## Clockwise param can be used to subtract a rect from a path when using
## even-odd winding rule.
path.rect(rect.x, rect.y, rect.w, rect.h, clockwise)
const splineCircleK = 4.0 * (-1.0 + sqrt(2.0)) / 3
## Reference for magic constant:
## https://dl3.pushbulletusercontent.com/a3fLVC8boTzRoxevD1OgCzRzERB9z2EZ/unknown.png
proc roundedRect*(
path: Path, x, y, w, h, nw, ne, se, sw: float32, clockwise = true
) {.raises: [].} =
## Adds a rounded rectangle.
## Clockwise param can be used to subtract a rect from a path when using
## even-odd winding rule.
var
nw = nw
ne = ne
se = se
sw = sw
maxRadius = min(w / 2, h / 2)
nw = max(0, min(nw, maxRadius))
ne = max(0, min(ne, maxRadius))
se = max(0, min(se, maxRadius))
sw = max(0, min(sw, maxRadius))
if nw == 0 and ne == 0 and se == 0 and sw == 0:
path.rect(x, y, w, h, clockwise)
return
let
s = splineCircleK
t1 = vec2(x + nw, y)
t2 = vec2(x + w - ne, y)
r1 = vec2(x + w, y + ne)
r2 = vec2(x + w, y + h - se)
b1 = vec2(x + w - se, y + h)
b2 = vec2(x + sw, y + h)
l1 = vec2(x, y + h - sw)
l2 = vec2(x, y + nw)
t1h = t1 + vec2(-nw * s, 0)
t2h = t2 + vec2(+ne * s, 0)
r1h = r1 + vec2(0, -ne * s)
r2h = r2 + vec2(0, +se * s)
b1h = b1 + vec2(+se * s, 0)
b2h = b2 + vec2(-sw * s, 0)
l1h = l1 + vec2(0, +sw * s)
l2h = l2 + vec2(0, -nw * s)
if clockwise:
path.moveTo(t1)
path.lineTo(t2)
path.bezierCurveTo(t2h, r1h, r1)
path.lineTo(r2)
path.bezierCurveTo(r2h, b1h, b1)
path.lineTo(b2)
path.bezierCurveTo(b2h, l1h, l1)
path.lineTo(l2)
path.bezierCurveTo(l2h, t1h, t1)
else:
path.moveTo(t1)
path.bezierCurveTo(t1h, l2h, l2)
path.lineTo(l1)
path.bezierCurveTo(l1h, b2h, b2)
path.lineTo(b1)
path.bezierCurveTo(b1h, r2h, r2)
path.lineTo(r1)
path.bezierCurveTo(r1h, t2h, t2)
path.lineTo(t1)
path.closePath()
proc roundedRect*(
path: Path, rect: Rect, nw, ne, se, sw: float32, clockwise = true
) {.inline, raises: [].} =
## Adds a rounded rectangle.
## Clockwise param can be used to subtract a rect from a path when using
## even-odd winding rule.
path.roundedRect(rect.x, rect.y, rect.w, rect.h, nw, ne, se, sw, clockwise)
proc ellipse*(path: Path, cx, cy, rx, ry: float32) {.raises: [].} =
## Adds a ellipse.
let
magicX = splineCircleK * rx
magicY = splineCircleK * ry
path.moveTo(cx + rx, cy)
path.bezierCurveTo(cx + rx, cy + magicY, cx + magicX, cy + ry, cx, cy + ry)
path.bezierCurveTo(cx - magicX, cy + ry, cx - rx, cy + magicY, cx - rx, cy)
path.bezierCurveTo(cx - rx, cy - magicY, cx - magicX, cy - ry, cx, cy - ry)
path.bezierCurveTo(cx + magicX, cy - ry, cx + rx, cy - magicY, cx + rx, cy)
path.closePath()
proc ellipse*(path: Path, center: Vec2, rx, ry: float32) {.inline, raises: [].} =
## Adds a ellipse.
path.ellipse(center.x, center.y, rx, ry)
proc circle*(path: Path, cx, cy, r: float32) {.inline, raises: [].} =
## Adds a circle.
path.ellipse(cx, cy, r, r)
proc circle*(path: Path, circle: Circle) {.inline, raises: [].} =
## Adds a circle.
path.ellipse(circle.pos.x, circle.pos.y, circle.radius, circle.radius)
proc polygon*(
path: Path, x, y, size: float32, sides: int
) {.raises: [PixieError].} =
## Adds an n-sided regular polygon at (x, y) with the parameter size.
## Polygons "face" north.
if sides <= 2:
raise newException(PixieError, "Invalid polygon sides value")
path.moveTo(x + size * sin(0.0), y - size * cos(0.0))
for side in 1 .. sides - 1:
path.lineTo(
x + size * sin(side.float32 * 2.0 * PI / sides.float32),
y - size * cos(side.float32 * 2.0 * PI / sides.float32)
)
path.closePath()
proc polygon*(
path: Path, pos: Vec2, size: float32, sides: int
) {.inline, raises: [PixieError].} =
## Adds a n-sided regular polygon at (x, y) with the parameter size.
path.polygon(pos.x, pos.y, size, sides)
proc commandsToShapes(
path: Path, closeSubpaths: bool, pixelScale: float32
): seq[Polygon] =
## Converts SVG-like commands to sequences of vectors.
var
start, at: Vec2
shape: Polygon
# Some commands use data from the previous command
var
prevCommandKind = Move
prevCtrl, prevCtrl2: Vec2
let errorMarginSq = pow(pixelErrorMargin / pixelScale, 2)
proc addSegment(shape: var Polygon, at, to: Vec2) =
# Don't add any 0 length lines
if at - to != vec2(0, 0):
# Don't double up points
if shape.len == 0 or shape[^1] != at:
shape.add(at)
shape.add(to)
proc addCubic(shape: var Polygon, at, ctrl1, ctrl2, to: Vec2) =
## Adds cubic segments to shape.
proc compute(at, ctrl1, ctrl2, to: Vec2, t: float32): Vec2 {.inline.} =
pow(1 - t, 3) * at +
pow(1 - t, 2) * 3 * t * ctrl1 +
(1 - t) * 3 * pow(t, 2) * ctrl2 +
pow(t, 3) * to
var
t: float32 # Where we are at on the curve from [0, 1]
step = 1.float32 # How far we want to try to move along the curve
prev = at
next = compute(at, ctrl1, ctrl2, to, t + step)
halfway = compute(at, ctrl1, ctrl2, to, t + step / 2)
while true:
if step <= epsilon(float32):
raise newException(PixieError, "Unable to discretize cubic")
let
midpoint = (prev + next) / 2
error = (midpoint - halfway).lengthSq
if error > errorMarginSq:
next = halfway
halfway = compute(at, ctrl1, ctrl2, to, t + step / 4)
step /= 2
else:
shape.addSegment(prev, next)
t += step
if t == 1:
break
prev = next
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
next = compute(at, ctrl1, ctrl2, to, t + step)
halfway = compute(at, ctrl1, ctrl2, to, t + step / 2)
proc addQuadratic(shape: var Polygon, at, ctrl, to: Vec2) =
## Adds quadratic segments to shape.
proc compute(at, ctrl, to: Vec2, t: float32): Vec2 {.inline.} =
pow(1 - t, 2) * at +
2 * (1 - t) * t * ctrl +
pow(t, 2) * to
var
t: float32 # Where we are at on the curve from [0, 1]
step = 1.float32 # How far we want to try to move along the curve
prev = at
next = compute(at, ctrl, to, t + step)
halfway = compute(at, ctrl, to, t + step / 2)
halfStepping = false
while true:
if step <= epsilon(float32):
raise newException(PixieError, "Unable to discretize quadratic")
let
midpoint = (prev + next) / 2
error = (midpoint - halfway).lengthSq
if error > errorMarginSq:
next = halfway
halfway = compute(at, ctrl, to, t + step / 4)
halfStepping = true
step /= 2
else:
shape.addSegment(prev, next)
t += step
if t == 1:
break
prev = next
if halfStepping:
step = min(step, 1 - t)
else:
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
next = compute(at, ctrl, to, t + step)
halfway = compute(at, ctrl, to, t + step / 2)
proc addArc(
shape: var Polygon,
at, radii: Vec2,
rotation: float32,
large, sweep: bool,
to: Vec2
) =
## Adds arc segments to shape.
type ArcParams = object
radii: Vec2
rotMat: Mat3
center: Vec2
theta, delta: float32
proc endpointToCenterArcParams(
at, radii: Vec2, rotation: float32, large, sweep: bool, to: Vec2
): ArcParams =
var
radii = vec2(abs(radii.x), abs(radii.y))
radiiSq = vec2(radii.x * radii.x, radii.y * radii.y)
let
radians: float32 = rotation / 180 * PI
d = vec2((at.x - to.x) / 2.0, (at.y - to.y) / 2.0)
p = vec2(
cos(radians) * d.x + sin(radians) * d.y,
-sin(radians) * d.x + cos(radians) * d.y
)
pSq = vec2(p.x * p.x, p.y * p.y)
let cr = pSq.x / radiiSq.x + pSq.y / radiiSq.y
if cr > 1:
radii *= sqrt(cr)
radiiSq = vec2(radii.x * radii.x, radii.y * radii.y)
let
dq = radiiSq.x * pSq.y + radiiSq.y * pSq.x
pq = (radiiSq.x * radiiSq.y - dq) / dq
var q = sqrt(max(0, pq))
if large == sweep:
q = -q
proc svgAngle(u, v: Vec2): float32 =
let
dot = dot(u, v)
len = length(u) * length(v)
result = arccos(clamp(dot / len, -1, 1))
if (u.x * v.y - u.y * v.x) < 0:
result = -result
let
cp = vec2(q * radii.x * p.y / radii.y, -q * radii.y * p.x / radii.x)
center = vec2(
cos(radians) * cp.x - sin(radians) * cp.y + (at.x + to.x) / 2,
sin(radians) * cp.x + cos(radians) * cp.y + (at.y + to.y) / 2
)
theta = svgAngle(vec2(1, 0), vec2((p.x-cp.x) / radii.x, (p.y - cp.y) / radii.y))
var delta = svgAngle(
vec2((p.x - cp.x) / radii.x, (p.y - cp.y) / radii.y),
vec2((-p.x - cp.x) / radii.x, (-p.y - cp.y) / radii.y)
)
delta = delta mod (PI * 2)
if sweep and delta < 0:
delta += 2 * PI
elif not sweep and delta > 0:
delta -= 2 * PI
# Normalize the delta
while delta > PI * 2:
delta -= PI * 2
while delta < -PI * 2:
delta += PI * 2
ArcParams(
radii: radii,
rotMat: rotate(-radians),
center: center,
theta: theta,
delta: delta
)
proc compute(arc: ArcParams, a: float32): Vec2 =
result = vec2(cos(a) * arc.radii.x, sin(a) * arc.radii.y)
result = arc.rotMat * result + arc.center
let arc = endpointToCenterArcParams(at, radii, rotation, large, sweep, to)
var
t: float32 # Where we are at on the curve from [0, 1]
step = 1.float32 # How far we want to try to move along the curve
prev = at
while t != 1:
if step <= epsilon(float32):
raise newException(PixieError, "Unable to discretize arc")
let
aPrev = arc.theta + arc.delta * t
a = arc.theta + arc.delta * (t + step)
next = arc.compute(a)
halfway = arc.compute(aPrev + (a - aPrev) / 2)
midpoint = (prev + next) / 2
error = (midpoint - halfway).lengthSq
if error > errorMarginSq:
let
quarterway = arc.compute(aPrev + (a - aPrev) / 4)
midpoint = (prev + halfway) / 2
halfwayError = (midpoint - quarterway).lengthSq
if halfwayError < errorMarginSq:
shape.addSegment(prev, halfway)
prev = halfway
t += step / 2
step = min(step / 2, 1 - t) # Assume next steps hould be the same size
else:
step = step / 4 # We know a half-step is too big
else:
shape.addSegment(prev, next)
prev = next
t += step
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
for command in path.commands:
if command.numbers.len != command.kind.parameterCount():
raise newException(PixieError, "Invalid path")
case command.kind:
of Move:
if shape.len > 0:
if closeSubpaths:
shape.addSegment(at, start)
result.add(shape)
shape = newSeq[Vec2]()
at.x = command.numbers[0]
at.y = command.numbers[1]
start = at
of Line:
let to = vec2(command.numbers[0], command.numbers[1])
shape.addSegment(at, to)
at = to
of HLine:
let to = vec2(command.numbers[0], at.y)
shape.addSegment(at, to)
at = to
of VLine:
let to = vec2(at.x, command.numbers[0])
shape.addSegment(at, to)
at = to
of Cubic:
let
ctrl1 = vec2(command.numbers[0], command.numbers[1])
ctrl2 = vec2(command.numbers[2], command.numbers[3])
to = vec2(command.numbers[4], command.numbers[5])
shape.addCubic(at, ctrl1, ctrl2, to)
at = to
prevCtrl2 = ctrl2
of SCubic:
let
ctrl2 = vec2(command.numbers[0], command.numbers[1])
to = vec2(command.numbers[2], command.numbers[3])
if prevCommandKind in {Cubic, SCubic, RCubic, RSCubic}:
let ctrl1 = at * 2 - prevCtrl2
shape.addCubic(at, ctrl1, ctrl2, to)
else:
shape.addCubic(at, at, ctrl2, to)
at = to
prevCtrl2 = ctrl2
of Quad:
let
ctrl = vec2(command.numbers[0], command.numbers[1])
to = vec2(command.numbers[2], command.numbers[3])
shape.addQuadratic(at, ctrl, to)
at = to
prevCtrl = ctrl
of TQuad:
let
to = vec2(command.numbers[0], command.numbers[1])
ctrl =
if prevCommandKind in {Quad, TQuad, RQuad, RTQuad}:
at * 2 - prevCtrl
else:
at
shape.addQuadratic(at, ctrl, to)
at = to
prevCtrl = ctrl
of Arc:
let
radii = vec2(command.numbers[0], command.numbers[1])
rotation = command.numbers[2]
large = command.numbers[3] == 1
sweep = command.numbers[4] == 1
to = vec2(command.numbers[5], command.numbers[6])
shape.addArc(at, radii, rotation, large, sweep, to)
at = to
of RMove:
if shape.len > 0:
result.add(shape)
shape = newSeq[Vec2]()
at.x += command.numbers[0]
at.y += command.numbers[1]
start = at
of RLine:
let to = vec2(at.x + command.numbers[0], at.y + command.numbers[1])
shape.addSegment(at, to)
at = to
of RHLine:
let to = vec2(at.x + command.numbers[0], at.y)
shape.addSegment(at, to)
at = to
of RVLine:
let to = vec2(at.x, at.y + command.numbers[0])
shape.addSegment(at, to)
at = to
of RCubic:
let
ctrl1 = vec2(at.x + command.numbers[0], at.y + command.numbers[1])
ctrl2 = vec2(at.x + command.numbers[2], at.y + command.numbers[3])
to = vec2(at.x + command.numbers[4], at.y + command.numbers[5])
shape.addCubic(at, ctrl1, ctrl2, to)
at = to
prevCtrl2 = ctrl2
of RSCubic:
let
ctrl2 = vec2(at.x + command.numbers[0], at.y + command.numbers[1])
to = vec2(at.x + command.numbers[2], at.y + command.numbers[3])
ctrl1 =
if prevCommandKind in {Cubic, SCubic, RCubic, RSCubic}:
at * 2 - prevCtrl2
else:
at
shape.addCubic(at, ctrl1, ctrl2, to)
at = to
prevCtrl2 = ctrl2
of RQuad:
let
ctrl = vec2(at.x + command.numbers[0], at.y + command.numbers[1])
to = vec2(at.x + command.numbers[2], at.y + command.numbers[3])
shape.addQuadratic(at, ctrl, to)
at = to
prevCtrl = ctrl
of RTQuad:
let
to = vec2(at.x + command.numbers[0], at.y + command.numbers[1])
ctrl =
if prevCommandKind in {Quad, TQuad, RQuad, RTQuad}:
at * 2 - prevCtrl
else:
at
shape.addQuadratic(at, ctrl, to)
at = to
prevCtrl = ctrl
of RArc:
let
radii = vec2(command.numbers[0], command.numbers[1])
rotation = command.numbers[2]
large = command.numbers[3] == 1
sweep = command.numbers[4] == 1
to = vec2(at.x + command.numbers[5], at.y + command.numbers[6])
shape.addArc(at, radii, rotation, large, sweep, to)
at = to
of Close:
if at != start:
shape.addSegment(at, start)
at = start
if shape.len > 0:
result.add(shape)
shape = newSeq[Vec2]()
prevCommandKind = command.kind
if shape.len > 0:
if closeSubpaths:
shape.addSegment(at, start)
result.add(shape)
proc shapesToSegments(shapes: seq[Polygon]): seq[(Segment, int16)] =
## Converts the shapes into a set of filtered segments with winding value.
for shape in shapes:
for segment in shape.segments:
if segment.at.y == segment.to.y: # Skip horizontal
continue
var
segment = segment
winding = 1.int16
if segment.at.y > segment.to.y:
swap(segment.at, segment.to)
winding = -1
# Quantize the segment to prevent leaks
segment = segment(
vec2(segment.at.x.quantize(1 / 256), segment.at.y.quantize(1 / 256)),
vec2(segment.to.x.quantize(1 / 256), segment.to.y.quantize(1 / 256))
)
result.add((segment, winding))
proc transform(shapes: var seq[Polygon], transform: Mat3) =
if transform != mat3():
for shape in shapes.mitems:
for vec in shape.mitems:
vec = transform * vec
proc computeBounds(segments: seq[(Segment, int16)]): Rect =
## Compute the bounds of the segments.
var
xMin = float32.high
xMax = float32.low
yMin = float32.high
yMax = float32.low
for i, (segment, _) in segments:
xMin = min(xMin, min(segment.at.x, segment.to.x))
xMax = max(xMax, max(segment.at.x, segment.to.x))
yMin = min(yMin, segment.at.y)
yMax = max(yMax, segment.to.y)
if xMin.isNaN() or xMax.isNaN() or yMin.isNaN() or yMax.isNaN():
discard
else:
result.x = xMin
result.y = yMin
result.w = xMax - xMin
result.h = yMax - yMin
proc computeBounds*(
path: Path, transform = mat3()
): Rect {.raises: [PixieError].} =
## Compute the bounds of the path.
var shapes = path.commandsToShapes(true, pixelScale(transform))
shapes.transform(transform)
computeBounds(shapes.shapesToSegments())
proc initPartitionEntry(segment: Segment, winding: int16): PartitionEntry =
result.segment = segment
result.winding = winding
let d = segment.at.x - segment.to.x
if d == 0:
result.b = segment.at.x # Leave m = 0, store the x we want in b
else:
result.m = (segment.at.y - segment.to.y) / d
result.b = segment.at.y - result.m * segment.at.x
proc requiresAntiAliasing(segment: Segment): bool {.inline.} =
## Returns true if the segment requires antialiasing.
template hasFractional(v: float32): bool =
v - trunc(v) != 0
if segment.at.x != segment.to.x or
segment.at.x.hasFractional() or # at.x and to.x are the same
segment.at.y.hasFractional() or
segment.to.y.hasFractional():
# AA is required if all segments are not vertical or have fractional > 0
return true
proc requiresAntiAliasing(entries: var seq[PartitionEntry]): bool =
## Returns true if the fill requires antialiasing.
for entry in entries:
if entry.segment.requiresAntiAliasing:
return true
proc partitionSegments(
segments: seq[(Segment, int16)], top, height: int
): seq[Partition] =
## Puts segments into the height partitions they intersect with.
let
maxPartitions = max(1, height div 4).uint32
numPartitions = min(maxPartitions, max(1, segments.len div 2).uint32)
result.setLen(numPartitions)
let
startY = top.uint32
partitionHeight = height.uint32 div numPartitions
# Set the bottom values for the partitions (y value where this partition ends)
result[0].top = top
result[0].bottom = top + partitionHeight.int
for i in 1 ..< result.len:
result[i].top = result[i - 1].bottom
result[i].bottom = result[i - 1].bottom + partitionHeight.int
# Ensure the final partition goes to the actual bottom
# This is needed since the final partition includes
# height - (height div numPartitions) * numPartitions
result[^1].bottom = top + height
var entries = newSeq[PartitionEntry](segments.len)
for i, (segment, winding) in segments:
entries[i] = initPartitionEntry(segment, winding)
if numPartitions == 1:
result[0].entries = move entries
else:
iterator partitionRange(
segment: Segment,
numPartitions, startY, partitionHeight: uint32
): uint32 =
var
atPartition = max(0, segment.at.y - startY.float32).uint32
toPartition = max(0, segment.to.y - startY.float32).uint32
atPartition = atPartition div partitionHeight
toPartition = toPartition div partitionHeight
atPartition = min(atPartition, numPartitions - 1)
toPartition = min(toPartition, numPartitions - 1)
for partitionIndex in atPartition .. toPartition:
yield partitionIndex
var entryCounts = newSeq[int](numPartitions)
for (segment, _) in segments:
for partitionIndex in segment.partitionRange(
numPartitions, startY, partitionHeight
):
inc entryCounts[partitionIndex]
for partitionIndex, entryCounts in entryCounts:
result[partitionIndex].entries.setLen(entryCounts)
var indexes = newSeq[int](numPartitions)
for i, (segment, winding) in segments:
for partitionIndex in segment.partitionRange(
numPartitions, startY, partitionHeight
):
result[partitionIndex].entries[indexes[partitionIndex]] = entries[i]
inc indexes[partitionIndex]
for partition in result.mitems:
partition.requiresAntiAliasing = requiresAntiAliasing(partition.entries)
# Clip the entries to the parition bounds
let
top = partition.top.float32
bottom = partition.bottom.float32
topLine = line(vec2(0, top), vec2(1000, top))
bottomLine = line(vec2(0, bottom), vec2(1000, bottom))
for entry in partition.entries.mitems:
if entry.segment.at.y <= top and entry.segment.to.y >= bottom:
var at: Vec2
discard intersects(entry.segment, topLine, at)
entry.segment.at = at
discard intersects(entry.segment, bottomLine, at)
entry.segment.to = at
if partition.entries.len == 2:
let
entry0 = partition.entries[0].segment
entry1 = partition.entries[1].segment
var at: Vec2
if not intersectsInside(entry0, entry1, at):
# These two segments do not intersect, enable shortcut
partition.twoNonintersectingSpanningSegments = true
# Ensure entry[0] is on the left
if entry1.at.x < entry0.at.x:
swap partition.entries[1], partition.entries[0]
proc maxEntryCount(partitions: var seq[Partition]): int =
for i in 0 ..< partitions.len:
result = max(result, partitions[i].entries.len)
proc fixed32(f: float32): Fixed32 {.inline.} =
Fixed32(f * 256)
proc integer(p: Fixed32): int {.inline.} =
p div 256
proc trunc(p: Fixed32): Fixed32 {.inline.} =
(p div 256) * 256
proc sortHits(hits: var seq[(Fixed32, int16)], len: int) {.inline.} =
## Insertion sort
for i in 1 ..< len:
var
j = i - 1
k = i
while j >= 0 and hits[j][0] > hits[k][0]:
swap(hits[j + 1], hits[j])
dec j
dec k
proc shouldFill(
windingRule: WindingRule, count: int
): bool {.inline.} =
## Should we fill based on the current winding rule and count?
case windingRule:
of NonZero:
count != 0
of EvenOdd:
count mod 2 != 0
iterator walk(
hits: seq[(Fixed32, int16)],
numHits: int,
windingRule: WindingRule,
y: int,
width: int
): (Fixed32, Fixed32, int) =
var
i, count: int
prevAt: Fixed32
while i < numHits:
let (at, winding) = hits[i]
if at > 0:
if shouldFill(windingRule, count):
if i < numHits - 1:
# Look ahead to see if the next hit is in the same spot as this hit.
# If it is, see if this hit and the next hit's windings cancel out.
# If they do, skip the hits. It will be yielded later in a
# larger chunk.
let (nextAt, nextWinding) = hits[i + 1]
if nextAt == at and winding + nextWinding == 0:
i += 2
continue
# Shortcut: we only care about when we stop filling (or the last hit).
# If we continue filling, move to next hit.
if windingRule == NonZero and count + winding != 0:
count += winding
inc i
continue
yield (prevAt, at, count)
prevAt = at
count += winding
inc i
when defined(pixieLeakCheck):
if prevAt != width.float32.fixed32 and count != 0:
echo "Leak detected: ", count, " @ (", prevAt, ", ", y, ")"
iterator walkInteger(
hits: seq[(int32, int16)],
numHits: int,
windingRule: WindingRule,
y, width: int
): (int, int) =
for (prevAt, at, count) in hits.walk(numHits, windingRule, y, width):
let
fillStart = prevAt.integer
fillLen = at.integer - fillStart
if fillLen <= 0:
continue
yield (fillStart, fillLen)
proc computeCoverage(
coverages: ptr UncheckedArray[uint8],
hits: var seq[(Fixed32, int16)],
numHits: var int,
aa: var bool,
width: int,
y, startX: int,
partitions: var seq[Partition],
partitionIndex: int,
windingRule: WindingRule
) {.inline.} =
aa = partitions[partitionIndex].requiresAntiAliasing
let
quality = if aa: 5 else: 1 # Must divide 255 cleanly (1, 3, 5, 15, 17, 51, 85)
sampleCoverage = (255 div quality).uint8
offset = 1 / quality.float32
initialOffset = offset / 2 + epsilon
var yLine = y.float32 + initialOffset - offset
for m in 0 ..< quality:
yLine += offset
numHits = 0
for entry in partitions[partitionIndex].entries.mitems:
if entry.segment.at.y <= yLine and entry.segment.to.y >= yLine:
let x =
if entry.m == 0:
entry.b
else:
(yLine - entry.b) / entry.m
hits[numHits] = (min(x, width.float32).fixed32, entry.winding)
inc numHits
if numHits > 0:
sortHits(hits, numHits)
if aa:
for (prevAt, at, count) in hits.walk(numHits, windingRule, y, width):
var fillStart = prevAt.integer
let
pixelCrossed = at.integer != prevAt.integer
leftCover =
if pixelCrossed:
prevAt.trunc + 1.0.fixed32 - prevAt
else:
at - prevAt
if leftCover != 0:
inc fillStart
coverages[prevAt.integer - startX] +=
(leftCover * sampleCoverage.int32).integer.uint8
if pixelCrossed:
let rightCover = at - at.trunc
if rightCover > 0:
coverages[at.integer - startX] +=
(rightCover * sampleCoverage.int32).integer.uint8
let fillLen = at.integer - fillStart
if fillLen > 0:
var i = fillStart
when defined(amd64) and allowSimd:
let sampleCoverageVec = mm_set1_epi8(sampleCoverage)
for _ in 0 ..< fillLen div 16:
var coverageVec = mm_loadu_si128(coverages[i - startX].addr)
coverageVec = mm_add_epi8(coverageVec, sampleCoverageVec)
mm_storeu_si128(coverages[i - startX].addr, coverageVec)
i += 16
for j in i ..< fillStart + fillLen:
coverages[j - startX] += sampleCoverage
proc clearUnsafe(target: Image | Mask, startX, startY, toX, toY: int) =
## Clears data from [start, to).
if startX == target.width or startY == target.height:
return
let
start = target.dataIndex(startX, startY)
len = target.dataIndex(toX, toY) - start
when type(target) is Image:
fillUnsafe(target.data, rgbx(0, 0, 0, 0), start, len)
else: # target is Mask
fillUnsafe(target.data, 0, start, len)
proc fillCoverage(
image: Image,
rgbx: ColorRGBX,
startX, y: int,
coverages: seq[uint8],
blendMode: BlendMode
) =
var
x = startX
dataIndex = image.dataIndex(x, y)
when allowSimd:
when defined(amd64):
iterator simd(
coverages: seq[uint8], x: var int, startX: int
): (M128i, bool, bool) =
for _ in 0 ..< coverages.len div 16:
let
coverageVec = mm_loadu_si128(coverages[x - startX].unsafeAddr)
eqZero = mm_cmpeq_epi8(coverageVec, mm_setzero_si128())
eq255 = mm_cmpeq_epi8(coverageVec, mm_set1_epi8(255))
allZeroes = mm_movemask_epi8(eqZero) == 0xffff
all255 = mm_movemask_epi8(eq255) == 0xffff
yield (coverageVec, allZeroes, all255)
x += 16
proc source(colorVec, coverageVec: M128i): M128i {.inline.} =
let
oddMask = mm_set1_epi16(cast[int16](0xff00))
div255 = mm_set1_epi16(cast[int16](0x8081))
var unpacked = unpackAlphaValues(coverageVec)
unpacked = mm_or_si128(unpacked, mm_srli_epi32(unpacked, 16))
var
sourceEven = mm_slli_epi16(colorVec, 8)
sourceOdd = mm_and_si128(colorVec, oddMask)
sourceEven = mm_mulhi_epu16(sourceEven, unpacked)
sourceOdd = mm_mulhi_epu16(sourceOdd, unpacked)
sourceEven = mm_srli_epi16(mm_mulhi_epu16(sourceEven, div255), 7)
sourceOdd = mm_srli_epi16(mm_mulhi_epu16(sourceOdd, div255), 7)
result = mm_or_si128(sourceEven, mm_slli_epi16(sourceOdd, 8))
let colorVec = mm_set1_epi32(cast[int32](rgbx))
proc source(rgbx: ColorRGBX, coverage: uint8): ColorRGBX {.inline.} =
if coverage == 0:
discard
elif coverage == 255:
result = rgbx
else:
result = rgbx(
((rgbx.r.uint32 * coverage) div 255).uint8,
((rgbx.g.uint32 * coverage) div 255).uint8,
((rgbx.b.uint32 * coverage) div 255).uint8,
((rgbx.a.uint32 * coverage) div 255).uint8
)
case blendMode:
of OverwriteBlend:
when allowSimd:
when defined(amd64):
for (coverageVec, allZeroes, all255) in simd(coverages, x, startX):
if allZeroes:
dataIndex += 16
else:
if all255:
for i in 0 ..< 4:
mm_storeu_si128(image.data[dataIndex].addr, colorVec)
dataIndex += 4
else:
var coverageVec = coverageVec
for i in 0 ..< 4:
let source = source(colorVec, coverageVec)
mm_storeu_si128(image.data[dataIndex].addr, source)
coverageVec = mm_srli_si128(coverageVec, 4)
dataIndex += 4
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage != 0:
image.data[dataIndex] = source(rgbx, coverage)
inc dataIndex
of NormalBlend:
when allowSimd:
when defined(amd64):
for (coverageVec, allZeroes, all255) in simd(coverages, x, startX):
if allZeroes:
dataIndex += 16
else:
if all255 and rgbx.a == 255:
for i in 0 ..< 4:
mm_storeu_si128(image.data[dataIndex].addr, colorVec)
dataIndex += 4
else:
var coverageVec = coverageVec
for i in 0 ..< 4:
let
backdrop = mm_loadu_si128(image.data[dataIndex].addr)
source = source(colorVec, coverageVec)
mm_storeu_si128(
image.data[dataIndex].addr,
blendNormalSimd(backdrop, source)
)
coverageVec = mm_srli_si128(coverageVec, 4)
dataIndex += 4
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage == 255 and rgbx.a == 255:
image.data[dataIndex] = rgbx
elif coverage == 0:
discard
else:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendNormal(backdrop, source(rgbx, coverage))
inc dataIndex
of MaskBlend:
{.linearScanEnd.}
when allowSimd:
when defined(amd64):
for (coverageVec, allZeroes, all255) in simd(coverages, x, startX):
if not allZeroes:
if all255:
dataIndex += 16
else:
var coverageVec = coverageVec
for i in 0 ..< 4:
let
backdrop = mm_loadu_si128(image.data[dataIndex].addr)
source = source(colorVec, coverageVec)
mm_storeu_si128(
image.data[dataIndex].addr,
blendMaskSimd(backdrop, source)
)
coverageVec = mm_srli_si128(coverageVec, 4)
dataIndex += 4
else:
for i in 0 ..< 4:
mm_storeu_si128(image.data[dataIndex].addr, mm_setzero_si128())
dataIndex += 4
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage == 0:
image.data[dataIndex] = rgbx(0, 0, 0, 0)
elif coverage == 255:
discard
else:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendMask(backdrop, source(rgbx, coverage))
inc dataIndex
image.clearUnsafe(0, y, startX, y)
image.clearUnsafe(startX + coverages.len, y, image.width, y)
of SubtractMaskBlend:
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage == 255 and rgbx.a == 255:
image.data[dataIndex] = rgbx(0, 0, 0, 0)
elif coverage != 0:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendSubtractMask(backdrop, source(rgbx, coverage))
inc dataIndex
of ExcludeMaskBlend:
for x in x ..< startX + coverages.len:
let
coverage = coverages[x - startX]
backdrop = image.data[dataIndex]
image.data[dataIndex] = blendExcludeMask(backdrop, source(rgbx, coverage))
inc dataIndex
else:
let blender = blendMode.blender()
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage != 0:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blender(backdrop, source(rgbx, coverage))
inc dataIndex
proc fillCoverage(
mask: Mask,
startX, y: int,
coverages: seq[uint8],
blendMode: BlendMode
) =
var
x = startX
dataIndex = mask.dataIndex(x, y)
template simdBlob(blendProc: untyped) =
when allowSimd:
when defined(amd64):
for _ in 0 ..< coverages.len div 16:
let
coveragesVec = mm_loadu_si128(coverages[x - startX].unsafeAddr)
eqZero = mm_cmpeq_epi8(coveragesVec, mm_setzero_si128())
allZeroes = mm_movemask_epi8(eqZero) == 0xffff
if not allZeroes:
let backdrop = mm_loadu_si128(mask.data[dataIndex].addr)
mm_storeu_si128(
mask.data[dataIndex].addr,
blendProc(backdrop, coveragesVec)
)
x += 16
dataIndex += 16
template blendBlob(blendProc: untyped) =
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage != 0:
let backdrop = mask.data[dataIndex]
mask.data[dataIndex] = blendProc(backdrop, coverage)
inc dataIndex
case blendMode:
of OverwriteBlend:
copyMem(
mask.unsafe[startX, y].addr,
coverages[0].unsafeAddr,
coverages.len
)
of NormalBlend:
simdBlob(maskBlendNormalSimd)
blendBlob(maskBlendNormal)
of MaskBlend:
{.linearScanEnd.}
when allowSimd:
when defined(amd64):
for _ in 0 ..< coverages.len div 16:
let
coveragesVec = mm_loadu_si128(coverages[x - startX].unsafeAddr)
eqZero = mm_cmpeq_epi8(coveragesVec, mm_setzero_si128())
allZeroes = mm_movemask_epi8(eqZero) == 0xffff
if not allZeroes:
let backdrop = mm_loadu_si128(mask.data[dataIndex].addr)
mm_storeu_si128(
mask.data[dataIndex].addr,
maskBlendMaskSimd(backdrop, coveragesVec)
)
else:
mm_storeu_si128(mask.data[dataIndex].addr, mm_setzero_si128())
x += 16
dataIndex += 16
for x in x ..< startX + coverages.len:
let coverage = coverages[x - startX]
if coverage != 0:
let backdrop = mask.data[dataIndex]
mask.data[dataIndex] = maskBlendMask(backdrop, coverage)
else:
mask.data[dataIndex] = 0
inc dataIndex
mask.clearUnsafe(0, y, startX, y)
mask.clearUnsafe(startX + coverages.len, y, mask.width, y)
of SubtractMaskBlend:
simdBlob(maskBlendSubtractSimd)
blendBlob(maskBlendSubtract)
of ExcludeMaskBlend:
simdBlob(maskBlendExcludeSimd)
blendBlob(maskBlendExclude)
else:
let maskBlender = blendMode.maskBlender()
blendBlob(maskBlender)
proc fillHits(
image: Image,
rgbx: ColorRGBX,
startX, y: int,
hits: seq[(Fixed32, int16)],
numHits: int,
windingRule: WindingRule,
blendMode: BlendMode
) =
template simdBlob(image: Image, x: var int, len: int, blendProc: untyped) =
when allowSimd:
when defined(amd64):
var p = cast[uint](image.data[image.dataIndex(x, y)].addr)
let
iterations = len div 4
colorVec = mm_set1_epi32(cast[int32](rgbx))
for _ in 0 ..< iterations:
let backdrop = mm_loadu_si128(cast[pointer](p))
mm_storeu_si128(cast[pointer](p), blendProc(backdrop, colorVec))
p += 16
x += iterations * 4
case blendMode:
of OverwriteBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
fillUnsafe(image.data, rgbx, image.dataIndex(start, y), len)
of NormalBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
if rgbx.a == 255:
fillUnsafe(image.data, rgbx, image.dataIndex(start, y), len)
else:
var x = start
simdBlob(image, x, len, blendNormalSimd)
var dataIndex = image.dataIndex(x, y)
for _ in x ..< start + len:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendNormal(backdrop, rgbx)
inc dataIndex
of MaskBlend:
{.linearScanEnd.}
var filledTo = startX
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
block: # Clear any gap between this fill and the previous fill
let gapBetween = start - filledTo
if gapBetween > 0:
fillUnsafe(
image.data,
rgbx(0, 0, 0, 0),
image.dataIndex(filledTo, y),
gapBetween
)
filledTo = start + len
block: # Handle this fill
if rgbx.a != 255:
var x = start
simdBlob(image, x, len, blendMaskSimd)
var dataIndex = image.dataIndex(x, y)
for _ in x ..< start + len:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendMask(backdrop, rgbx)
image.clearUnsafe(0, y, startX, y)
image.clearUnsafe(filledTo, y, image.width, y)
of SubtractMaskBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
var dataIndex = image.dataIndex(start, y)
for _ in 0 ..< len:
if rgbx.a == 255:
image.data[dataIndex] = rgbx(0, 0, 0, 0)
else:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendSubtractMask(backdrop, rgbx)
inc dataIndex
of ExcludeMaskBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
var dataIndex = image.dataIndex(start, y)
for _ in 0 ..< len:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blendExcludeMask(backdrop, rgbx)
inc dataIndex
else:
let blender = blendMode.blender()
for (start, len) in hits.walkInteger(numHits, windingRule, y, image.width):
var dataIndex = image.dataIndex(start, y)
for _ in 0 ..< len:
let backdrop = image.data[dataIndex]
image.data[dataIndex] = blender(backdrop, rgbx)
inc dataIndex
proc fillHits(
mask: Mask,
startX, y: int,
hits: seq[(Fixed32, int16)],
numHits: int,
windingRule: WindingRule,
blendMode: BlendMode
) =
template simdBlob(mask: Mask, x: var int, len: int, blendProc: untyped) =
when allowSimd:
when defined(amd64):
var p = cast[uint](mask.data[mask.dataIndex(x, y)].addr)
let
iterations = len div 16
vec255 = mm_set1_epi8(255)
for _ in 0 ..< iterations:
let backdrop = mm_loadu_si128(cast[pointer](p))
mm_storeu_si128(cast[pointer](p), blendProc(backdrop, vec255))
p += 16
x += iterations * 16
case blendMode:
of NormalBlend, OverwriteBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
fillUnsafe(mask.data, 255, mask.dataIndex(start, y), len)
of MaskBlend:
{.linearScanEnd.}
var filledTo = startX
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
let gapBetween = start - filledTo
if gapBetween > 0:
fillUnsafe(mask.data, 0, mask.dataIndex(filledTo, y), gapBetween)
filledTo = start + len
mask.clearUnsafe(0, y, startX, y)
mask.clearUnsafe(filledTo, y, mask.width, y)
of SubtractMaskBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
var x = start
simdBlob(mask, x, len, maskBlendSubtractSimd)
var dataIndex = mask.dataIndex(x, y)
for _ in x ..< start + len:
let backdrop = mask.data[dataIndex]
mask.data[dataIndex] = maskBlendSubtract(backdrop, 255)
inc dataIndex
of ExcludeMaskBlend:
for (start, len) in hits.walkInteger(numHits, windingRule, y, mask.width):
var x = start
simdBlob(mask, x, len, maskBlendExcludeSimd)
var dataIndex = mask.dataIndex(x, y)
for _ in x ..< start + len:
let backdrop = mask.data[dataIndex]
mask.data[dataIndex] = maskBlendExclude(backdrop, 255)
inc dataIndex
else:
failUnsupportedBlendMode(blendMode)
proc fillShapes(
image: Image,
shapes: var seq[Polygon],
color: SomeColor,
windingRule: WindingRule,
blendMode: BlendMode
) =
# Figure out the total bounds of all the shapes,
# rasterize only within the total bounds
let
rgbx = color.asRgbx()
segments = shapes.shapesToSegments()
bounds = computeBounds(segments).snapToPixels()
startX = max(0, bounds.x.int)
startY = max(0, bounds.y.int)
pathWidth =
if startX < image.width:
min(bounds.w.int, image.width - startX)
else:
0
pathHeight = min(image.height, (bounds.y + bounds.h).int)
if pathWidth == 0:
return
if pathWidth < 0:
raise newException(PixieError, "Path int overflow detected")
var
partitions = partitionSegments(segments, startY, pathHeight - startY)
partitionIndex: int
entryIndices = newSeq[int](partitions.maxEntryCount)
numEntryIndices: int
coverages = newSeq[uint8](pathWidth)
hits = newSeq[(Fixed32, int16)](entryIndices.len)
numHits: int
aa: bool
var y = startY
while y < pathHeight:
if y >= partitions[partitionIndex].bottom:
inc partitionIndex
let
partitionTop = partitions[partitionIndex].top
partitionBottom = partitions[partitionIndex].bottom
partitionHeight = partitionBottom - partitionTop
if partitionHeight == 0:
continue
if partitions[partitionIndex].twoNonintersectingSpanningSegments:
if not partitions[partitionIndex].requiresAntiAliasing:
# No AA required, must be 2 vertical pixel-aligned lines
let
left = partitions[partitionIndex].entries[0].segment.at.x.int
right = partitions[partitionIndex].entries[1].segment.at.x.int
minX = left.clamp(0, image.width)
maxX = right.clamp(0, image.width)
skipBlending =
blendMode == OverwriteBlend or
(blendMode == NormalBlend and rgbx.a == 255)
if skipBlending and minX == 0 and maxX == image.width:
# We can be greedy, just do one big mult-row fill
let
start = image.dataIndex(0, y)
len = image.dataIndex(0, y + partitionHeight) - start
fillUnsafe(image.data, rgbx, start, len)
else:
for r in 0 ..< partitionHeight:
hits[0] = (cast[Fixed32](minX * 256), 1.int16)
hits[1] = (cast[Fixed32](maxX * 256), -1.int16)
image.fillHits(rgbx, 0, y + r, hits, 2, NonZero, blendMode)
y += partitionHeight
continue
var allEntriesInScanlineSpanIt = true
numEntryIndices = 0
if partitions[partitionIndex].twoNonintersectingSpanningSegments:
numEntryIndices = 2
entryIndices[0] = 0
entryIndices[1] = 1
else:
for i in 0 ..< partitions[partitionIndex].entries.len:
if partitions[partitionIndex].entries[i].segment.to.y < y.float32 or
partitions[partitionIndex].entries[i].segment.at.y >= (y + 1).float32:
continue
if partitions[partitionIndex].entries[i].segment.at.y > y.float32 or
partitions[partitionIndex].entries[i].segment.to.y < (y + 1).float32:
allEntriesInScanlineSpanIt = false
break
entryIndices[numEntryIndices] = i
inc numEntryIndices
if allEntriesInScanlineSpanIt and numEntryIndices == 2:
var
left = partitions[partitionIndex].entries[entryIndices[0]]
right = partitions[partitionIndex].entries[entryIndices[1]]
block:
# Ensure left is actually on the left
let
maybeLeftMaxX = max(left.segment.at.x, left.segment.to.x)
maybeRightMaxX = max(right.segment.at.x, right.segment.to.x)
if maybeLeftMaxX > maybeRightMaxX:
swap left, right
# Use trapezoid coverage at the edges and fill in the middle
when allowSimd and defined(amd64):
let vecRgbx = mm_set_ps(
rgbx.a.float32,
rgbx.b.float32,
rgbx.g.float32,
rgbx.r.float32
)
proc solveX(entry: PartitionEntry, y: float32): float32 =
if entry.m == 0:
entry.b
else:
(y - entry.b) / entry.m
proc solveY(entry: PartitionEntry, x: float32): float32 =
entry.m * x + entry.b
var
leftTop = vec2(0, y.float32)
leftBottom = vec2(0, (y + 1).float32)
leftTop.x = left.solveX(leftTop.y.float32)
leftBottom.x = left.solveX(leftBottom.y)
var
rightTop = vec2(0, y.float32)
rightBottom = vec2(0, (y + 1).float32)
rightTop.x = right.solveX(rightTop.y)
rightBottom.x = right.solveX(rightBottom.y)
let
leftMaxX = max(leftTop.x, leftBottom.x)
rightMinX = min(rightTop.x, rightBottom.x)
leftCoverEnd = leftMaxX.ceil.int
rightCoverBegin = rightMinX.trunc.int
if leftCoverEnd < rightCoverBegin:
# Only take this shortcut if the partial coverage areas on the
# left and the right do not overlap
let blender = blendMode.blender()
block: # Left-side partial coverage
let
inverted = leftTop.x < leftBottom.x
sliverStart = min(leftTop.x, leftBottom.x)
rectStart = max(leftTop.x, leftBottom.x)
var
pen = sliverStart
prevPen = pen
penY = if inverted: y.float32 else: (y + 1).float32
prevPenY = penY
for x in sliverStart.int ..< rectStart.ceil.int:
prevPen = pen
pen = (x + 1).float32
var rightRectArea = 0.float32
if pen > rectStart:
rightRectArea = pen - rectStart
pen = rectStart
prevPenY = penY
penY = left.solveY(pen)
if x < 0 or x >= image.width:
continue
let
run = pen - prevPen
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
rectArea =
if inverted:
(prevPenY - y.float32) * run
else:
((y + 1).float32 - prevPenY) * run
area = triangleArea + rectArea + rightRectArea
dataIndex = image.dataIndex(x, y)
backdrop = image.data[dataIndex]
source =
when allowSimd and defined(amd64):
applyOpacity(vecRgbx, area)
else:
rgbx * area
image.data[dataIndex] = blender(backdrop, source)
block: # Right-side partial coverage
let
inverted = rightTop.x > rightBottom.x
rectEnd = min(rightTop.x, rightBottom.x)
sliverEnd = max(rightTop.x, rightBottom.x)
var
pen = rectEnd
prevPen = pen
penY = if inverted: (y + 1).float32 else: y.float32
prevPenY = penY
for x in rectEnd.int ..< sliverEnd.ceil.int:
prevPen = pen
pen = (x + 1).float32
let leftRectArea = prevPen.fractional
if pen > sliverEnd:
pen = sliverEnd
prevPenY = penY
penY = right.solveY(pen)
if x < 0 or x >= image.width:
continue
let
run = pen - prevPen
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
rectArea =
if inverted:
(penY - y.float32) * run
else:
((y + 1).float32 - penY) * run
area = leftRectArea + triangleArea + rectArea
dataIndex = image.dataIndex(x, y)
backdrop = image.data[dataIndex]
source =
when allowSimd and defined(amd64):
applyOpacity(vecRgbx, area)
else:
rgbx * area
image.data[dataIndex] = blender(backdrop, source)
let
fillBegin = leftCoverEnd.clamp(0, image.width)
fillEnd = rightCoverBegin.clamp(0, image.width)
if fillEnd - fillBegin > 0:
hits[0] = (fixed32(fillBegin.float32), 1.int16)
hits[1] = (fixed32(fillEnd.float32), -1.int16)
image.fillHits(rgbx, 0, y, hits, 2, NonZero, blendMode)
inc y
continue
computeCoverage(
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
hits,
numHits,
aa,
image.width,
y,
startX,
partitions,
partitionIndex,
windingRule
)
if aa:
image.fillCoverage(
rgbx,
startX,
y,
coverages,
blendMode
)
zeroMem(coverages[0].addr, coverages.len)
else:
image.fillHits(
rgbx,
startX,
y,
hits,
numHits,
windingRule,
blendMode
)
inc y
if blendMode == MaskBlend:
image.clearUnsafe(0, 0, 0, startY)
image.clearUnsafe(0, pathHeight, 0, image.height)
proc fillShapes(
mask: Mask,
shapes: seq[Polygon],
windingRule: WindingRule,
blendMode: BlendMode
) =
# Figure out the total bounds of all the shapes,
# rasterize only within the total bounds
let
segments = shapes.shapesToSegments()
bounds = computeBounds(segments).snapToPixels()
startX = max(0, bounds.x.int)
startY = max(0, bounds.y.int)
pathWidth =
if startX < mask.width:
min(bounds.w.int, mask.width - startX)
else:
0
pathHeight = min(mask.height, (bounds.y + bounds.h).int)
if pathWidth == 0:
return
if pathWidth < 0:
raise newException(PixieError, "Path int overflow detected")
var
partitions = partitionSegments(segments, startY, pathHeight)
partitionIndex: int
coverages = newSeq[uint8](pathWidth)
hits = newSeq[(Fixed32, int16)](partitions.maxEntryCount)
numHits: int
aa: bool
for y in startY ..< pathHeight:
if y >= partitions[partitionIndex].bottom:
inc partitionIndex
computeCoverage(
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
hits,
numHits,
aa,
mask.width,
y,
startX,
partitions,
partitionIndex,
windingRule
)
if aa:
mask.fillCoverage(startX, y, coverages, blendMode)
zeroMem(coverages[0].addr, coverages.len)
else:
mask.fillHits(startX, y, hits, numHits, windingRule, blendMode)
if blendMode == MaskBlend:
mask.clearUnsafe(0, 0, 0, startY)
mask.clearUnsafe(0, pathHeight, 0, mask.height)
proc miterLimitToAngle*(limit: float32): float32 {.inline.} =
## Converts miter-limit-ratio to miter-limit-angle.
arcsin(1 / limit) * 2
proc angleToMiterLimit*(angle: float32): float32 {.inline.} =
## Converts miter-limit-angle to miter-limit-ratio.
1 / sin(angle / 2)
proc strokeShapes(
shapes: seq[Polygon],
strokeWidth: float32,
lineCap: LineCap,
lineJoin: LineJoin,
miterLimit: float32,
dashes: seq[float32],
pixelScale: float32
): seq[Polygon] =
if strokeWidth <= 0:
return
let
halfStroke = strokeWidth / 2
miterAngleLimit = miterLimitToAngle(miterLimit)
proc makeCircle(at: Vec2): Polygon =
let path = newPath()
path.ellipse(at, halfStroke, halfStroke)
path.commandsToShapes(true, pixelScale)[0]
proc makeRect(at, to: Vec2): Polygon =
# Rectangle corners
let
tangent = (to - at).normalize()
normal = vec2(tangent.y, tangent.x)
a = vec2(
at.x + normal.x * halfStroke,
at.y - normal.y * halfStroke
)
b = vec2(
to.x + normal.x * halfStroke,
to.y - normal.y * halfStroke
)
c = vec2(
to.x - normal.x * halfStroke,
to.y + normal.y * halfStroke
)
d = vec2(
at.x - normal.x * halfStroke,
at.y + normal.y * halfStroke
)
@[a, b, c, d, a]
proc addJoin(shape: var seq[Polygon], prevPos, pos, nextPos: Vec2) =
let minArea = pixelErrorMargin / pixelScale
if lineJoin == RoundJoin:
let area = PI.float32 * halfStroke * halfStroke
if area > minArea:
shape.add makeCircle(pos)
return
let angle = fixAngle(angle(nextPos - pos) - angle(prevPos - pos))
if abs(abs(angle) - PI) > epsilon:
var
a = (pos - prevPos).normalize() * halfStroke
b = (pos - nextPos).normalize() * halfStroke
if angle >= 0:
a = vec2(-a.y, a.x)
b = vec2(b.y, -b.x)
else:
a = vec2(a.y, -a.x)
b = vec2(-b.y, b.x)
var lineJoin = lineJoin
if lineJoin == MiterJoin and abs(angle) < miterAngleLimit:
lineJoin = BevelJoin
case lineJoin:
of MiterJoin:
let
la = line(prevPos + a, pos + a)
lb = line(nextPos + b, pos + b)
var at: Vec2
if la.intersects(lb, at):
let
bisectorLengthSq = (at - pos).lengthSq
areaSq = 0.25.float32 * (
a.lengthSq * bisectorLengthSq + b.lengthSq * bisectorLengthSq
)
if areaSq > (minArea * minArea):
shape.add @[pos + a, at, pos + b, pos, pos + a]
of BevelJoin:
let areaSq = 0.25.float32 * a.lengthSq * b.lengthSq
if areaSq > (minArea * minArea):
shape.add @[a + pos, b + pos, pos, a + pos]
of RoundJoin:
discard # Handled above, skipping angle calculation
for shape in shapes:
var shapeStroke: seq[Polygon]
if shape[0] != shape[^1]:
# This shape does not end at the same point it starts so draw the
# first line cap.
case lineCap:
of ButtCap:
discard
of RoundCap:
shapeStroke.add(makeCircle(shape[0]))
of SquareCap:
let tangent = (shape[1] - shape[0]).normalize()
shapeStroke.add(makeRect(
shape[0] - tangent * halfStroke,
shape[0]
))
var dashes = dashes
if dashes.len mod 2 != 0:
dashes.add(dashes)
# Make sure gaps and dashes are more then zero, otherwise it will hang.
for d in dashes:
if d <= 0.0:
raise newException(PixieError, "Invalid line dash value")
for i in 1 ..< shape.len:
let
pos = shape[i]
prevPos = shape[i - 1]
if dashes.len > 0:
var distance = dist(prevPos, pos)
let dir = dir(pos, prevPos)
var currPos = prevPos
block dashLoop:
while true:
for i, d in dashes:
if i mod 2 == 0:
let d = min(distance, d)
shapeStroke.add(makeRect(currPos, currPos + dir * d))
currPos += dir * d
distance -= d
if distance <= 0:
break dashLoop
else:
shapeStroke.add(makeRect(prevPos, pos))
# If we need a line join
if i < shape.len - 1:
shapeStroke.addJoin(prevPos, pos, shape[i + 1])
if shape[0] == shape[^1]:
shapeStroke.addJoin(shape[^2], shape[^1], shape[1])
else:
case lineCap:
of ButtCap:
discard
of RoundCap:
shapeStroke.add(makeCircle(shape[^1]))
of SquareCap:
let tangent = (shape[^1] - shape[^2]).normalize()
shapeStroke.add(makeRect(
shape[^1] + tangent * halfStroke,
shape[^1]
))
result.add(shapeStroke)
proc parseSomePath(
path: SomePath, closeSubpaths: bool, pixelScale: float32
): seq[Polygon] {.inline.} =
## Given SomePath, parse it in different ways.
when type(path) is string:
parsePath(path).commandsToShapes(closeSubpaths, pixelScale)
elif type(path) is Path:
path.commandsToShapes(closeSubpaths, pixelScale)
proc fillPath*(
mask: Mask,
path: SomePath,
transform = mat3(),
windingRule = NonZero,
blendMode = NormalBlend
) {.raises: [PixieError].} =
## Fills a path.
var shapes = parseSomePath(path, true, transform.pixelScale())
shapes.transform(transform)
mask.fillShapes(shapes, windingRule, blendMode)
proc fillPath*(
image: Image,
path: SomePath,
paint: Paint,
transform = mat3(),
windingRule = NonZero
) {.raises: [PixieError].} =
## Fills a path.
paint.opacity = clamp(paint.opacity, 0, 1)
if paint.opacity == 0:
return
if paint.kind == SolidPaint:
if paint.color.a > 0 or paint.blendMode == OverwriteBlend:
var shapes = parseSomePath(path, true, transform.pixelScale())
shapes.transform(transform)
var color = paint.color
color.a *= paint.opacity
image.fillShapes(shapes, color, windingRule, paint.blendMode)
return
let
mask = newMask(image.width, image.height)
fill = newImage(image.width, image.height)
mask.fillPath(path, transform, windingRule)
# Draw the image (maybe tiled) or gradients. Do this with opaque paint and
# and then apply the paint's opacity to the mask.
let savedOpacity = paint.opacity
paint.opacity = 1
case paint.kind:
of SolidPaint:
discard # Handled above
of ImagePaint:
fill.draw(paint.image, paint.imageMat)
of TiledImagePaint:
fill.drawTiled(paint.image, paint.imageMat)
of LinearGradientPaint, RadialGradientPaint, AngularGradientPaint:
fill.fillGradient(paint)
paint.opacity = savedOpacity
if paint.opacity != 1:
mask.applyOpacity(paint.opacity)
fill.draw(mask)
image.draw(fill, blendMode = paint.blendMode)
proc strokePath*(
mask: Mask,
path: SomePath,
transform = mat3(),
strokeWidth: float32 = 1.0,
lineCap = ButtCap,
lineJoin = MiterJoin,
miterLimit = defaultMiterLimit,
dashes: seq[float32] = @[],
blendMode = NormalBlend
) {.raises: [PixieError].} =
## Strokes a path.
let pixelScale = transform.pixelScale()
var strokeShapes = strokeShapes(
parseSomePath(path, false, pixelScale),
strokeWidth,
lineCap,
lineJoin,
miterLimit,
dashes,
pixelScale
)
strokeShapes.transform(transform)
mask.fillShapes(strokeShapes, NonZero, blendMode)
proc strokePath*(
image: Image,
path: SomePath,
paint: Paint,
transform = mat3(),
strokeWidth: float32 = 1.0,
lineCap = ButtCap,
lineJoin = MiterJoin,
miterLimit = defaultMiterLimit,
dashes: seq[float32] = @[]
) {.raises: [PixieError].} =
## Strokes a path.
paint.opacity = clamp(paint.opacity, 0, 1)
if paint.opacity == 0:
return
if paint.kind == SolidPaint:
if paint.color.a > 0 or paint.blendMode == OverwriteBlend:
var strokeShapes = strokeShapes(
parseSomePath(path, false, transform.pixelScale()),
strokeWidth,
lineCap,
lineJoin,
miterLimit,
dashes,
pixelScale(transform)
)
strokeShapes.transform(transform)
var color = paint.color
color.a *= paint.opacity
image.fillShapes(strokeShapes, color, NonZero, paint.blendMode)
return
let
mask = newMask(image.width, image.height)
fill = newImage(image.width, image.height)
mask.strokePath(
path,
transform,
strokeWidth,
lineCap,
lineJoin,
miterLimit,
dashes
)
# Draw the image (maybe tiled) or gradients. Do this with opaque paint and
# and then apply the paint's opacity to the mask.
let savedOpacity = paint.opacity
paint.opacity = 1
case paint.kind:
of SolidPaint:
discard # Handled above
of ImagePaint:
fill.draw(paint.image, paint.imageMat)
of TiledImagePaint:
fill.drawTiled(paint.image, paint.imageMat)
of LinearGradientPaint, RadialGradientPaint, AngularGradientPaint:
fill.fillGradient(paint)
paint.opacity = savedOpacity
if paint.opacity != 1:
mask.applyOpacity(paint.opacity)
fill.draw(mask)
image.draw(fill, blendMode = paint.blendMode)
proc overlaps(
shapes: seq[Polygon],
test: Vec2,
windingRule: WindingRule
): bool =
var hits: seq[(Fixed32, int16)]
let
scanline = line(vec2(0, test.y), vec2(1000, test.y))
segments = shapes.shapesToSegments()
for (segment, winding) in segments:
if segment.at.y <= scanline.a.y and segment.to.y >= scanline.a.y:
var at: Vec2
if scanline.intersects(segment, at):
if segment.to != at:
hits.add((at.x.fixed32, winding))
sortHits(hits, hits.len)
let testX = test.x.fixed32
var count: int
for (at, winding) in hits:
if at > testX:
return shouldFill(windingRule, count)
count += winding
proc fillOverlaps*(
path: Path,
test: Vec2,
transform = mat3(), ## Applied to the path, not the test point.
windingRule = NonZero
): bool {.raises: [PixieError].} =
## Returns whether or not the specified point is contained in the current path.
var shapes = path.commandsToShapes(true, transform.pixelScale())
shapes.transform(transform)
shapes.overlaps(test, windingRule)
proc strokeOverlaps*(
path: Path,
test: Vec2,
transform = mat3(), ## Applied to the path, not the test point.
strokeWidth: float32 = 1.0,
lineCap = ButtCap,
lineJoin = MiterJoin,
miterLimit = defaultMiterLimit,
dashes: seq[float32] = @[],
): bool {.raises: [PixieError].} =
## Returns whether or not the specified point is inside the area contained
## by the stroking of a path.
let pixelScale = transform.pixelScale()
var strokeShapes = strokeShapes(
path.commandsToShapes(false, pixelScale),
strokeWidth,
lineCap,
lineJoin,
miterLimit,
dashes,
pixelScale
)
strokeShapes.transform(transform)
strokeShapes.overlaps(test, NonZero)
when defined(release):
{.pop.}