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