cubic curve compute (#50)
* cubic curve compute * curve -> segments with error budget * dynamic precision on curves
|
@ -296,22 +296,34 @@ proc commandsToPolygons*(commands: seq[PathCommand]): seq[seq[Vec2]] =
|
||||||
polygon.add(at)
|
polygon.add(at)
|
||||||
polygon.add(to)
|
polygon.add(to)
|
||||||
|
|
||||||
proc getCurvePoint(points: seq[Vec2], t: float32): Vec2 =
|
proc drawCurve(at, ctrl1, ctrl2, to: Vec2) =
|
||||||
if points.len == 1:
|
|
||||||
return points[0]
|
|
||||||
else:
|
|
||||||
var newPoints = newSeq[Vec2](points.len - 1)
|
|
||||||
for i in 0 ..< newPoints.len:
|
|
||||||
newPoints[i] = points[i] * (1-t) + points[i + 1] * t
|
|
||||||
return getCurvePoint(newPoints, t)
|
|
||||||
|
|
||||||
proc drawCurve(points: seq[Vec2]) =
|
proc compute(at, ctrl1, ctrl2, to: Vec2, t: float32): Vec2 {.inline.} =
|
||||||
let n = 10
|
pow(1 - t, 3) * at +
|
||||||
var a = at
|
3 * pow(1 - t, 2) * t * ctrl1 +
|
||||||
for t in 1..n:
|
3 * (1 - t) * pow(t, 2) * ctrl2 +
|
||||||
var b = getCurvePoint(points, float32(t) / float32(n))
|
pow(t, 3) * to
|
||||||
drawLine(a, b)
|
|
||||||
a = b
|
var prev = at
|
||||||
|
|
||||||
|
proc discretize(i, steps: int) =
|
||||||
|
let
|
||||||
|
tPrev = (i - 1).float32 / steps.float32
|
||||||
|
t = i.float32 / steps.float32
|
||||||
|
next = compute(at, ctrl1, ctrl2, to, t)
|
||||||
|
halfway = compute(at, ctrl1, ctrl2, to, tPrev + (t - tPrev) / 2)
|
||||||
|
error = ((prev + next) / 2 - halfway).length
|
||||||
|
|
||||||
|
if error >= 0.25:
|
||||||
|
# Error too large, double precision for this step
|
||||||
|
discretize(i * 2 - 1, steps * 2)
|
||||||
|
discretize(i * 2, steps * 2)
|
||||||
|
else:
|
||||||
|
drawLine(prev, next)
|
||||||
|
prev = next
|
||||||
|
|
||||||
|
for i in 1 .. 2:
|
||||||
|
discretize(i, 2)
|
||||||
|
|
||||||
proc drawQuad(p0, p1, p2: Vec2) =
|
proc drawQuad(p0, p1, p2: Vec2) =
|
||||||
let devx = p0.x - 2.0 * p1.x + p2.x
|
let devx = p0.x - 2.0 * p1.x + p2.x
|
||||||
|
@ -393,7 +405,7 @@ proc commandsToPolygons*(commands: seq[PathCommand]): seq[seq[Vec2]] =
|
||||||
ctr2.y = command.numbers[3]
|
ctr2.y = command.numbers[3]
|
||||||
to.x = command.numbers[4]
|
to.x = command.numbers[4]
|
||||||
to.y = command.numbers[5]
|
to.y = command.numbers[5]
|
||||||
drawCurve(@[at, ctr, ctr2, to])
|
drawCurve(at, ctr, ctr2, to)
|
||||||
at = to
|
at = to
|
||||||
|
|
||||||
of Arc:
|
of Arc:
|
||||||
|
@ -486,7 +498,7 @@ proc commandsToPolygons*(commands: seq[PathCommand]): seq[seq[Vec2]] =
|
||||||
ctr2.y = at.y + command.numbers[3]
|
ctr2.y = at.y + command.numbers[3]
|
||||||
to.x = at.x + command.numbers[4]
|
to.x = at.x + command.numbers[4]
|
||||||
to.y = at.y + command.numbers[5]
|
to.y = at.y + command.numbers[5]
|
||||||
drawCurve(@[at, ctr, ctr2, to])
|
drawCurve(at, ctr, ctr2, to)
|
||||||
at = to
|
at = to
|
||||||
|
|
||||||
of RSCubic:
|
of RSCubic:
|
||||||
|
@ -496,7 +508,7 @@ proc commandsToPolygons*(commands: seq[PathCommand]): seq[seq[Vec2]] =
|
||||||
ctr2.y = at.y + command.numbers[1]
|
ctr2.y = at.y + command.numbers[1]
|
||||||
to.x = at.x + command.numbers[2]
|
to.x = at.x + command.numbers[2]
|
||||||
to.y = at.y + command.numbers[3]
|
to.y = at.y + command.numbers[3]
|
||||||
drawCurve(@[at, ctr, ctr2, to])
|
drawCurve(at, ctr, ctr2, to)
|
||||||
at = to
|
at = to
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
Before Width: | Height: | Size: 348 KiB After Width: | Height: | Size: 347 KiB |
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 13 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 24 KiB |
Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 33 KiB |