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 @@
+
+
+
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 @@
+
+
+
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 @@
+
+
+
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 @@
+
+
+
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 @@
+
+
+
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 @@
+
+
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"