stroke optimization

This commit is contained in:
Ryan Oldenburg 2021-11-25 05:02:14 -06:00
parent 4543537447
commit b9fd904042

View file

@ -639,7 +639,7 @@ proc polygon*(
path.polygon(pos.x, pos.y, size, sides) path.polygon(pos.x, pos.y, size, sides)
proc commandsToShapes( proc commandsToShapes(
path: Path, closeSubpaths = false, pixelScale: float32 = 1.0 path: Path, closeSubpaths: bool, pixelScale: float32
): seq[seq[Vec2]] = ): seq[seq[Vec2]] =
## Converts SVG-like commands to sequences of vectors. ## Converts SVG-like commands to sequences of vectors.
var var
@ -1080,7 +1080,7 @@ proc computeBounds*(
path: Path, transform = mat3() path: Path, transform = mat3()
): Rect {.raises: [PixieError].} = ): Rect {.raises: [PixieError].} =
## Compute the bounds of the path. ## Compute the bounds of the path.
var shapes = path.commandsToShapes() var shapes = path.commandsToShapes(true, pixelScale(transform))
shapes.transform(transform) shapes.transform(transform)
computeBounds(shapes.shapesToSegments()) computeBounds(shapes.shapesToSegments())
@ -1657,7 +1657,8 @@ proc strokeShapes(
lineCap: LineCap, lineCap: LineCap,
lineJoin: LineJoin, lineJoin: LineJoin,
miterLimit: float32, miterLimit: float32,
dashes: seq[float32] dashes: seq[float32],
pixelScale: float32
): seq[seq[Vec2]] = ): seq[seq[Vec2]] =
if strokeWidth <= 0: if strokeWidth <= 0:
return return
@ -1669,7 +1670,7 @@ proc strokeShapes(
proc makeCircle(at: Vec2): seq[Vec2] = proc makeCircle(at: Vec2): seq[Vec2] =
let path = newPath() let path = newPath()
path.ellipse(at, halfStroke, halfStroke) path.ellipse(at, halfStroke, halfStroke)
path.commandsToShapes()[0] path.commandsToShapes(true, pixelScale)[0]
proc makeRect(at, to: Vec2): seq[Vec2] = proc makeRect(at, to: Vec2): seq[Vec2] =
# Rectangle corners # Rectangle corners
@ -1695,7 +1696,15 @@ proc strokeShapes(
@[a, b, c, d, a] @[a, b, c, d, a]
proc makeJoin(prevPos, pos, nextPos: Vec2): seq[Vec2] = proc addJoin(shape: var seq[seq[Vec2]], prevPos, pos, nextPos: Vec2) =
let minArea = 0.1 / pixelScale # 10% of a pixel
if lineJoin == ljRound:
let area = PI.float32 * halfStroke * halfStroke
if area > minArea:
shape.add makeCircle(pos)
return
let angle = fixAngle(angle(nextPos - pos) - angle(prevPos - pos)) let angle = fixAngle(angle(nextPos - pos) - angle(prevPos - pos))
if abs(abs(angle) - PI) > epsilon: if abs(abs(angle) - PI) > epsilon:
var var
@ -1719,13 +1728,21 @@ proc strokeShapes(
lb = line(nextPos + b, pos + b) lb = line(nextPos + b, pos + b)
var at: Vec2 var at: Vec2
if la.intersects(lb, at): if la.intersects(lb, at):
return @[pos + a, at, pos + b, pos, pos + a] 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 ljBevel: of ljBevel:
return @[a + pos, b + pos, pos, a + pos] let areaSq = 0.25.float32 * a.lengthSq * b.lengthSq
if areaSq > (minArea * minArea):
shape.add @[a + pos, b + pos, pos, a + pos]
of ljRound: of ljRound:
return makeCircle(pos) discard # Handled above, skipping angle calculation
for shape in shapes: for shape in shapes:
var shapeStroke: seq[seq[Vec2]] var shapeStroke: seq[seq[Vec2]]
@ -1773,10 +1790,10 @@ proc strokeShapes(
# If we need a line join # If we need a line join
if i < shape.len - 1: if i < shape.len - 1:
shapeStroke.add(makeJoin(prevPos, pos, shape[i + 1])) shapeStroke.addJoin(prevPos, pos, shape[i + 1])
if shape[0] == shape[^1]: if shape[0] == shape[^1]:
shapeStroke.add(makeJoin(shape[^2], shape[^1], shape[1])) shapeStroke.addJoin(shape[^2], shape[^1], shape[1])
else: else:
case lineCap: case lineCap:
of lcButt: of lcButt:
@ -1793,7 +1810,7 @@ proc strokeShapes(
result.add(shapeStroke) result.add(shapeStroke)
proc parseSomePath( proc parseSomePath(
path: SomePath, closeSubpaths: bool, pixelScale: float32 = 1.0 path: SomePath, closeSubpaths: bool, pixelScale: float32
): seq[seq[Vec2]] {.inline.} = ): seq[seq[Vec2]] {.inline.} =
## Given SomePath, parse it in different ways. ## Given SomePath, parse it in different ways.
when type(path) is string: when type(path) is string:
@ -1874,13 +1891,15 @@ proc strokePath*(
blendMode = bmNormal blendMode = bmNormal
) {.raises: [PixieError].} = ) {.raises: [PixieError].} =
## Strokes a path. ## Strokes a path.
let pixelScale = transform.pixelScale()
var strokeShapes = strokeShapes( var strokeShapes = strokeShapes(
parseSomePath(path, false, transform.pixelScale()), parseSomePath(path, false, pixelScale),
strokeWidth, strokeWidth,
lineCap, lineCap,
lineJoin, lineJoin,
miterLimit, miterLimit,
dashes dashes,
pixelScale
) )
strokeShapes.transform(transform) strokeShapes.transform(transform)
mask.fillShapes(strokeShapes, wrNonZero, blendMode) mask.fillShapes(strokeShapes, wrNonZero, blendMode)
@ -1908,7 +1927,8 @@ proc strokePath*(
lineCap, lineCap,
lineJoin, lineJoin,
miterLimit, miterLimit,
dashes dashes,
pixelScale(transform)
) )
strokeShapes.transform(transform) strokeShapes.transform(transform)
var color = paint.color var color = paint.color
@ -1985,7 +2005,7 @@ proc fillOverlaps*(
windingRule = wrNonZero windingRule = wrNonZero
): bool {.raises: [PixieError].} = ): bool {.raises: [PixieError].} =
## Returns whether or not the specified point is contained in the current path. ## Returns whether or not the specified point is contained in the current path.
var shapes = parseSomePath(path, true, transform.pixelScale()) var shapes = path.commandsToShapes(true, transform.pixelScale())
shapes.transform(transform) shapes.transform(transform)
shapes.overlaps(test, windingRule) shapes.overlaps(test, windingRule)
@ -2001,13 +2021,15 @@ proc strokeOverlaps*(
): bool {.raises: [PixieError].} = ): bool {.raises: [PixieError].} =
## Returns whether or not the specified point is inside the area contained ## Returns whether or not the specified point is inside the area contained
## by the stroking of a path. ## by the stroking of a path.
let pixelScale = transform.pixelScale()
var strokeShapes = strokeShapes( var strokeShapes = strokeShapes(
parseSomePath(path, false, transform.pixelScale()), path.commandsToShapes(false, pixelScale),
strokeWidth, strokeWidth,
lineCap, lineCap,
lineJoin, lineJoin,
miterLimit, miterLimit,
dashes dashes,
pixelScale
) )
strokeShapes.transform(transform) strokeShapes.transform(transform)
strokeShapes.overlaps(test, wrNonZero) strokeShapes.overlaps(test, wrNonZero)