stroke optimization
This commit is contained in:
parent
4543537447
commit
b9fd904042
1 changed files with 39 additions and 17 deletions
|
@ -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)
|
||||||
|
|
Loading…
Reference in a new issue