commit
f627c56d8d
7 changed files with 446 additions and 144 deletions
|
@ -1,8 +1,8 @@
|
|||
import pixie/images, pixie/paths, pixie/common, pixie/blends,
|
||||
import pixie/images, pixie/masks, pixie/paths, pixie/common, pixie/blends,
|
||||
pixie/fileformats/bmp, pixie/fileformats/png, pixie/fileformats/jpg,
|
||||
pixie/fileformats/svg, flatty/binny, os
|
||||
|
||||
export images, paths, common, blends
|
||||
export images, masks, paths, common, blends
|
||||
|
||||
type
|
||||
FileFormat* = enum
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
import chroma, pixie/common, math, zippy, zippy/crc, flatty/binny, pixie/images
|
||||
import chroma, pixie/common, math, zippy, zippy/crc, flatty/binny,
|
||||
pixie/images, pixie/masks
|
||||
|
||||
# See http://www.libpng.org/pub/png/spec/1.2/PNG-Contents.html
|
||||
|
||||
|
@ -501,5 +502,16 @@ proc encodePng*(image: Image): string =
|
|||
image.width, image.height, 4, image.data[0].addr, image.data.len * 4
|
||||
))
|
||||
|
||||
proc encodePng*(mask: Mask): string =
|
||||
## Encodes the mask data into the PNG file format.
|
||||
if mask.data.len == 0:
|
||||
raise newException(
|
||||
PixieError,
|
||||
"Mask has no data (are height and width 0?)"
|
||||
)
|
||||
cast[string](encodePng(
|
||||
mask.width, mask.height, 1, mask.data[0].addr, mask.data.len
|
||||
))
|
||||
|
||||
when defined(release):
|
||||
{.pop.}
|
||||
|
|
67
src/pixie/masks.nim
Normal file
67
src/pixie/masks.nim
Normal file
|
@ -0,0 +1,67 @@
|
|||
import common, vmath
|
||||
|
||||
type
|
||||
Mask* = ref object
|
||||
## Mask object that holds mask opacity data.
|
||||
width*, height*: int
|
||||
data*: seq[uint8]
|
||||
|
||||
when defined(release):
|
||||
{.push checks: off.}
|
||||
|
||||
proc newMask*(width, height: int): Mask =
|
||||
## Creates a new mask with the parameter dimensions.
|
||||
if width <= 0 or height <= 0:
|
||||
raise newException(PixieError, "Mask width and height must be > 0")
|
||||
|
||||
result = Mask()
|
||||
result.width = width
|
||||
result.height = height
|
||||
result.data = newSeq[uint8](width * height)
|
||||
|
||||
proc wh*(mask: Mask): Vec2 {.inline.} =
|
||||
## Return with and height as a size vector.
|
||||
vec2(mask.width.float32, mask.height.float32)
|
||||
|
||||
proc copy*(mask: Mask): Mask =
|
||||
## Copies the image data into a new image.
|
||||
result = newMask(mask.width, mask.height)
|
||||
result.data = mask.data
|
||||
|
||||
proc `$`*(mask: Mask): string =
|
||||
## Prints the mask size.
|
||||
"<Mask " & $mask.width & "x" & $mask.height & ">"
|
||||
|
||||
proc inside*(mask: Mask, x, y: int): bool {.inline.} =
|
||||
## Returns true if (x, y) is inside the mask.
|
||||
x >= 0 and x < mask.width and y >= 0 and y < mask.height
|
||||
|
||||
proc dataIndex*(mask: Mask, x, y: int): int {.inline.} =
|
||||
mask.width * y + x
|
||||
|
||||
proc getValueUnsafe*(mask: Mask, x, y: int): uint8 {.inline.} =
|
||||
## Gets a color from (x, y) coordinates.
|
||||
## * No bounds checking *
|
||||
## Make sure that x, y are in bounds.
|
||||
## Failure in the assumptions will case unsafe memory reads.
|
||||
result = mask.data[mask.width * y + x]
|
||||
|
||||
proc `[]`*(mask: Mask, x, y: int): uint8 {.inline.} =
|
||||
## Gets a pixel at (x, y) or returns transparent black if outside of bounds.
|
||||
if mask.inside(x, y):
|
||||
return mask.getValueUnsafe(x, y)
|
||||
|
||||
proc setValueUnsafe*(mask: Mask, x, y: int, value: uint8) {.inline.} =
|
||||
## Sets a value from (x, y) coordinates.
|
||||
## * No bounds checking *
|
||||
## Make sure that x, y are in bounds.
|
||||
## Failure in the assumptions will case unsafe memory writes.
|
||||
mask.data[mask.dataIndex(x, y)] = value
|
||||
|
||||
proc `[]=`*(mask: Mask, x, y: int, value: uint8) {.inline.} =
|
||||
## Sets a pixel at (x, y) or does nothing if outside of bounds.
|
||||
if mask.inside(x, y):
|
||||
mask.setValueUnsafe(x, y, value)
|
||||
|
||||
when defined(release):
|
||||
{.pop.}
|
|
@ -1,4 +1,4 @@
|
|||
import common, strutils, vmath, images, chroma, bumpy, blends
|
||||
import common, strutils, vmath, images, masks, chroma, bumpy, blends
|
||||
|
||||
when defined(amd64) and not defined(pixieNoSimd):
|
||||
import nimsimd/sse2
|
||||
|
@ -767,13 +767,18 @@ proc computeBounds(seqs: varargs[seq[(Segment, int16)]]): Rect =
|
|||
result.w = xMax - xMin
|
||||
result.h = yMax - yMin
|
||||
|
||||
proc fillShapes(
|
||||
image: Image,
|
||||
shapes: seq[seq[Vec2]],
|
||||
color: ColorRGBA,
|
||||
windingRule: WindingRule
|
||||
) =
|
||||
var topHalf, bottomHalf, fullHeight: seq[(Segment, int16)]
|
||||
proc shouldFill(windingRule: WindingRule, count: int): bool {.inline.} =
|
||||
case windingRule:
|
||||
of wrNonZero:
|
||||
count != 0
|
||||
of wrEvenOdd:
|
||||
count mod 2 != 0
|
||||
|
||||
proc partitionSegments(shapes: seq[seq[Vec2]], middle: int): tuple[
|
||||
topHalf: seq[(Segment, int16)],
|
||||
bottomHalf: seq[(Segment, int16)],
|
||||
fullHeight: seq[(Segment, int16)]
|
||||
] =
|
||||
for shape in shapes:
|
||||
for segment in shape.segments:
|
||||
if segment.at.y == segment.to.y: # Skip horizontal
|
||||
|
@ -784,12 +789,107 @@ proc fillShapes(
|
|||
if segment.at.y > segment.to.y:
|
||||
swap(segment.at, segment.to)
|
||||
winding = -1
|
||||
if ceil(segment.to.y).int < image.height div 2:
|
||||
topHalf.add((segment, winding))
|
||||
elif segment.at.y.int >= image.height div 2:
|
||||
bottomHalf.add((segment, winding))
|
||||
if ceil(segment.to.y).int < middle:
|
||||
result.topHalf.add((segment, winding))
|
||||
elif segment.at.y.int >= middle:
|
||||
result.bottomHalf.add((segment, winding))
|
||||
else:
|
||||
fullHeight.add((segment, winding))
|
||||
result.fullHeight.add((segment, winding))
|
||||
|
||||
proc computeCoverages(
|
||||
coverages: var seq[uint8],
|
||||
hits: var seq[(float32, int16)],
|
||||
size: Vec2,
|
||||
y: int,
|
||||
topHalf, bottomHalf, fullHeight: seq[(Segment, int16)],
|
||||
windingRule: WindingRule
|
||||
) =
|
||||
const
|
||||
quality = 5 # Must divide 255 cleanly
|
||||
sampleCoverage = 255.uint8 div quality
|
||||
ep = 0.0001 * PI
|
||||
offset = 1 / quality.float32
|
||||
initialOffset = offset / 2
|
||||
|
||||
proc intersects(
|
||||
scanline: Line,
|
||||
segment: Segment,
|
||||
winding: int16,
|
||||
hits: var seq[(float32, int16)],
|
||||
numHits: var int
|
||||
) {.inline.} =
|
||||
if segment.at.y <= scanline.a.y and segment.to.y >= scanline.a.y:
|
||||
var at: Vec2
|
||||
if scanline.intersects(segment, at):# and segment.to != at:
|
||||
if numHits == hits.len:
|
||||
hits.setLen(hits.len * 2)
|
||||
hits[numHits] = (at.x.clamp(0, scanline.b.x), winding)
|
||||
inc numHits
|
||||
|
||||
var numHits: int
|
||||
|
||||
# Do scanlines for this row
|
||||
for m in 0 ..< quality:
|
||||
let
|
||||
yLine = y.float32 + initialOffset + offset * m.float32 + ep
|
||||
scanline = Line(a: vec2(0, yLine), b: vec2(size.x, yLine))
|
||||
numHits = 0
|
||||
if y < size.y.int div 2:
|
||||
for (segment, winding) in topHalf:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
else:
|
||||
for (segment, winding) in bottomHalf:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
for (segment, winding) in fullHeight:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
|
||||
quickSort(hits, 0, numHits - 1)
|
||||
|
||||
var
|
||||
x: float32
|
||||
count: int
|
||||
for i in 0 ..< numHits:
|
||||
let (at, winding) = hits[i]
|
||||
|
||||
var
|
||||
fillStart = x.int
|
||||
leftCover = if at.int - x.int > 0: trunc(x) + 1 - x else: at - x
|
||||
if leftCover != 0:
|
||||
inc fillStart
|
||||
if shouldFill(windingRule, count):
|
||||
coverages[x.int] += (leftCover * sampleCoverage.float32).uint8
|
||||
|
||||
if at.int - x.int > 0:
|
||||
let rightCover = at - trunc(at)
|
||||
if rightCover > 0 and shouldFill(windingRule, count):
|
||||
coverages[at.int] += (rightCover * sampleCoverage.float32).uint8
|
||||
|
||||
let fillLen = at.int - fillStart
|
||||
if fillLen > 0 and shouldFill(windingRule, count):
|
||||
var i = fillStart
|
||||
when defined(amd64) and not defined(pixieNoSimd):
|
||||
let vSampleCoverage = mm_set1_epi8(cast[int8](sampleCoverage))
|
||||
for j in countup(i, fillStart + fillLen - 16, 16):
|
||||
let current = mm_loadu_si128(coverages[j].addr)
|
||||
mm_storeu_si128(
|
||||
coverages[j].addr,
|
||||
mm_add_epi8(current, vSampleCoverage)
|
||||
)
|
||||
i += 16
|
||||
for j in i ..< fillStart + fillLen:
|
||||
coverages[j] += sampleCoverage
|
||||
|
||||
count += winding
|
||||
x = at
|
||||
|
||||
proc fillShapes(
|
||||
image: Image,
|
||||
shapes: seq[seq[Vec2]],
|
||||
color: ColorRGBA,
|
||||
windingRule: WindingRule
|
||||
) =
|
||||
let (topHalf, bottomHalf, fullHeight) =
|
||||
partitionSegments(shapes, image.height div 2)
|
||||
|
||||
# Figure out the total bounds of all the shapes,
|
||||
# rasterize only within the total bounds
|
||||
|
@ -799,105 +899,29 @@ proc fillShapes(
|
|||
startY = max(0, bounds.y.int)
|
||||
stopY = min(image.height, (bounds.y + bounds.h).int)
|
||||
|
||||
const
|
||||
quality = 5 # Must divide 255 cleanly
|
||||
sampleCoverage = 255.uint8 div quality
|
||||
ep = 0.0001 * PI
|
||||
offset = 1 / quality.float32
|
||||
initialOffset = offset / 2
|
||||
|
||||
var
|
||||
coverages = newSeq[uint8](image.width)
|
||||
hits = newSeq[(float32, int16)](4)
|
||||
numHits: int
|
||||
|
||||
for y in startY ..< stopY:
|
||||
# Reset buffer for this row
|
||||
zeroMem(coverages[0].addr, coverages.len)
|
||||
|
||||
proc intersects(
|
||||
scanline: Line,
|
||||
segment: Segment,
|
||||
winding: int16,
|
||||
hits: var seq[(float32, int16)],
|
||||
numHits: var int
|
||||
) {.inline.} =
|
||||
if segment.at.y <= scanline.a.y and segment.to.y >= scanline.a.y:
|
||||
var at: Vec2
|
||||
if scanline.intersects(segment, at):# and segment.to != at:
|
||||
if numHits == hits.len:
|
||||
hits.setLen(hits.len * 2)
|
||||
hits[numHits] = (at.x.clamp(0, scanline.b.x), winding)
|
||||
inc numHits
|
||||
|
||||
# Do scanlines for this row
|
||||
for m in 0 ..< quality:
|
||||
let
|
||||
yLine = y.float32 + initialOffset + offset * m.float32 + ep
|
||||
scanline = Line(a: vec2(0, yLine), b: vec2(image.width.float32, yLine))
|
||||
numHits = 0
|
||||
if y < image.height div 2:
|
||||
for (segment, winding) in topHalf:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
else:
|
||||
for (segment, winding) in bottomHalf:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
for (segment, winding) in fullHeight:
|
||||
scanline.intersects(segment, winding, hits, numHits)
|
||||
|
||||
quickSort(hits, 0, numHits - 1)
|
||||
|
||||
proc shouldFill(windingRule: WindingRule, count: int): bool {.inline.} =
|
||||
case windingRule:
|
||||
of wrNonZero:
|
||||
count != 0
|
||||
of wrEvenOdd:
|
||||
count mod 2 != 0
|
||||
|
||||
var
|
||||
x: float32
|
||||
count: int
|
||||
for i in 0 ..< numHits:
|
||||
let (at, winding) = hits[i]
|
||||
|
||||
var
|
||||
fillStart = x.int
|
||||
leftCover = if at.int - x.int > 0: trunc(x) + 1 - x else: at - x
|
||||
if leftCover != 0:
|
||||
inc fillStart
|
||||
if shouldFill(windingRule, count):
|
||||
coverages[x.int] += (leftCover * sampleCoverage.float32).uint8
|
||||
|
||||
if at.int - x.int > 0:
|
||||
let rightCover = at - trunc(at)
|
||||
if rightCover > 0 and shouldFill(windingRule, count):
|
||||
coverages[at.int] += (rightCover * sampleCoverage.float32).uint8
|
||||
|
||||
let fillLen = at.int - fillStart
|
||||
if fillLen > 0 and shouldFill(windingRule, count):
|
||||
var i = fillStart
|
||||
when defined(amd64) and not defined(pixieNoSimd):
|
||||
let vSampleCoverage = mm_set1_epi8(cast[int8](sampleCoverage))
|
||||
for j in countup(i, fillStart + fillLen - 16, 16):
|
||||
let current = mm_loadu_si128(coverages[j].addr)
|
||||
mm_storeu_si128(
|
||||
coverages[j].addr,
|
||||
mm_add_epi8(current, vSampleCoverage)
|
||||
)
|
||||
i += 16
|
||||
for j in i ..< fillStart + fillLen:
|
||||
coverages[j] += sampleCoverage
|
||||
|
||||
count += winding
|
||||
x = at
|
||||
computeCoverages(
|
||||
coverages,
|
||||
hits,
|
||||
image.wh,
|
||||
y,
|
||||
topHalf, bottomHalf, fullHeight,
|
||||
windingRule
|
||||
)
|
||||
|
||||
# Apply the coverage and blend
|
||||
var x = startX
|
||||
when defined(amd64) and not defined(pixieNoSimd):
|
||||
# When supported, SIMD blend as much as possible
|
||||
|
||||
let
|
||||
coverageMask1 = cast[M128i]([0xffffffff, 0, 0, 0]) # First 32 bits
|
||||
coverageMask1 = cast[M128i]([uint32.high, 0, 0, 0]) # First 32 bits
|
||||
coverageMask2 = mm_set1_epi32(cast[int32](0x000000ff)) # Only `r`
|
||||
oddMask = mm_set1_epi16(cast[int16](0xff00))
|
||||
div255 = mm_set1_epi16(cast[int16](0x8081))
|
||||
|
@ -982,6 +1006,148 @@ proc fillShapes(
|
|||
image.setRgbaUnsafe(x, y, blendNormalPremultiplied(backdrop, source))
|
||||
inc x
|
||||
|
||||
proc fillShapes(
|
||||
mask: Mask,
|
||||
shapes: seq[seq[Vec2]],
|
||||
windingRule: WindingRule
|
||||
) =
|
||||
let (topHalf, bottomHalf, fullHeight) =
|
||||
partitionSegments(shapes, mask.height div 2)
|
||||
|
||||
# Figure out the total bounds of all the shapes,
|
||||
# rasterize only within the total bounds
|
||||
let
|
||||
bounds = computeBounds(topHalf, bottomHalf, fullHeight)
|
||||
startX = max(0, bounds.x.int)
|
||||
startY = max(0, bounds.y.int)
|
||||
stopY = min(mask.height, (bounds.y + bounds.h).int)
|
||||
|
||||
var
|
||||
coverages = newSeq[uint8](mask.width)
|
||||
hits = newSeq[(float32, int16)](4)
|
||||
|
||||
for y in startY ..< stopY:
|
||||
# Reset buffer for this row
|
||||
zeroMem(coverages[0].addr, coverages.len)
|
||||
|
||||
computeCoverages(
|
||||
coverages,
|
||||
hits,
|
||||
mask.wh,
|
||||
y,
|
||||
topHalf, bottomHalf, fullHeight,
|
||||
windingRule
|
||||
)
|
||||
|
||||
# Apply the coverage and blend
|
||||
var x = startX
|
||||
when defined(amd64) and not defined(pixieNoSimd):
|
||||
# When supported, SIMD blend as much as possible
|
||||
let
|
||||
oddMask = mm_set1_epi16(cast[int16](0xff00))
|
||||
v255high = mm_set1_epi16(cast[int16](255.uint16 shl 8))
|
||||
div255 = mm_set1_epi16(cast[int16](0x8081))
|
||||
|
||||
for _ in countup(x, coverages.len - 16, 16):
|
||||
var coverage = mm_loadu_si128(coverages[x].addr)
|
||||
|
||||
let eqZero = mm_cmpeq_epi16(coverage, mm_setzero_si128())
|
||||
if mm_movemask_epi8(eqZero) != 0xffff:
|
||||
# If the coverages are not all zero
|
||||
var
|
||||
coverageEven = mm_slli_epi16(mm_andnot_si128(oddMask, coverage), 8)
|
||||
coverageOdd = mm_and_si128(coverage, oddMask)
|
||||
|
||||
let
|
||||
evenK = mm_sub_epi16(v255high, coverageEven)
|
||||
oddK = mm_sub_epi16(v255high, coverageOdd)
|
||||
|
||||
var
|
||||
backdrop = mm_loadu_si128(mask.data[mask.dataIndex(x, y)].addr)
|
||||
backdropEven = mm_slli_epi16(mm_andnot_si128(oddMask, backdrop), 8)
|
||||
backdropOdd = mm_and_si128(backdrop, oddMask)
|
||||
|
||||
# backdrop * k
|
||||
backdropEven = mm_mulhi_epu16(backdropEven, evenK)
|
||||
backdropOdd = mm_mulhi_epu16(backdropOdd, oddK)
|
||||
|
||||
# div 255
|
||||
backdropEven = mm_srli_epi16(mm_mulhi_epu16(backdropEven, div255), 7)
|
||||
backdropOdd = mm_srli_epi16(mm_mulhi_epu16(backdropOdd, div255), 7)
|
||||
|
||||
# Shift from high to low bits
|
||||
coverageEven = mm_srli_epi16(coverageEven, 8)
|
||||
coverageOdd = mm_srli_epi16(coverageOdd, 8)
|
||||
|
||||
var
|
||||
blendedEven = mm_add_epi16(coverageEven, backdropEven)
|
||||
blendedOdd = mm_add_epi16(coverageOdd, backdropOdd)
|
||||
|
||||
blendedOdd = mm_slli_epi16(blendedOdd, 8)
|
||||
|
||||
mm_storeu_si128(
|
||||
mask.data[mask.dataIndex(x, y)].addr,
|
||||
mm_or_si128(blendedEven, blendedOdd)
|
||||
)
|
||||
|
||||
x += 16
|
||||
|
||||
while x < mask.width:
|
||||
if x + 8 <= coverages.len:
|
||||
let peeked = cast[ptr uint64](coverages[x].addr)[]
|
||||
if peeked == 0:
|
||||
x += 8
|
||||
continue
|
||||
let coverage = coverages[x]
|
||||
if coverage != 0:
|
||||
let
|
||||
backdrop = mask.getValueUnsafe(x, y)
|
||||
blended =
|
||||
coverage + ((backdrop.uint32 * (255 - coverage)) div 255).uint8
|
||||
mask.setValueUnsafe(x, y, blended)
|
||||
inc x
|
||||
|
||||
proc strokeShapes(
|
||||
shapes: seq[seq[Vec2]],
|
||||
strokeWidth: float32,
|
||||
windingRule: WindingRule
|
||||
): seq[seq[Vec2]] =
|
||||
if strokeWidth == 0:
|
||||
return
|
||||
|
||||
let
|
||||
widthLeft = strokeWidth / 2
|
||||
widthRight = strokeWidth / 2
|
||||
|
||||
for shape in shapes:
|
||||
var
|
||||
strokeShape: seq[Vec2]
|
||||
back: seq[Vec2]
|
||||
for segment in shape.segments:
|
||||
let
|
||||
tangent = (segment.at - segment.to).normalize()
|
||||
normal = vec2(-tangent.y, tangent.x)
|
||||
left = segment(
|
||||
segment.at - normal * widthLeft,
|
||||
segment.to - normal * widthLeft
|
||||
)
|
||||
right = segment(
|
||||
segment.at + normal * widthRight,
|
||||
segment.to + normal * widthRight
|
||||
)
|
||||
|
||||
strokeShape.add([right.at, right.to])
|
||||
back.add([left.at, left.to])
|
||||
|
||||
# Add the back side reversed
|
||||
for i in 1 .. back.len:
|
||||
strokeShape.add(back[^i])
|
||||
|
||||
strokeShape.add(strokeShape[0])
|
||||
|
||||
if strokeShape.len > 0:
|
||||
result.add(strokeShape)
|
||||
|
||||
proc parseSomePath(path: SomePath): seq[seq[Vec2]] {.inline.} =
|
||||
when type(path) is string:
|
||||
parsePath(path).commandsToShapes()
|
||||
|
@ -1024,47 +1190,36 @@ proc fillPath*(
|
|||
segment = mat * segment
|
||||
image.fillShapes(shapes, color, windingRule)
|
||||
|
||||
proc strokeShapes(
|
||||
shapes: seq[seq[Vec2]],
|
||||
color: ColorRGBA,
|
||||
strokeWidth: float32,
|
||||
windingRule: WindingRule
|
||||
): seq[seq[Vec2]] =
|
||||
if strokeWidth == 0:
|
||||
return
|
||||
proc fillPath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
windingRule = wrNonZero
|
||||
) {.inline.} =
|
||||
mask.fillShapes(parseSomePath(path), windingRule)
|
||||
|
||||
let
|
||||
widthLeft = strokeWidth / 2
|
||||
widthRight = strokeWidth / 2
|
||||
proc fillPath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
pos: Vec2,
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
var shapes = parseSomePath(path)
|
||||
for shape in shapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment += pos
|
||||
mask.fillShapes(shapes, color, windingRule)
|
||||
|
||||
for shape in shapes:
|
||||
var
|
||||
strokeShape: seq[Vec2]
|
||||
back: seq[Vec2]
|
||||
for segment in shape.segments:
|
||||
let
|
||||
tangent = (segment.at - segment.to).normalize()
|
||||
normal = vec2(-tangent.y, tangent.x)
|
||||
left = segment(
|
||||
segment.at - normal * widthLeft,
|
||||
segment.to - normal * widthLeft
|
||||
)
|
||||
right = segment(
|
||||
segment.at + normal * widthRight,
|
||||
segment.to + normal * widthRight
|
||||
)
|
||||
|
||||
strokeShape.add([right.at, right.to])
|
||||
back.add([left.at, left.to])
|
||||
|
||||
# Add the back side reversed
|
||||
for i in 1 .. back.len:
|
||||
strokeShape.add(back[^i])
|
||||
|
||||
strokeShape.add(strokeShape[0])
|
||||
|
||||
if strokeShape.len > 0:
|
||||
result.add(strokeShape)
|
||||
proc fillPath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
mat: Mat3,
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
var shapes = parseSomePath(path)
|
||||
for shape in shapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment = mat * segment
|
||||
mask.fillShapes(shapes, windingRule)
|
||||
|
||||
proc strokePath*(
|
||||
image: Image,
|
||||
|
@ -1075,7 +1230,6 @@ proc strokePath*(
|
|||
) =
|
||||
let strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
color,
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
|
@ -1091,7 +1245,6 @@ proc strokePath*(
|
|||
) =
|
||||
var strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
color,
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
|
@ -1110,7 +1263,6 @@ proc strokePath*(
|
|||
) =
|
||||
var strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
color,
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
|
@ -1119,5 +1271,52 @@ proc strokePath*(
|
|||
segment = mat * segment
|
||||
image.fillShapes(strokeShapes, color, windingRule)
|
||||
|
||||
proc strokePath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
strokeWidth = 1.0,
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
let strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
mask.fillShapes(strokeShapes, windingRule)
|
||||
|
||||
proc strokePath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
strokeWidth = 1.0,
|
||||
pos: Vec2,
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
var strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
for shape in strokeShapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment += pos
|
||||
mask.fillShapes(strokeShapes, windingRule)
|
||||
|
||||
proc strokePath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
strokeWidth = 1.0,
|
||||
mat: Mat3,
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
var strokeShapes = strokeShapes(
|
||||
parseSomePath(path),
|
||||
strokeWidth,
|
||||
windingRule
|
||||
)
|
||||
for shape in strokeShapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment = mat * segment
|
||||
mask.fillShapes(strokeShapes, windingRule)
|
||||
|
||||
when defined(release):
|
||||
{.pop.}
|
||||
|
|
BIN
tests/images/paths/pathRectangleMask.png
Normal file
BIN
tests/images/paths/pathRectangleMask.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 180 B |
BIN
tests/images/paths/pathRoundRectMask.png
Normal file
BIN
tests/images/paths/pathRoundRectMask.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 488 B |
|
@ -1,4 +1,4 @@
|
|||
import pixie, chroma
|
||||
import pixie, chroma, pixie/fileformats/png
|
||||
|
||||
block:
|
||||
let pathStr = """
|
||||
|
@ -178,3 +178,27 @@ block:
|
|||
image.fillPath(path, rgba(255, 0, 0, 255))
|
||||
image.toStraightAlpha()
|
||||
image.writeFile("tests/images/paths/pathRoundRect.png")
|
||||
|
||||
block:
|
||||
let
|
||||
mask = newMask(100, 100)
|
||||
pathStr = "M 10 10 H 90 V 90 H 10 L 10 10"
|
||||
mask.fillPath(pathStr)
|
||||
writeFile("tests/images/paths/pathRectangleMask.png", mask.encodePng())
|
||||
|
||||
block:
|
||||
let
|
||||
mask = newMask(100, 100)
|
||||
r = 10.0
|
||||
x = 10.0
|
||||
y = 10.0
|
||||
h = 80.0
|
||||
w = 80.0
|
||||
var path: Path
|
||||
path.moveTo(x + r, y)
|
||||
path.arcTo(x + w, y, x + w, y + h, r)
|
||||
path.arcTo(x + w, y + h, x, y + h, r)
|
||||
path.arcTo(x, y + h, x, y, r)
|
||||
path.arcTo(x, y, x + w, y, r)
|
||||
mask.fillPath(path)
|
||||
writeFile("tests/images/paths/pathRoundRectMask.png", mask.encodePng())
|
||||
|
|
Loading…
Reference in a new issue