Don't return a new image when filling paths.

This commit is contained in:
treeform 2021-01-02 16:19:19 -08:00
parent b125dbabae
commit 5a33f0a3a2
19 changed files with 29 additions and 64 deletions

View file

@ -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* = "<?xml"
@ -111,13 +111,11 @@ proc draw(
d = node.attr("d")
ctx = decodeCtx(ctxStack[^1], node)
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)
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 & ".")

View file

@ -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.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

After

Width:  |  Height:  |  Size: 971 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 975 B

After

Width:  |  Height:  |  Size: 876 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2 KiB

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 981 B

After

Width:  |  Height:  |  Size: 851 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 358 B

After

Width:  |  Height:  |  Size: 354 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 810 B

After

Width:  |  Height:  |  Size: 779 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 714 B

After

Width:  |  Height:  |  Size: 706 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1 KiB

After

Width:  |  Height:  |  Size: 1,019 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 360 B

After

Width:  |  Height:  |  Size: 358 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 346 KiB

After

Width:  |  Height:  |  Size: 348 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 13 KiB

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 24 KiB

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 33 KiB

After

Width:  |  Height:  |  Size: 33 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4 KiB

After

Width:  |  Height:  |  Size: 4 KiB

View file

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