diff --git a/src/pixie/fileformats/svg.nim b/src/pixie/fileformats/svg.nim index 6681e92..14ac205 100644 --- a/src/pixie/fileformats/svg.nim +++ b/src/pixie/fileformats/svg.nim @@ -1,7 +1,7 @@ ## Load and Save SVG files. import chroma, pixie/images, pixie/common, pixie/paths, vmath, xmlparser, - xmltree, strutils, strutils, bumpy + xmltree, strutils, strutils const svgSignature* = " 0: - let (bounds, strokeImg) = strokePathBounds( + img.strokePath( d, ctx.stroke, ctx.strokeWidth, ctx.transform ) - img.draw(strokeImg, bounds.xy) of "line": let @@ -219,13 +217,11 @@ proc draw( let d = $path if ctx.fill != ColorRGBA(): - let (bounds, fillImg) = fillPathBounds(d, ctx.fill, ctx.transform) - img.draw(fillImg, bounds.xy) + img.fillPath(d, ctx.fill, ctx.transform) if ctx.stroke != ColorRGBA() and ctx.strokeWidth > 0: - let (bounds, strokeImg) = strokePathBounds( + img.strokePath( d, ctx.stroke, ctx.strokeWidth, ctx.transform ) - img.draw(strokeImg, bounds.xy) else: raise newException(PixieError, "Unsupported SVG tag: " & node.tag & ".") diff --git a/src/pixie/paths.nim b/src/pixie/paths.nim index 5ef0586..0870c3d 100644 --- a/src/pixie/paths.nim +++ b/src/pixie/paths.nim @@ -1,4 +1,4 @@ -import vmath, images, chroma, strutils, algorithm, common, bumpy +import vmath, images, chroma, strutils, algorithm, common, bumpy, blends type WindingRule* = enum @@ -594,15 +594,17 @@ proc computeBounds(polys: seq[seq[Vec2]]): Rect = {.push checks: off, stacktrace: off.} proc fillPolygons*( + image: Image, size: Vec2, polys: seq[seq[Vec2]], color: ColorRGBA, windingRule: WindingRule, - quality = 4, -): Image = - const ep = 0.0001 * PI + blendMode: BlendMode = bmNormal, + quality = 4 +) = - result = newImage(size.x.int, size.y.int) + const ep = 0.0001 * PI + let mixer = blendMode.mixer() proc scanLineHits( polys: seq[seq[Vec2]], @@ -630,12 +632,12 @@ proc fillPolygons*( var hits = newSeq[(float32, bool)]() - alphas = newSeq[float32](result.width) - for y in 0 ..< result.height: + alphas = newSeq[float32](image.width) + for y in 0 ..< image.height: # Reset alphas for this row. zeroMem(alphas[0].addr, alphas.len * 4) - # Do scanlines for this row. + # Do scan lines for this row. for m in 0 ..< quality: polys.scanLineHits(hits, size, y, float32(m) / float32(quality)) if hits.len == 0: @@ -643,7 +645,7 @@ proc fillPolygons*( var penFill = 0 curHit = 0 - for x in 0 ..< result.width: + for x in 0 ..< image.width: var penEdge: float32 case windingRule of wrNonZero: @@ -669,11 +671,13 @@ proc fillPolygons*( inc curHit alphas[x] += penEdge - for x in 0 ..< result.width: + for x in 0 ..< image.width: let a = clamp(abs(alphas[x]) / float32(quality), 0.0, 1.0) - var colorWithAlpha = color - colorWithAlpha.a = uint8(a * 255.0) - result.setRgbaUnsafe(x, y, colorWithAlpha) + if a > 0: + var colorWithAlpha = color + colorWithAlpha.a = uint8(a * 255.0) + let rgba = image.getRgbaUnsafe(x, y) + image.setRgbaUnsafe(x, y, mixer(rgba, colorWithAlpha)) {.pop.} @@ -698,8 +702,7 @@ proc fillPath*( ) = let polys = parseSomePath(path) - tmp = fillPolygons(image.wh, polys, color, windingRule) - image.draw(tmp) + image.fillPolygons(image.wh, polys, color, windingRule) proc fillPath*( image: Image, @@ -712,8 +715,7 @@ proc fillPath*( for poly in polys.mitems: for i, p in poly.mpairs: poly[i] = p + pos - let tmp = fillPolygons(image.wh, polys, color, windingRule) - image.draw(tmp) + image.fillPolygons(image.wh, polys, color, windingRule) proc fillPath*( image: Image, @@ -726,25 +728,7 @@ proc fillPath*( for poly in polys.mitems: for i, p in poly.mpairs: poly[i] = mat * p - let tmp = fillPolygons(image.wh, polys, color, windingRule) - image.draw(tmp) - -proc fillPathBounds*( - path: SomePath, - color: ColorRGBA, - mat: Mat3, - windingRule = wrNonZero - ): (Rect, Image) = - var polys = parseSomePath(path) - for poly in polys.mitems: - for i, p in poly.mpairs: - poly[i] = mat * p - var bounds = computeBounds(polys) - for poly in polys.mitems: - for i, p in poly.mpairs: - poly[i] = p - bounds.xy - var image = fillPolygons(bounds.wh, polys, color, windingRule) - return (bounds, image) + image.fillPolygons(image.wh, polys, color, windingRule) proc strokePath*( image: Image, @@ -761,8 +745,7 @@ proc strokePath*( polys = parseSomePath(path) (strokeL, strokeR) = (strokeWidth/2, strokeWidth/2) polys2 = strokePolygons(polys, strokeL, strokeR) - tmp = fillPolygons(image.wh, polys2, color, windingRule) - image.draw(tmp) + image.fillPath(polys2, color, windingRule) proc strokePath*( image: Image, @@ -787,8 +770,7 @@ proc strokePath*( for poly in polys2.mitems: for i, p in poly.mpairs: poly[i] = p + pos - let tmp = fillPolygons(image.wh, polys2, color, windingRule) - image.draw(tmp) + image.fillPolygons(image.wh, polys2, color, windingRule) proc strokePath*( image: Image, @@ -804,20 +786,7 @@ proc strokePath*( for poly in polys2.mitems: for i, p in poly.mpairs: poly[i] = mat * p - let tmp = fillPolygons(image.wh, polys2, color, windingRule) - image.draw(tmp) - -proc strokePathBounds*( - path: SomePath, - color: ColorRGBA, - strokeWidth: float32, - mat: Mat3, - windingRule = wrNonZero -): (Rect, Image) = - var polys = parseSomePath(path) - let (strokeL, strokeR) = (strokeWidth/2, strokeWidth/2) - var polys2 = strokePolygons(polys, strokeL, strokeR) - fillPathBounds(polys2, color, mat, windingRule) + image.fillPolygons(image.wh, polys2, color, windingRule) proc addPath*(path: Path, other: Path) = ## Adds a path to the current path. diff --git a/tests/images/paths/pathBottomArc.png b/tests/images/paths/pathBottomArc.png index f535a0e..549d69c 100644 Binary files a/tests/images/paths/pathBottomArc.png and b/tests/images/paths/pathBottomArc.png differ diff --git a/tests/images/paths/pathCornerArc.png b/tests/images/paths/pathCornerArc.png index 26f348f..0d625f7 100644 Binary files a/tests/images/paths/pathCornerArc.png and b/tests/images/paths/pathCornerArc.png differ diff --git a/tests/images/paths/pathHeart.png b/tests/images/paths/pathHeart.png index 6c4c9ad..53ca348 100644 Binary files a/tests/images/paths/pathHeart.png and b/tests/images/paths/pathHeart.png differ diff --git a/tests/images/paths/pathInvertedCornerArc.png b/tests/images/paths/pathInvertedCornerArc.png index 50691b0..b3ef73d 100644 Binary files a/tests/images/paths/pathInvertedCornerArc.png and b/tests/images/paths/pathInvertedCornerArc.png differ diff --git a/tests/images/paths/pathRedRectangle.png b/tests/images/paths/pathRedRectangle.png index d649f68..50d16e2 100644 Binary files a/tests/images/paths/pathRedRectangle.png and b/tests/images/paths/pathRedRectangle.png differ diff --git a/tests/images/paths/pathRotatedArc.png b/tests/images/paths/pathRotatedArc.png index a4d3a32..a26dd26 100644 Binary files a/tests/images/paths/pathRotatedArc.png and b/tests/images/paths/pathRotatedArc.png differ diff --git a/tests/images/paths/pathRoundRect.png b/tests/images/paths/pathRoundRect.png index 360e0fc..239c284 100644 Binary files a/tests/images/paths/pathRoundRect.png and b/tests/images/paths/pathRoundRect.png differ diff --git a/tests/images/paths/pathStroke1.png b/tests/images/paths/pathStroke1.png index 48666be..9041512 100644 Binary files a/tests/images/paths/pathStroke1.png and b/tests/images/paths/pathStroke1.png differ diff --git a/tests/images/paths/pathStroke2.png b/tests/images/paths/pathStroke2.png index 7e99ee7..a3ac4b3 100644 Binary files a/tests/images/paths/pathStroke2.png and b/tests/images/paths/pathStroke2.png differ diff --git a/tests/images/paths/pathStroke3.png b/tests/images/paths/pathStroke3.png index 21006c7..06d6937 100644 Binary files a/tests/images/paths/pathStroke3.png and b/tests/images/paths/pathStroke3.png differ diff --git a/tests/images/paths/pathYellowRectangle.png b/tests/images/paths/pathYellowRectangle.png index c1b1b2b..5417412 100644 Binary files a/tests/images/paths/pathYellowRectangle.png and b/tests/images/paths/pathYellowRectangle.png differ diff --git a/tests/images/svg/Ghostscript_Tiger.png b/tests/images/svg/Ghostscript_Tiger.png index 0c98687..dece4b0 100644 Binary files a/tests/images/svg/Ghostscript_Tiger.png and b/tests/images/svg/Ghostscript_Tiger.png differ diff --git a/tests/images/svg/circle01.png b/tests/images/svg/circle01.png index d4d4ddf..f6d36db 100644 Binary files a/tests/images/svg/circle01.png and b/tests/images/svg/circle01.png differ diff --git a/tests/images/svg/ellipse01.png b/tests/images/svg/ellipse01.png index d3b815a..d2fd2f7 100644 Binary files a/tests/images/svg/ellipse01.png and b/tests/images/svg/ellipse01.png differ diff --git a/tests/images/svg/quad01.png b/tests/images/svg/quad01.png index 0ff6c06..8c19391 100644 Binary files a/tests/images/svg/quad01.png and b/tests/images/svg/quad01.png differ diff --git a/tests/images/svg/triangle01.png b/tests/images/svg/triangle01.png index 603107f..38b481a 100644 Binary files a/tests/images/svg/triangle01.png and b/tests/images/svg/triangle01.png differ diff --git a/tests/test_svg.nim b/tests/test_svg.nim index 2148145..2d09953 100644 --- a/tests/test_svg.nim +++ b/tests/test_svg.nim @@ -20,4 +20,4 @@ for file in files: gold = readImage(&"tests/images/svg/{file}.png") doAssert image.data == gold.data - # image.writeFile(&"{file}.png") + #image.writeFile(&"tests/images/svg/{file}.png")