use trapezoid method more
This commit is contained in:
parent
af84081817
commit
0abafbca6f
1 changed files with 235 additions and 161 deletions
|
@ -1118,6 +1118,15 @@ proc initPartitionEntry(segment: Segment, winding: int16): PartitionEntry =
|
||||||
result.m = (segment.at.y - segment.to.y) / d
|
result.m = (segment.at.y - segment.to.y) / d
|
||||||
result.b = segment.at.y - result.m * segment.at.x
|
result.b = segment.at.y - result.m * segment.at.x
|
||||||
|
|
||||||
|
proc solveX(entry: PartitionEntry, y: float32): float32 {.inline.} =
|
||||||
|
if entry.m == 0:
|
||||||
|
entry.b
|
||||||
|
else:
|
||||||
|
(y - entry.b) / entry.m
|
||||||
|
|
||||||
|
proc solveY(entry: PartitionEntry, x: float32): float32 {.inline.} =
|
||||||
|
entry.m * x + entry.b
|
||||||
|
|
||||||
proc requiresAntiAliasing(segment: Segment): bool {.inline.} =
|
proc requiresAntiAliasing(segment: Segment): bool {.inline.} =
|
||||||
## Returns true if the segment requires antialiasing.
|
## Returns true if the segment requires antialiasing.
|
||||||
|
|
||||||
|
@ -1321,16 +1330,16 @@ proc computeCoverage(
|
||||||
coverages: ptr UncheckedArray[uint8],
|
coverages: ptr UncheckedArray[uint8],
|
||||||
hits: var seq[(Fixed32, int16)],
|
hits: var seq[(Fixed32, int16)],
|
||||||
numHits: var int,
|
numHits: var int,
|
||||||
aa: var bool,
|
|
||||||
width: int,
|
width: int,
|
||||||
y, startX: int,
|
y, startX: int,
|
||||||
partitions: var seq[Partition],
|
partitions: var seq[Partition],
|
||||||
partitionIndex: int,
|
partitionIndex: int,
|
||||||
|
entryIndices: seq[int],
|
||||||
|
numEntryIndices: int,
|
||||||
windingRule: WindingRule
|
windingRule: WindingRule
|
||||||
) {.inline.} =
|
) {.inline.} =
|
||||||
aa = partitions[partitionIndex].requiresAntiAliasing
|
|
||||||
|
|
||||||
let
|
let
|
||||||
|
aa = partitions[partitionIndex].requiresAntiAliasing
|
||||||
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.float32
|
||||||
|
@ -1340,7 +1349,10 @@ proc computeCoverage(
|
||||||
for m in 0 ..< quality:
|
for m in 0 ..< quality:
|
||||||
yLine += offset
|
yLine += offset
|
||||||
numHits = 0
|
numHits = 0
|
||||||
for entry in partitions[partitionIndex].entries.mitems:
|
for i in 0 ..< numEntryIndices:
|
||||||
|
let
|
||||||
|
entryIndex = entryIndices[i]
|
||||||
|
entry = partitions[partitionIndex].entries[entryIndex].addr
|
||||||
if entry.segment.at.y <= yLine and entry.segment.to.y >= yLine:
|
if entry.segment.at.y <= yLine and entry.segment.to.y >= yLine:
|
||||||
let x =
|
let x =
|
||||||
if entry.m == 0:
|
if entry.m == 0:
|
||||||
|
@ -1429,8 +1441,8 @@ proc fillCoverage(
|
||||||
|
|
||||||
proc source(colorVec, coverageVec: M128i): M128i {.inline.} =
|
proc source(colorVec, coverageVec: M128i): M128i {.inline.} =
|
||||||
let
|
let
|
||||||
oddMask = mm_set1_epi16(cast[int16](0xff00))
|
oddMask = mm_set1_epi16(0xff00)
|
||||||
div255 = mm_set1_epi16(cast[int16](0x8081))
|
div255 = mm_set1_epi16(0x8081)
|
||||||
|
|
||||||
var unpacked = unpackAlphaValues(coverageVec)
|
var unpacked = unpackAlphaValues(coverageVec)
|
||||||
unpacked = mm_or_si128(unpacked, mm_srli_epi32(unpacked, 16))
|
unpacked = mm_or_si128(unpacked, mm_srli_epi32(unpacked, 16))
|
||||||
|
@ -1838,7 +1850,7 @@ proc fillHits(
|
||||||
|
|
||||||
proc fillShapes(
|
proc fillShapes(
|
||||||
image: Image,
|
image: Image,
|
||||||
shapes: var seq[Polygon],
|
shapes: seq[Polygon],
|
||||||
color: SomeColor,
|
color: SomeColor,
|
||||||
windingRule: WindingRule,
|
windingRule: WindingRule,
|
||||||
blendMode: BlendMode
|
blendMode: BlendMode
|
||||||
|
@ -1869,10 +1881,10 @@ proc fillShapes(
|
||||||
partitionIndex: int
|
partitionIndex: int
|
||||||
entryIndices = newSeq[int](partitions.maxEntryCount)
|
entryIndices = newSeq[int](partitions.maxEntryCount)
|
||||||
numEntryIndices: int
|
numEntryIndices: int
|
||||||
|
trapezoidSegments = newSeq[Segment](entryIndices.len)
|
||||||
coverages = newSeq[uint8](pathWidth)
|
coverages = newSeq[uint8](pathWidth)
|
||||||
hits = newSeq[(Fixed32, int16)](entryIndices.len)
|
hits = newSeq[(Fixed32, int16)](entryIndices.len)
|
||||||
numHits: int
|
numHits: int
|
||||||
aa: bool
|
|
||||||
|
|
||||||
var y = startY
|
var y = startY
|
||||||
while y < pathHeight:
|
while y < pathHeight:
|
||||||
|
@ -1880,18 +1892,19 @@ proc fillShapes(
|
||||||
inc partitionIndex
|
inc partitionIndex
|
||||||
|
|
||||||
let
|
let
|
||||||
partitionTop = partitions[partitionIndex].top
|
partition = partitions[partitionIndex].addr
|
||||||
partitionBottom = partitions[partitionIndex].bottom
|
partitionTop = partition.top
|
||||||
|
partitionBottom = partition.bottom
|
||||||
partitionHeight = partitionBottom - partitionTop
|
partitionHeight = partitionBottom - partitionTop
|
||||||
if partitionHeight == 0:
|
if partitionHeight == 0:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if partitions[partitionIndex].twoNonintersectingSpanningSegments:
|
if partition.twoNonintersectingSpanningSegments:
|
||||||
if not partitions[partitionIndex].requiresAntiAliasing:
|
if not partition.requiresAntiAliasing:
|
||||||
# No AA required, must be 2 vertical pixel-aligned lines
|
# No AA required, must be 2 vertical pixel-aligned lines
|
||||||
let
|
let
|
||||||
left = partitions[partitionIndex].entries[0].segment.at.x.int
|
left = partition.entries[0].segment.at.x.int
|
||||||
right = partitions[partitionIndex].entries[1].segment.at.x.int
|
right = partition.entries[1].segment.at.x.int
|
||||||
minX = left.clamp(0, image.width)
|
minX = left.clamp(0, image.width)
|
||||||
maxX = right.clamp(0, image.width)
|
maxX = right.clamp(0, image.width)
|
||||||
skipBlending =
|
skipBlending =
|
||||||
|
@ -1912,182 +1925,216 @@ proc fillShapes(
|
||||||
y += partitionHeight
|
y += partitionHeight
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
let
|
||||||
|
scanTop = y.float32
|
||||||
|
scanBottom = (y + 1).float32
|
||||||
|
|
||||||
var allEntriesInScanlineSpanIt = true
|
var allEntriesInScanlineSpanIt = true
|
||||||
numEntryIndices = 0
|
numEntryIndices = 0
|
||||||
|
if partition.twoNonintersectingSpanningSegments:
|
||||||
if partitions[partitionIndex].twoNonintersectingSpanningSegments:
|
|
||||||
numEntryIndices = 2
|
numEntryIndices = 2
|
||||||
entryIndices[0] = 0
|
entryIndices[0] = 0
|
||||||
entryIndices[1] = 1
|
entryIndices[1] = 1
|
||||||
else:
|
else:
|
||||||
for i in 0 ..< partitions[partitionIndex].entries.len:
|
for i in 0 ..< partition.entries.len:
|
||||||
if partitions[partitionIndex].entries[i].segment.to.y < y.float32 or
|
if partition.entries[i].segment.to.y <= scanTop or
|
||||||
partitions[partitionIndex].entries[i].segment.at.y >= (y + 1).float32:
|
partition.entries[i].segment.at.y >= scanBottom:
|
||||||
continue
|
continue
|
||||||
if partitions[partitionIndex].entries[i].segment.at.y > y.float32 or
|
if partition.entries[i].segment.at.y > scanTop or
|
||||||
partitions[partitionIndex].entries[i].segment.to.y < (y + 1).float32:
|
partition.entries[i].segment.to.y < scanBottom:
|
||||||
allEntriesInScanlineSpanIt = false
|
allEntriesInScanlineSpanIt = false
|
||||||
break
|
|
||||||
entryIndices[numEntryIndices] = i
|
entryIndices[numEntryIndices] = i
|
||||||
inc numEntryIndices
|
inc numEntryIndices
|
||||||
|
|
||||||
if allEntriesInScanlineSpanIt and numEntryIndices == 2:
|
if allEntriesInScanlineSpanIt and numEntryIndices mod 2 == 0:
|
||||||
var
|
for i in 0 ..< numEntryIndices:
|
||||||
left = partitions[partitionIndex].entries[entryIndices[0]]
|
let index = entryIndices[i]
|
||||||
right = partitions[partitionIndex].entries[entryIndices[1]]
|
trapezoidSegments[index].at.y = scanTop
|
||||||
block:
|
trapezoidSegments[index].to.y = scanBottom
|
||||||
# Ensure left is actually on the left
|
trapezoidSegments[index].at.x =
|
||||||
|
partition.entries[index].solveX(scanTop)
|
||||||
|
trapezoidSegments[index].to.x =
|
||||||
|
partition.entries[index].solveX(scanBottom)
|
||||||
|
|
||||||
|
# Sort the segments by midpoint. If they intersect this will be wrong
|
||||||
|
# but it will get caught when we check partial coverage overlap and we
|
||||||
|
# won't take the shortcut.
|
||||||
|
|
||||||
|
var noEntriesInScanlineOverlap = true
|
||||||
|
|
||||||
|
proc midpointX(segment: Segment): float32 {.inline.} =
|
||||||
|
(segment.at.x + segment.to.x) * 0.5
|
||||||
|
|
||||||
|
for i in 1 ..< numEntryIndices:
|
||||||
|
var
|
||||||
|
j = i - 1
|
||||||
|
k = i
|
||||||
|
while j >= 0 and
|
||||||
|
trapezoidSegments[entryIndices[j]].midpointX >
|
||||||
|
trapezoidSegments[entryIndices[k]].midpointX:
|
||||||
|
swap(entryIndices[j + 1], entryIndices[j])
|
||||||
|
dec j
|
||||||
|
dec k
|
||||||
|
|
||||||
|
# Only take this shortcut if the partial coverage areas on the
|
||||||
|
# left and the right do not overlap
|
||||||
|
for i in 0 ..< numEntryIndices - 1:
|
||||||
let
|
let
|
||||||
maybeLeftMaxX = max(left.segment.at.x, left.segment.to.x)
|
left = trapezoidSegments[entryIndices[i]]
|
||||||
maybeRightMaxX = max(right.segment.at.x, right.segment.to.x)
|
right = trapezoidSegments[entryIndices[i + 1]]
|
||||||
if maybeLeftMaxX > maybeRightMaxX:
|
leftMaxX = max(left.at.x, left.to.x)
|
||||||
swap left, right
|
rightMinX = min(right.at.x, right.to.x)
|
||||||
|
if leftMaxX.ceil.int > rightMinX.int:
|
||||||
|
noEntriesInScanlineOverlap = false
|
||||||
|
break
|
||||||
|
|
||||||
# Use trapezoid coverage at the edges and fill in the middle
|
if noEntriesInScanlineOverlap:
|
||||||
|
# Confirm the pairs of points represent simple fills between them
|
||||||
|
var
|
||||||
|
onlySimpleFillPairs = true
|
||||||
|
i, windingCount: int
|
||||||
|
while i < numEntryIndices:
|
||||||
|
windingCount += partition.entries[entryIndices[i]].winding
|
||||||
|
if not windingRule.shouldFill(windingCount):
|
||||||
|
onlySimpleFillPairs = false
|
||||||
|
break
|
||||||
|
windingCount += partition.entries[entryIndices[i + 1]].winding
|
||||||
|
if windingRule.shouldFill(windingCount):
|
||||||
|
onlySimpleFillPairs = false
|
||||||
|
break
|
||||||
|
i += 2
|
||||||
|
|
||||||
when allowSimd and defined(amd64):
|
if onlySimpleFillPairs:
|
||||||
let vecRgbx = mm_set_ps(
|
var i: int
|
||||||
rgbx.a.float32,
|
while i < numEntryIndices:
|
||||||
rgbx.b.float32,
|
|
||||||
rgbx.g.float32,
|
|
||||||
rgbx.r.float32
|
|
||||||
)
|
|
||||||
|
|
||||||
proc solveX(entry: PartitionEntry, y: float32): float32 =
|
|
||||||
if entry.m == 0:
|
|
||||||
entry.b
|
|
||||||
else:
|
|
||||||
(y - entry.b) / entry.m
|
|
||||||
|
|
||||||
proc solveY(entry: PartitionEntry, x: float32): float32 =
|
|
||||||
entry.m * x + entry.b
|
|
||||||
|
|
||||||
var
|
|
||||||
leftTop = vec2(0, y.float32)
|
|
||||||
leftBottom = vec2(0, (y + 1).float32)
|
|
||||||
leftTop.x = left.solveX(leftTop.y.float32)
|
|
||||||
leftBottom.x = left.solveX(leftBottom.y)
|
|
||||||
|
|
||||||
var
|
|
||||||
rightTop = vec2(0, y.float32)
|
|
||||||
rightBottom = vec2(0, (y + 1).float32)
|
|
||||||
rightTop.x = right.solveX(rightTop.y)
|
|
||||||
rightBottom.x = right.solveX(rightBottom.y)
|
|
||||||
|
|
||||||
let
|
|
||||||
leftMaxX = max(leftTop.x, leftBottom.x)
|
|
||||||
rightMinX = min(rightTop.x, rightBottom.x)
|
|
||||||
leftCoverEnd = leftMaxX.ceil.int
|
|
||||||
rightCoverBegin = rightMinX.trunc.int
|
|
||||||
|
|
||||||
if leftCoverEnd < rightCoverBegin:
|
|
||||||
# Only take this shortcut if the partial coverage areas on the
|
|
||||||
# left and the right do not overlap
|
|
||||||
|
|
||||||
let blender = blendMode.blender()
|
|
||||||
|
|
||||||
block: # Left-side partial coverage
|
|
||||||
let
|
|
||||||
inverted = leftTop.x < leftBottom.x
|
|
||||||
sliverStart = min(leftTop.x, leftBottom.x)
|
|
||||||
rectStart = max(leftTop.x, leftBottom.x)
|
|
||||||
var
|
|
||||||
pen = sliverStart
|
|
||||||
prevPen = pen
|
|
||||||
penY = if inverted: y.float32 else: (y + 1).float32
|
|
||||||
prevPenY = penY
|
|
||||||
for x in sliverStart.int ..< rectStart.ceil.int:
|
|
||||||
prevPen = pen
|
|
||||||
pen = (x + 1).float32
|
|
||||||
var rightRectArea = 0.float32
|
|
||||||
if pen > rectStart:
|
|
||||||
rightRectArea = pen - rectStart
|
|
||||||
pen = rectStart
|
|
||||||
prevPenY = penY
|
|
||||||
penY = left.solveY(pen)
|
|
||||||
if x < 0 or x >= image.width:
|
|
||||||
continue
|
|
||||||
let
|
let
|
||||||
run = pen - prevPen
|
left = partition.entries[entryIndices[i]]
|
||||||
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
|
right = partition.entries[entryIndices[i + 1]]
|
||||||
rectArea =
|
trapLeft = trapezoidSegments[entryIndices[i]]
|
||||||
if inverted:
|
trapRight = trapezoidSegments[entryIndices[i + 1]]
|
||||||
(prevPenY - y.float32) * run
|
|
||||||
else:
|
# Use trapezoid coverage at the edges and fill in the middle
|
||||||
((y + 1).float32 - prevPenY) * run
|
|
||||||
area = triangleArea + rectArea + rightRectArea
|
when allowSimd and defined(amd64):
|
||||||
dataIndex = image.dataIndex(x, y)
|
let vecRgbx = mm_set_ps(
|
||||||
backdrop = image.data[dataIndex]
|
rgbx.a.float32,
|
||||||
source =
|
rgbx.b.float32,
|
||||||
when allowSimd and defined(amd64):
|
rgbx.g.float32,
|
||||||
applyOpacity(vecRgbx, area)
|
rgbx.r.float32
|
||||||
else:
|
)
|
||||||
rgbx * area
|
|
||||||
image.data[dataIndex] = blender(backdrop, source)
|
|
||||||
|
|
||||||
block: # Right-side partial coverage
|
|
||||||
let
|
|
||||||
inverted = rightTop.x > rightBottom.x
|
|
||||||
rectEnd = min(rightTop.x, rightBottom.x)
|
|
||||||
sliverEnd = max(rightTop.x, rightBottom.x)
|
|
||||||
var
|
|
||||||
pen = rectEnd
|
|
||||||
prevPen = pen
|
|
||||||
penY = if inverted: (y + 1).float32 else: y.float32
|
|
||||||
prevPenY = penY
|
|
||||||
for x in rectEnd.int ..< sliverEnd.ceil.int:
|
|
||||||
prevPen = pen
|
|
||||||
pen = (x + 1).float32
|
|
||||||
let leftRectArea = prevPen.fractional
|
|
||||||
if pen > sliverEnd:
|
|
||||||
pen = sliverEnd
|
|
||||||
prevPenY = penY
|
|
||||||
penY = right.solveY(pen)
|
|
||||||
if x < 0 or x >= image.width:
|
|
||||||
continue
|
|
||||||
let
|
let
|
||||||
run = pen - prevPen
|
leftMaxX = max(trapLeft.at.x, trapLeft.to.x)
|
||||||
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
|
rightMinX = min(trapRight.at.x, trapRight.to.x)
|
||||||
rectArea =
|
leftCoverEnd = leftMaxX.ceil.int
|
||||||
if inverted:
|
rightCoverBegin = rightMinX.trunc.int
|
||||||
(penY - y.float32) * run
|
blender = blendMode.blender()
|
||||||
else:
|
|
||||||
((y + 1).float32 - penY) * run
|
|
||||||
area = leftRectArea + triangleArea + rectArea
|
|
||||||
dataIndex = image.dataIndex(x, y)
|
|
||||||
backdrop = image.data[dataIndex]
|
|
||||||
source =
|
|
||||||
when allowSimd and defined(amd64):
|
|
||||||
applyOpacity(vecRgbx, area)
|
|
||||||
else:
|
|
||||||
rgbx * area
|
|
||||||
image.data[dataIndex] = blender(backdrop, source)
|
|
||||||
|
|
||||||
let
|
block: # Left-side partial coverage
|
||||||
fillBegin = leftCoverEnd.clamp(0, image.width)
|
let
|
||||||
fillEnd = rightCoverBegin.clamp(0, image.width)
|
inverted = trapLeft.at.x < trapLeft.to.x
|
||||||
if fillEnd - fillBegin > 0:
|
sliverStart = min(trapLeft.at.x, trapLeft.to.x)
|
||||||
hits[0] = (fixed32(fillBegin.float32), 1.int16)
|
rectStart = leftMaxX
|
||||||
hits[1] = (fixed32(fillEnd.float32), -1.int16)
|
var
|
||||||
image.fillHits(rgbx, 0, y, hits, 2, NonZero, blendMode)
|
pen = sliverStart
|
||||||
|
prevPen = pen
|
||||||
|
penY = if inverted: y.float32 else: (y + 1).float32
|
||||||
|
prevPenY = penY
|
||||||
|
for x in sliverStart.int ..< rectStart.ceil.int:
|
||||||
|
prevPen = pen
|
||||||
|
pen = (x + 1).float32
|
||||||
|
var rightRectArea = 0.float32
|
||||||
|
if pen > rectStart:
|
||||||
|
rightRectArea = pen - rectStart
|
||||||
|
pen = rectStart
|
||||||
|
prevPenY = penY
|
||||||
|
penY = left.solveY(pen)
|
||||||
|
if x < 0 or x >= image.width:
|
||||||
|
continue
|
||||||
|
let
|
||||||
|
run = pen - prevPen
|
||||||
|
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
|
||||||
|
rectArea =
|
||||||
|
if inverted:
|
||||||
|
(prevPenY - y.float32) * run
|
||||||
|
else:
|
||||||
|
((y + 1).float32 - prevPenY) * run
|
||||||
|
area = triangleArea + rectArea + rightRectArea
|
||||||
|
dataIndex = image.dataIndex(x, y)
|
||||||
|
backdrop = image.data[dataIndex]
|
||||||
|
source =
|
||||||
|
when allowSimd and defined(amd64):
|
||||||
|
applyOpacity(vecRgbx, area)
|
||||||
|
else:
|
||||||
|
rgbx * area
|
||||||
|
image.data[dataIndex] = blender(backdrop, source)
|
||||||
|
|
||||||
inc y
|
block: # Right-side partial coverage
|
||||||
continue
|
let
|
||||||
|
inverted = trapRight.at.x > trapRight.to.x
|
||||||
|
rectEnd = rightMinX
|
||||||
|
sliverEnd = max(trapRight.at.x, trapRight.to.x)
|
||||||
|
var
|
||||||
|
pen = rectEnd
|
||||||
|
prevPen = pen
|
||||||
|
penY = if inverted: (y + 1).float32 else: y.float32
|
||||||
|
prevPenY = penY
|
||||||
|
for x in rectEnd.int ..< sliverEnd.ceil.int:
|
||||||
|
prevPen = pen
|
||||||
|
pen = (x + 1).float32
|
||||||
|
let leftRectArea = prevPen.fractional
|
||||||
|
if pen > sliverEnd:
|
||||||
|
pen = sliverEnd
|
||||||
|
prevPenY = penY
|
||||||
|
penY = right.solveY(pen)
|
||||||
|
if x < 0 or x >= image.width:
|
||||||
|
continue
|
||||||
|
let
|
||||||
|
run = pen - prevPen
|
||||||
|
triangleArea = 0.5.float32 * run * abs(penY - prevPenY)
|
||||||
|
rectArea =
|
||||||
|
if inverted:
|
||||||
|
(penY - y.float32) * run
|
||||||
|
else:
|
||||||
|
((y + 1).float32 - penY) * run
|
||||||
|
area = leftRectArea + triangleArea + rectArea
|
||||||
|
dataIndex = image.dataIndex(x, y)
|
||||||
|
backdrop = image.data[dataIndex]
|
||||||
|
source =
|
||||||
|
when allowSimd and defined(amd64):
|
||||||
|
applyOpacity(vecRgbx, area)
|
||||||
|
else:
|
||||||
|
rgbx * area
|
||||||
|
image.data[dataIndex] = blender(backdrop, source)
|
||||||
|
|
||||||
|
let
|
||||||
|
fillBegin = leftCoverEnd.clamp(0, image.width)
|
||||||
|
fillEnd = rightCoverBegin.clamp(0, image.width)
|
||||||
|
if fillEnd - fillBegin > 0:
|
||||||
|
hits[0] = (fixed32(fillBegin.float32), 1.int16)
|
||||||
|
hits[1] = (fixed32(fillEnd.float32), -1.int16)
|
||||||
|
image.fillHits(rgbx, 0, y, hits, 2, NonZero, blendMode)
|
||||||
|
|
||||||
|
i += 2
|
||||||
|
|
||||||
|
inc y
|
||||||
|
continue
|
||||||
|
|
||||||
computeCoverage(
|
computeCoverage(
|
||||||
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
|
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
|
||||||
hits,
|
hits,
|
||||||
numHits,
|
numHits,
|
||||||
aa,
|
|
||||||
image.width,
|
image.width,
|
||||||
y,
|
y,
|
||||||
startX,
|
startX,
|
||||||
partitions,
|
partitions,
|
||||||
partitionIndex,
|
partitionIndex,
|
||||||
|
entryIndices,
|
||||||
|
numEntryIndices,
|
||||||
windingRule
|
windingRule
|
||||||
)
|
)
|
||||||
|
|
||||||
if aa:
|
if partitions[partitionIndex].requiresAntiAliasing:
|
||||||
image.fillCoverage(
|
image.fillCoverage(
|
||||||
rgbx,
|
rgbx,
|
||||||
startX,
|
startX,
|
||||||
|
@ -2142,28 +2189,55 @@ proc fillShapes(
|
||||||
var
|
var
|
||||||
partitions = partitionSegments(segments, startY, pathHeight)
|
partitions = partitionSegments(segments, startY, pathHeight)
|
||||||
partitionIndex: int
|
partitionIndex: int
|
||||||
|
entryIndices = newSeq[int](partitions.maxEntryCount)
|
||||||
|
numEntryIndices: int
|
||||||
coverages = newSeq[uint8](pathWidth)
|
coverages = newSeq[uint8](pathWidth)
|
||||||
hits = newSeq[(Fixed32, int16)](partitions.maxEntryCount)
|
hits = newSeq[(Fixed32, int16)](partitions.maxEntryCount)
|
||||||
numHits: int
|
numHits: int
|
||||||
aa: bool
|
|
||||||
|
|
||||||
for y in startY ..< pathHeight:
|
for y in startY ..< pathHeight:
|
||||||
if y >= partitions[partitionIndex].bottom:
|
if y >= partitions[partitionIndex].bottom:
|
||||||
inc partitionIndex
|
inc partitionIndex
|
||||||
|
|
||||||
|
let
|
||||||
|
partition = partitions[partitionIndex].addr
|
||||||
|
partitionTop = partition.top
|
||||||
|
partitionBottom = partition.bottom
|
||||||
|
partitionHeight = partitionBottom - partitionTop
|
||||||
|
if partitionHeight == 0:
|
||||||
|
continue
|
||||||
|
|
||||||
|
let
|
||||||
|
scanTop = y.float32
|
||||||
|
scanBottom = (y + 1).float32
|
||||||
|
|
||||||
|
numEntryIndices = 0
|
||||||
|
if partition.twoNonintersectingSpanningSegments:
|
||||||
|
numEntryIndices = 2
|
||||||
|
entryIndices[0] = 0
|
||||||
|
entryIndices[1] = 1
|
||||||
|
else:
|
||||||
|
for i in 0 ..< partition.entries.len:
|
||||||
|
if partition.entries[i].segment.to.y < scanTop or
|
||||||
|
partition.entries[i].segment.at.y >= scanBottom:
|
||||||
|
continue
|
||||||
|
entryIndices[numEntryIndices] = i
|
||||||
|
inc numEntryIndices
|
||||||
|
|
||||||
computeCoverage(
|
computeCoverage(
|
||||||
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
|
cast[ptr UncheckedArray[uint8]](coverages[0].addr),
|
||||||
hits,
|
hits,
|
||||||
numHits,
|
numHits,
|
||||||
aa,
|
|
||||||
mask.width,
|
mask.width,
|
||||||
y,
|
y,
|
||||||
startX,
|
startX,
|
||||||
partitions,
|
partitions,
|
||||||
partitionIndex,
|
partitionIndex,
|
||||||
|
entryIndices,
|
||||||
|
numEntryIndices,
|
||||||
windingRule
|
windingRule
|
||||||
)
|
)
|
||||||
if aa:
|
if partitions[partitionIndex].requiresAntiAliasing:
|
||||||
mask.fillCoverage(startX, y, coverages, blendMode)
|
mask.fillCoverage(startX, y, coverages, blendMode)
|
||||||
zeroMem(coverages[0].addr, coverages.len)
|
zeroMem(coverages[0].addr, coverages.len)
|
||||||
else:
|
else:
|
||||||
|
|
Loading…
Reference in a new issue