diff --git a/src/pixie/fileformats/svg.nim b/src/pixie/fileformats/svg.nim index 90560f5..b2ffc57 100644 --- a/src/pixie/fileformats/svg.nim +++ b/src/pixie/fileformats/svg.nim @@ -49,22 +49,42 @@ proc decodeCtx(inherited: Ctx, node: XmlNode): Ctx = if transform == "": discard # Inherit else: - if transform.startsWith("matrix("): - let arr = transform[7..^2].split(",") - if arr.len != 6: - failInvalid() - var m = mat3() - m[0] = parseFloat(arr[0]) - m[1] = parseFloat(arr[1]) - m[3] = parseFloat(arr[2]) - m[4] = parseFloat(arr[3]) - m[6] = parseFloat(arr[4]) - m[7] = parseFloat(arr[5]) - result.transform = result.transform * m - else: + template failInvalidTransform(transform: string) = raise newException( - PixieError, "Unsupported SVG transform: " & transform & "." - ) + PixieError, "Unsupported SVG transform: " & transform & "." + ) + + var remaining = transform + while remaining.len > 0: + let index = remaining.find(")") + if index == -1: + failInvalidTransform(transform) + let f = remaining[0 .. index].strip() + remaining = remaining[index + 1 .. ^1] + + if f.startsWith("matrix("): + let arr = f[7 .. ^2].split(",") + if arr.len != 6: + failInvalidTransform(transform) + var m = mat3() + m[0] = parseFloat(arr[0]) + m[1] = parseFloat(arr[1]) + m[3] = parseFloat(arr[2]) + m[4] = parseFloat(arr[3]) + m[6] = parseFloat(arr[4]) + m[7] = parseFloat(arr[5]) + result.transform = result.transform * m + elif f.startsWith("translate("): + let + components = f[10 .. ^2].split(" ") + tx = parseFloat(components[0]) + ty = parseFloat(components[1]) + result.transform = result.transform * translate(vec2(tx, ty)) + elif f.startsWith("rotate("): + let angle = parseFloat(f[7 .. ^2]) * -PI / 180 + result.transform = result.transform * rotationMat3(angle) + else: + failInvalidTransform(transform) proc draw( img: Image, node: XmlNode, ctxStack: var seq[Ctx] @@ -99,6 +119,53 @@ proc draw( ) img.draw(strokeImg, bounds.xy) + of "line": + let + ctx = decodeCtx(ctxStack[^1], node) + x1 = parseFloat(node.attr("x1")) + y1 = parseFloat(node.attr("y1")) + x2 = parseFloat(node.attr("x2")) + y2 = parseFloat(node.attr("y2")) + + let path = newPath() + path.moveTo(x1, y1) + path.lineTo(x2, y2) + path.closePath() + + if ctx.fill != ColorRGBA(): + img.fillPath(path, ctx.fill, ctx.transform) + if ctx.stroke != ColorRGBA() and ctx.strokeWidth > 0: + img.strokePath(path, ctx.stroke, ctx.strokeWidth, ctx.transform) + + of "polyline", "polygon": + let + ctx = decodeCtx(ctxStack[^1], node) + points = node.attr("points") + + var vecs: seq[Vec2] + for pair in points.split(" "): + let parts = pair.split(",") + if parts.len != 2: + failInvalid() + vecs.add(vec2(parseFloat(parts[0]), parseFloat(parts[1]))) + + if vecs.len == 0: + failInvalid() + + let path = newPath() + path.moveTo(vecs[0]) + for i in 1 ..< vecs.len: + path.lineTo(vecs[i]) + + # The difference between polyline and polygon is whether we close the path + if node.tag == "polygon": + path.closePath() + + if ctx.fill != ColorRGBA(): + img.fillPath(path, ctx.fill, ctx.transform) + if ctx.stroke != ColorRGBA() and ctx.strokeWidth > 0: + img.strokePath(path, ctx.stroke, ctx.strokeWidth, ctx.transform) + of "rect": let ctx = decodeCtx(ctxStack[^1], node) @@ -106,12 +173,14 @@ proc draw( y = parseFloat(node.attr("y")) width = parseFloat(node.attr("width")) height = parseFloat(node.attr("height")) - path = newPath() + + let path = newPath() path.moveTo(x, y) - path.lineTo(x + width, x) + path.lineTo(x + width, y) path.lineTo(x + width, y + height) path.lineTo(x, y + height) path.closePath() + if ctx.fill != ColorRGBA(): img.fillPath(path, ctx.fill, ctx.transform) if ctx.stroke != ColorRGBA() and ctx.strokeWidth > 0: @@ -126,13 +195,15 @@ proc draw( cy = parseFloat(node.attr("cy")) r = parseFloat(node.attr("r")) magic = (4.0 * (-1.0 + sqrt(2.0)) / 3) * r - path = newPath() + + let path = newPath() path.moveTo(cx + r, cy) path.bezierCurveTo(cx + r, cy + magic, cx + magic, cy + r, cx, cy + r) path.bezierCurveTo(cx - magic, cy + r, cx - r, cy + magic, cx - r, cy) path.bezierCurveTo(cx - r, cy - magic, cx - magic, cy - r, cx, cy - r) path.bezierCurveTo(cx + magic, cy - r, cx + r, cy - magic, cx + r, cy) path.closePath() + let d = $path if ctx.fill != ColorRGBA(): let (bounds, fillImg) = fillPathBounds(d, ctx.fill, ctx.transform) diff --git a/src/pixie/paths.nim b/src/pixie/paths.nim index eebebca..9500e05 100644 --- a/src/pixie/paths.nim +++ b/src/pixie/paths.nim @@ -828,11 +828,17 @@ proc moveTo*(path: Path, x, y: float32) = path.commands.add PathCommand(kind: Move, numbers: @[x, y]) path.at = vec2(x, y) +proc moveTo*(path: Path, pos: Vec2) = + path.moveTo(pos.x, pos.y) + proc lineTo*(path: Path, x, y: float32) = ## Connects the last point in the subpath to the (x, y) coordinates with a straight line. path.commands.add PathCommand(kind: Line, numbers: @[x, y]) path.at = vec2(x, y) +proc lineTo*(path: Path, pos: Vec2) = + path.lineTo(pos.x, pos.y) + proc bezierCurveTo*(path: Path, x1, y1, x2, y2, x3, y3: float32) = ## Adds a cubic Bézier curve to the path. It requires three points. The first two points are control points and the third one is the end point. The starting point is the last point in the current path, which can be changed using moveTo() before creating the Bézier curve. path.commands.add(PathCommand(kind: Cubic, numbers: @[ diff --git a/tests/images/svg/circle01.png b/tests/images/svg/circle01.png new file mode 100644 index 0000000..d4d4ddf Binary files /dev/null and b/tests/images/svg/circle01.png differ diff --git a/tests/images/svg/circle01.svg b/tests/images/svg/circle01.svg new file mode 100644 index 0000000..bf71ce9 --- /dev/null +++ b/tests/images/svg/circle01.svg @@ -0,0 +1,13 @@ + + + Example circle01 - circle filled with red and stroked with blue + + + + + + + diff --git a/tests/images/svg/line01.png b/tests/images/svg/line01.png new file mode 100644 index 0000000..4ee524a Binary files /dev/null and b/tests/images/svg/line01.png differ diff --git a/tests/images/svg/line01.svg b/tests/images/svg/line01.svg new file mode 100644 index 0000000..bb1d2bf --- /dev/null +++ b/tests/images/svg/line01.svg @@ -0,0 +1,23 @@ + + + Example line01 - lines expressed in user coordinates + + + + + + + + + + + + + diff --git a/tests/images/svg/polygon01.png b/tests/images/svg/polygon01.png new file mode 100644 index 0000000..60d4d91 Binary files /dev/null and b/tests/images/svg/polygon01.png differ diff --git a/tests/images/svg/polygon01.svg b/tests/images/svg/polygon01.svg new file mode 100644 index 0000000..900cf65 --- /dev/null +++ b/tests/images/svg/polygon01.svg @@ -0,0 +1,18 @@ + + + Example polygon01 - star and hexagon + + + + + + + + diff --git a/tests/images/svg/polyline01.png b/tests/images/svg/polyline01.png new file mode 100644 index 0000000..4b8d0c7 Binary files /dev/null and b/tests/images/svg/polyline01.png differ diff --git a/tests/images/svg/polyline01.svg b/tests/images/svg/polyline01.svg new file mode 100644 index 0000000..9ac958d --- /dev/null +++ b/tests/images/svg/polyline01.svg @@ -0,0 +1,19 @@ + + + Example polyline01 - increasingly larger bars + + + + + + + diff --git a/tests/images/svg/rect01.png b/tests/images/svg/rect01.png new file mode 100644 index 0000000..7fcfcbb Binary files /dev/null and b/tests/images/svg/rect01.png differ diff --git a/tests/images/svg/rect01.svg b/tests/images/svg/rect01.svg new file mode 100644 index 0000000..d61727e --- /dev/null +++ b/tests/images/svg/rect01.svg @@ -0,0 +1,13 @@ + + + Example rect01 - rectangle with sharp corners + + + + + + + diff --git a/tests/images/svg/rect02.png b/tests/images/svg/rect02.png new file mode 100644 index 0000000..ecf92f2 Binary files /dev/null and b/tests/images/svg/rect02.png differ diff --git a/tests/images/svg/rect02.svg b/tests/images/svg/rect02.svg new file mode 100644 index 0000000..b943ff4 --- /dev/null +++ b/tests/images/svg/rect02.svg @@ -0,0 +1,17 @@ + + + Example rect02 - rounded rectangles + + + + + + + + + + diff --git a/tests/test_svg.nim b/tests/test_svg.nim index 435fbae..9356964 100644 --- a/tests/test_svg.nim +++ b/tests/test_svg.nim @@ -1,6 +1,12 @@ import pixie/fileformats/svg, pixie, strformat const files = [ + "line01", + "polyline01", + "polygon01", + "rect01", + "rect02", + "circle01", "triangle01", "quad01", "Ghostscript_Tiger"