leak fix, iterative discretize, lengthSq error
This commit is contained in:
parent
92010d3b35
commit
5ea3c64968
2 changed files with 71 additions and 55 deletions
|
@ -45,7 +45,7 @@ type
|
||||||
startY, partitionHeight: uint32
|
startY, partitionHeight: uint32
|
||||||
|
|
||||||
const
|
const
|
||||||
epsilon: float32 = 0.0001 * PI ## Tiny value used for some computations.
|
epsilon = 0.0001 * PI ## Tiny value used for some computations.
|
||||||
defaultMiterLimit*: float32 = 4
|
defaultMiterLimit*: float32 = 4
|
||||||
|
|
||||||
when defined(release):
|
when defined(release):
|
||||||
|
@ -651,7 +651,7 @@ proc commandsToShapes(
|
||||||
prevCommandKind = Move
|
prevCommandKind = Move
|
||||||
prevCtrl, prevCtrl2: Vec2
|
prevCtrl, prevCtrl2: Vec2
|
||||||
|
|
||||||
let errorMargin = 0.2 / pixelScale
|
let errorMargin = pow(0.2.float32 / pixelScale, 2)
|
||||||
|
|
||||||
proc addSegment(shape: var seq[Vec2], at, to: Vec2) =
|
proc addSegment(shape: var seq[Vec2], at, to: Vec2) =
|
||||||
# Don't add any 0 length lines
|
# Don't add any 0 length lines
|
||||||
|
@ -669,27 +669,29 @@ proc commandsToShapes(
|
||||||
(1 - t) * 3 * pow(t, 2) * ctrl2 +
|
(1 - t) * 3 * pow(t, 2) * ctrl2 +
|
||||||
pow(t, 3) * to
|
pow(t, 3) * to
|
||||||
|
|
||||||
var prev = at
|
var
|
||||||
|
t: float32 # Where we are at on the curve from [0, 1]
|
||||||
proc discretize(shape: var seq[Vec2], i, steps: int) =
|
step = 1.float32 # How far we want to try to move along the curve
|
||||||
# Closure captures at, ctrl1, ctrl2, to and prev
|
prev = at
|
||||||
|
next = compute(at, ctrl1, ctrl2, to, t + step)
|
||||||
|
halfway = compute(at, ctrl1, ctrl2, to, t + step / 2)
|
||||||
|
while true:
|
||||||
let
|
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)
|
|
||||||
midpoint = (prev + next) / 2
|
midpoint = (prev + next) / 2
|
||||||
error = (midpoint - halfway).length
|
error = (midpoint - halfway).lengthSq
|
||||||
|
if error > errorMargin:
|
||||||
if error >= errorMargin:
|
next = halfway
|
||||||
# Error too large, double precision for this step
|
halfway = compute(at, ctrl1, ctrl2, to, t + step / 4)
|
||||||
shape.discretize(i * 2 - 1, steps * 2)
|
step /= 2
|
||||||
shape.discretize(i * 2, steps * 2)
|
|
||||||
else:
|
else:
|
||||||
shape.addSegment(prev, next)
|
shape.addSegment(prev, next)
|
||||||
|
t += step
|
||||||
|
if t == 1:
|
||||||
|
break
|
||||||
prev = next
|
prev = next
|
||||||
|
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
|
||||||
shape.discretize(1, 1)
|
next = compute(at, ctrl1, ctrl2, to, t + step)
|
||||||
|
halfway = compute(at, ctrl1, ctrl2, to, t + step / 2)
|
||||||
|
|
||||||
proc addQuadratic(shape: var seq[Vec2], at, ctrl, to: Vec2) =
|
proc addQuadratic(shape: var seq[Vec2], at, ctrl, to: Vec2) =
|
||||||
## Adds quadratic segments to shape.
|
## Adds quadratic segments to shape.
|
||||||
|
@ -698,27 +700,34 @@ proc commandsToShapes(
|
||||||
2 * (1 - t) * t * ctrl +
|
2 * (1 - t) * t * ctrl +
|
||||||
pow(t, 2) * to
|
pow(t, 2) * to
|
||||||
|
|
||||||
var prev = at
|
var
|
||||||
|
t: float32 # Where we are at on the curve from [0, 1]
|
||||||
proc discretize(shape: var seq[Vec2], i, steps: int) =
|
step = 1.float32 # How far we want to try to move along the curve
|
||||||
# Closure captures at, ctrl, to and prev
|
prev = at
|
||||||
|
next = compute(at, ctrl, to, t + step)
|
||||||
|
halfway = compute(at, ctrl, to, t + step / 2)
|
||||||
|
halfStepping = false
|
||||||
|
while true:
|
||||||
let
|
let
|
||||||
tPrev = (i - 1).float32 / steps.float32
|
|
||||||
t = i.float32 / steps.float32
|
|
||||||
next = compute(at, ctrl, to, t)
|
|
||||||
halfway = compute(at, ctrl, to, tPrev + (t - tPrev) / 2)
|
|
||||||
midpoint = (prev + next) / 2
|
midpoint = (prev + next) / 2
|
||||||
error = (midpoint - halfway).length
|
error = (midpoint - halfway).lengthSq
|
||||||
|
if error > errorMargin:
|
||||||
if error >= errorMargin:
|
next = halfway
|
||||||
# Error too large, double precision for this step
|
halfway = compute(at, ctrl, to, t + step / 4)
|
||||||
shape.discretize(i * 2 - 1, steps * 2)
|
step /= 2
|
||||||
shape.discretize(i * 2, steps * 2)
|
halfStepping = true
|
||||||
else:
|
else:
|
||||||
shape.addSegment(prev, next)
|
shape.addSegment(prev, next)
|
||||||
|
t += step
|
||||||
|
if t == 1:
|
||||||
|
break
|
||||||
prev = next
|
prev = next
|
||||||
|
if halfStepping:
|
||||||
shape.discretize(1, 1)
|
step = min(step, 1 - t)
|
||||||
|
else:
|
||||||
|
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
|
||||||
|
next = compute(at, ctrl, to, t + step)
|
||||||
|
halfway = compute(at, ctrl, to, t + step / 2)
|
||||||
|
|
||||||
proc addArc(
|
proc addArc(
|
||||||
shape: var seq[Vec2],
|
shape: var seq[Vec2],
|
||||||
|
@ -808,28 +817,37 @@ proc commandsToShapes(
|
||||||
result = vec2(cos(a) * arc.radii.x, sin(a) * arc.radii.y)
|
result = vec2(cos(a) * arc.radii.x, sin(a) * arc.radii.y)
|
||||||
result = arc.rotMat * result + arc.center
|
result = arc.rotMat * result + arc.center
|
||||||
|
|
||||||
var prev = at
|
let arc = endpointToCenterArcParams(at, radii, rotation, large, sweep, to)
|
||||||
|
|
||||||
proc discretize(shape: var seq[Vec2], arc: ArcParams, i, steps: int) =
|
var
|
||||||
|
t: float32 # Where we are at on the curve from [0, 1]
|
||||||
|
step = 1.float32 # How far we want to try to move along the curve
|
||||||
|
prev = at
|
||||||
|
while t != 1:
|
||||||
let
|
let
|
||||||
step = arc.delta / steps.float32
|
aPrev = arc.theta + arc.delta * t
|
||||||
aPrev = arc.theta + step * (i - 1).float32
|
a = arc.theta + arc.delta * (t + step)
|
||||||
a = arc.theta + step * i.float32
|
|
||||||
next = arc.compute(a)
|
next = arc.compute(a)
|
||||||
halfway = arc.compute(aPrev + (a - aPrev) / 2)
|
halfway = arc.compute(aPrev + (a - aPrev) / 2)
|
||||||
midpoint = (prev + next) / 2
|
midpoint = (prev + next) / 2
|
||||||
error = (midpoint - halfway).length
|
error = (midpoint - halfway).lengthSq
|
||||||
|
if error > errorMargin:
|
||||||
if error >= errorMargin:
|
let
|
||||||
# Error too large, try again with doubled precision
|
quarterway = arc.compute(aPrev + (a - aPrev) / 4)
|
||||||
shape.discretize(arc, i * 2 - 1, steps * 2)
|
midpoint = (prev + halfway) / 2
|
||||||
shape.discretize(arc, i * 2, steps * 2)
|
halfwayError = (midpoint - quarterway).lengthSq
|
||||||
|
if halfwayError < errorMargin:
|
||||||
|
shape.addSegment(prev, halfway)
|
||||||
|
prev = halfway
|
||||||
|
t += step / 2
|
||||||
|
step = min(step / 2, 1 - t) # Assume next steps hould be the same size
|
||||||
|
else:
|
||||||
|
step = step / 4 # We know a half-step is too big
|
||||||
else:
|
else:
|
||||||
shape.addSegment(prev, next)
|
shape.addSegment(prev, next)
|
||||||
prev = next
|
prev = next
|
||||||
|
t += step
|
||||||
let arc = endpointToCenterArcParams(at, radii, rotation, large, sweep, to)
|
step = min(step * 2, 1 - t) # Optimistically attempt larger steps
|
||||||
shape.discretize(arc, 1, 1)
|
|
||||||
|
|
||||||
for command in path.commands:
|
for command in path.commands:
|
||||||
if command.numbers.len != command.kind.parameterCount():
|
if command.numbers.len != command.kind.parameterCount():
|
||||||
|
@ -1214,7 +1232,7 @@ proc computeCoverage(
|
||||||
let
|
let
|
||||||
quality = if aa: 5 else: 1 # Must divide 255 cleanly (1, 3, 5, 15, 17, 51, 85)
|
quality = if aa: 5 else: 1 # Must divide 255 cleanly (1, 3, 5, 15, 17, 51, 85)
|
||||||
sampleCoverage = (255 div quality).uint8
|
sampleCoverage = (255 div quality).uint8
|
||||||
offset = 1 / quality.float32
|
offset = 1 / quality.float64
|
||||||
initialOffset = offset / 2 + epsilon
|
initialOffset = offset / 2 + epsilon
|
||||||
|
|
||||||
if aa: # Coverage is only used for anti-aliasing
|
if aa: # Coverage is only used for anti-aliasing
|
||||||
|
@ -1224,7 +1242,7 @@ proc computeCoverage(
|
||||||
let
|
let
|
||||||
partitionIndex = partitioning.getIndexForY(y)
|
partitionIndex = partitioning.getIndexForY(y)
|
||||||
partitionEntryCount = partitioning.partitions[partitionIndex].len
|
partitionEntryCount = partitioning.partitions[partitionIndex].len
|
||||||
var yLine = y.float32 + initialOffset - offset
|
var yLine = y.float64 + initialOffset - offset
|
||||||
for m in 0 ..< quality:
|
for m in 0 ..< quality:
|
||||||
yLine += offset
|
yLine += offset
|
||||||
numHits = 0
|
numHits = 0
|
||||||
|
@ -1813,7 +1831,6 @@ proc fillPath*(
|
||||||
var shapes = parseSomePath(path, true, transform.pixelScale())
|
var shapes = parseSomePath(path, true, transform.pixelScale())
|
||||||
shapes.transform(transform)
|
shapes.transform(transform)
|
||||||
var color = paint.color
|
var color = paint.color
|
||||||
if paint.opacity != 1:
|
|
||||||
color.a *= paint.opacity
|
color.a *= paint.opacity
|
||||||
image.fillShapes(shapes, color, windingRule, paint.blendMode)
|
image.fillShapes(shapes, color, windingRule, paint.blendMode)
|
||||||
return
|
return
|
||||||
|
@ -1897,7 +1914,6 @@ proc strokePath*(
|
||||||
)
|
)
|
||||||
strokeShapes.transform(transform)
|
strokeShapes.transform(transform)
|
||||||
var color = paint.color
|
var color = paint.color
|
||||||
if paint.opacity != 1:
|
|
||||||
color.a *= paint.opacity
|
color.a *= paint.opacity
|
||||||
image.fillShapes(strokeShapes, color, wrNonZero, paint.blendMode)
|
image.fillShapes(strokeShapes, color, wrNonZero, paint.blendMode)
|
||||||
return
|
return
|
||||||
|
|
|
@ -3,4 +3,4 @@ import benchy, pixie/fileformats/svg
|
||||||
let data = readFile("tests/fileformats/svg/Ghostscript_Tiger.svg")
|
let data = readFile("tests/fileformats/svg/Ghostscript_Tiger.svg")
|
||||||
|
|
||||||
timeIt "svg decode":
|
timeIt "svg decode":
|
||||||
keep decodeSvg(data)
|
discard decodeSvg(data)
|
||||||
|
|
Loading…
Reference in a new issue