diff --git a/bindings/bindings.nim b/bindings/bindings.nim index 82dbc0e..83e0758 100644 --- a/bindings/bindings.nim +++ b/bindings/bindings.nim @@ -156,6 +156,7 @@ exportRefObject Mask: applyOpacity(Mask, float32) invert(Mask) blur(Mask, float32, uint8) + resize(Mask, int, int) draw(Mask, Mask, Mat3, BlendMode) draw(Mask, Image, Mat3, BlendMode) fillText(Mask, Font, string, Mat3, Vec2, HorizontalAlignment, VerticalAlignment) diff --git a/src/pixie/blends.nim b/src/pixie/blends.nim index 01a5c39..1ef0900 100644 --- a/src/pixie/blends.nim +++ b/src/pixie/blends.nim @@ -1,8 +1,8 @@ ## Blending modes. -import chroma, common, math +import chroma, common, internal, std/math -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 # See https://www.w3.org/TR/compositing-1/ @@ -274,7 +274,7 @@ proc blendSoftLight(backdrop, source: ColorRGBX): ColorRGBX = source = source.rgba() var rgba: ColorRGBA - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let vb = mm_setr_ps( backdrop.r.float32, @@ -479,7 +479,7 @@ proc masker*(blendMode: BlendMode): Masker {.raises: [PixieError].} = else: raise newException(PixieError, "No masker for " & $blendMode) -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: type BlenderSimd* = proc(blackdrop, source: M128i): M128i {.gcsafe, raises: [].} ## Function signature returned by blenderSimd. diff --git a/src/pixie/fileformats/jpeg.nim b/src/pixie/fileformats/jpeg.nim index f2eebf3..22bf277 100644 --- a/src/pixie/fileformats/jpeg.nim +++ b/src/pixie/fileformats/jpeg.nim @@ -1,7 +1,7 @@ -import pixie/common, pixie/images, pixie/masks, sequtils, strutils, chroma, - std/decls, flatty/binny +import chroma, flatty/binny, pixie/common, pixie/images, pixie/internal, + pixie/masks, sequtils, std/decls, strutils -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 # This JPEG decoder is loosely based on stb_image which is public domain. @@ -649,7 +649,7 @@ proc decodeProgressiveContinuationBlock( data[zig] = cast[int16](state.receiveExtend(s.int) * (1 shl shift)) else: - var bit = 1 shl state.successiveApproxLow + let bit = 1 shl state.successiveApproxLow if state.eobRun != 0: dec state.eobRun @@ -681,9 +681,9 @@ proc decodeProgressiveContinuationBlock( if s != 1: failInvalid("bad huffman code") if state.readBit() != 0: - s = bit.int + s = bit else: - s = -bit.int + s = -bit while k <= state.spectralEnd: let zig = deZigZag[k] @@ -881,7 +881,7 @@ proc quantizationAndIDCTPass(state: var DecoderState) = for row in 0 ..< w: var data {.byaddr.} = state.components[comp].blocks[row][column] - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: for i in 0 ..< 8: # 8 per pass var q = mm_loadu_si128(state.quantizationTables[qTableId][i * 8].addr) q = mm_unpacklo_epi8(q, mm_setzero_si128()) @@ -906,13 +906,17 @@ proc magnifyXBy2(mask: Mask): Mask = let n = 3 * mask.unsafe[x, y].uint16 if x == 0: result.unsafe[x * 2 + 0, y] = mask.unsafe[x, y] - result.unsafe[x * 2 + 1, y] = ((n + mask.unsafe[x + 1, y].uint16 + 2) div 4).uint8 + result.unsafe[x * 2 + 1, y] = + ((n + mask.unsafe[x + 1, y].uint16 + 2) div 4).uint8 elif x == mask.width - 1: - result.unsafe[x * 2 + 0, y] = ((n + mask.unsafe[x - 1, y].uint16 + 2) div 4).uint8 + result.unsafe[x * 2 + 0, y] = + ((n + mask.unsafe[x - 1, y].uint16 + 2) div 4).uint8 result.unsafe[x * 2 + 1, y] = mask.unsafe[x, y] else: - result.unsafe[x * 2 + 0, y] = ((n + mask.unsafe[x - 1, y].uint16) div 4).uint8 - result.unsafe[x * 2 + 1, y] = ((n + mask.unsafe[x + 1, y].uint16) div 4).uint8 + result.unsafe[x * 2 + 0, y] = + ((n + mask.unsafe[x - 1, y].uint16) div 4).uint8 + result.unsafe[x * 2 + 1, y] = + ((n + mask.unsafe[x + 1, y].uint16) div 4).uint8 proc magnifyYBy2(mask: Mask): Mask = ## Smooth magnify by power of 2 only in the Y direction. @@ -922,13 +926,17 @@ proc magnifyYBy2(mask: Mask): Mask = let n = 3 * mask.unsafe[x, y].uint16 if y == 0: result.unsafe[x, y * 2 + 0] = mask.unsafe[x, y] - result.unsafe[x, y * 2 + 1] = ((n + mask.unsafe[x, y + 1].uint16 + 2) div 4).uint8 + result.unsafe[x, y * 2 + 1] = + ((n + mask.unsafe[x, y + 1].uint16 + 2) div 4).uint8 elif y == mask.height - 1: - result.unsafe[x, y * 2 + 0] = ((n + mask.unsafe[x, y - 1].uint16 + 2) div 4).uint8 + result.unsafe[x, y * 2 + 0] = + ((n + mask.unsafe[x, y - 1].uint16 + 2) div 4).uint8 result.unsafe[x, y * 2 + 1] = mask.unsafe[x, y] else: - result.unsafe[x, y * 2 + 0] = ((n + mask.unsafe[x, y - 1].uint16) div 4).uint8 - result.unsafe[x, y * 2 + 1] = ((n + mask.unsafe[x, y + 1].uint16) div 4).uint8 + result.unsafe[x, y * 2 + 0] = + ((n + mask.unsafe[x, y - 1].uint16) div 4).uint8 + result.unsafe[x, y * 2 + 1] = + ((n + mask.unsafe[x, y + 1].uint16) div 4).uint8 proc yCbCrToRgbx(py, pcb, pcr: uint8): ColorRGBX = ## Takes a 3 component yCbCr outputs and populates image. diff --git a/src/pixie/fileformats/qoi.nim b/src/pixie/fileformats/qoi.nim index 887191b..bc6c7f9 100644 --- a/src/pixie/fileformats/qoi.nim +++ b/src/pixie/fileformats/qoi.nim @@ -26,10 +26,10 @@ type Index = array[indexLen, ColorRGBA] -func hash(p: ColorRGBA): int = +proc hash(p: ColorRGBA): int = (p.r.int * 3 + p.g.int * 5 + p.b.int * 7 + p.a.int * 11) mod indexLen -func newImage*(qoi: Qoi): Image = +proc newImage*(qoi: Qoi): Image = ## Converts raw QOI data to `Image`. result = newImage(qoi.width, qoi.height) copyMem(result.data[0].addr, qoi.data[0].addr, qoi.data.len * 4) diff --git a/src/pixie/fileformats/svg.nim b/src/pixie/fileformats/svg.nim index 168ad1f..1a5e78a 100644 --- a/src/pixie/fileformats/svg.nim +++ b/src/pixie/fileformats/svg.nim @@ -519,7 +519,6 @@ proc parseSvg*( var rootProps = initSvgProperties() rootProps = root.parseSvgProperties(rootProps) - if viewBoxMinX != 0 or viewBoxMinY != 0: let viewBoxMin = vec2(-viewBoxMinX.float32, -viewBoxMinY.float32) rootprops.transform = rootprops.transform * translate(viewBoxMin) diff --git a/src/pixie/fontformats/opentype.nim b/src/pixie/fontformats/opentype.nim index 5d0e9e1..fde88ae 100644 --- a/src/pixie/fontformats/opentype.nim +++ b/src/pixie/fontformats/opentype.nim @@ -1,5 +1,5 @@ import flatty/binny, flatty/encode, math, pixie/common, pixie/paths, sets, - strutils, tables, unicode, vmath + strutils, tables, unicode, vmath ## See https://docs.microsoft.com/en-us/typography/opentype/spec/ @@ -663,7 +663,7 @@ proc parseNameTable(buf: string, offset: int): NameTable = if record.platformID == 3 and record.encodingID == 1 and record.languageID == 1033: - record.text = fromUTF16BE(record.text) + record.text = fromUTF16BE(record.text) record.text = record.text result.nameRecords.add(record) diff --git a/src/pixie/images.nim b/src/pixie/images.nim index ce3b415..a99c62d 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -1,6 +1,6 @@ import blends, bumpy, chroma, common, masks, pixie/internal, vmath -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 const h = 0.5.float32 @@ -29,7 +29,7 @@ proc newImage*(width, height: int): Image {.raises: [PixieError].} = proc newImage*(mask: Mask): Image {.raises: [PixieError].} = result = newImage(mask.width, mask.height) var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: for _ in 0 ..< mask.data.len div 16: var alphas = mm_loadu_si128(mask.data[i].addr) for j in 0 ..< 4: @@ -63,7 +63,7 @@ proc dataIndex*(image: Image, x, y: int): int {.inline, raises: [].} = template unsafe*(src: Image): UnsafeImage = cast[UnsafeImage](src) -template `[]`*(view: UnsafeImage, x, y: int): ColorRGBX = +template `[]`*(view: UnsafeImage, x, y: int): var ColorRGBX = ## Gets a color from (x, y) coordinates. ## * No bounds checking * ## Make sure that x, y are in bounds. @@ -106,7 +106,7 @@ proc isOneColor*(image: Image): bool {.raises: [].} = let color = image.data[0] var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let colorVec = mm_set1_epi32(cast[int32](color)) for _ in 0 ..< image.data.len div 8: let @@ -127,7 +127,7 @@ proc isTransparent*(image: Image): bool {.raises: [].} = result = true var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let vecZero = mm_setzero_si128() for _ in 0 ..< image.data.len div 16: let @@ -254,7 +254,7 @@ proc minifyBy2*(image: Image, power = 1): Image {.raises: [PixieError].} = ) for y in 0 ..< resultEvenHeight: var x: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let oddMask = mm_set1_epi16(cast[int16](0xff00)) first32 = cast[M128i]([uint32.high, 0, 0, 0]) @@ -348,7 +348,7 @@ proc magnifyBy2*(image: Image, power = 1): Image {.raises: [PixieError].} = for y in 0 ..< image.height: # Write one row of pixels duplicated by scale var x: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if scale == 2: while x <= image.width - 4: let @@ -391,7 +391,7 @@ proc applyOpacity*(target: Image | Mask, opacity: float32) {.raises: [].} = return var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: when type(target) is Image: let byteLen = target.data.len * 4 else: @@ -447,7 +447,7 @@ proc applyOpacity*(target: Image | Mask, opacity: float32) {.raises: [].} = proc invert*(target: Image) {.raises: [].} = ## Inverts all of the colors and alpha. var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let vec255 = mm_set1_epi8(cast[int8](255)) let byteLen = target.data.len * 4 for _ in 0 ..< byteLen div 16: @@ -536,7 +536,7 @@ proc newMask*(image: Image): Mask {.raises: [PixieError].} = result = newMask(image.width, image.height) var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: for _ in 0 ..< image.data.len div 16: let a = mm_loadu_si128(image.data[i + 0].addr) @@ -798,7 +798,7 @@ proc drawUber( ) continue - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: case blendMode: of OverwriteBlend: for _ in 0 ..< (xStop - xStart) div 16: @@ -1155,6 +1155,21 @@ proc resize*(srcImage: Image, width, height: int): Image {.raises: [PixieError]. OverwriteBlend ) +proc resize*(srcMask: Mask, width, height: int): Mask {.raises: [PixieError].} = + ## Resize a mask to a given height and width. + if width == srcMask.width and height == srcMask.height: + result = srcMask.copy() + else: + result = newMask(width, height) + result.draw( + srcMask, + scale(vec2( + width.float32 / srcMask.width.float32, + height.float32 / srcMask.height.float32 + )), + OverwriteBlend + ) + proc shadow*( image: Image, offset: Vec2, spread, blur: float32, color: SomeColor ): Image {.raises: [PixieError].} = diff --git a/src/pixie/internal.nim b/src/pixie/internal.nim index eea791f..ead70e4 100644 --- a/src/pixie/internal.nim +++ b/src/pixie/internal.nim @@ -1,6 +1,8 @@ import chroma, system/memory, vmath -when defined(amd64) and not defined(pixieNoSimd): +const allowSimd* = not defined(pixieNoSimd) and not defined(tcc) + +when defined(amd64) and allowSimd: import nimsimd/sse2 template currentExceptionAsPixieError*(): untyped = @@ -59,7 +61,7 @@ proc fillUnsafe*( nimSetMem(data[start].addr, rgbx.r.cint, len * 4) else: var i = start - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: # When supported, SIMD fill until we run out of room let colorVec = mm_set1_epi32(cast[int32](rgbx)) for _ in 0 ..< len div 8: @@ -93,7 +95,7 @@ proc toStraightAlpha*(data: var seq[ColorRGBA | ColorRGBX]) {.raises: [].} = proc toPremultipliedAlpha*(data: var seq[ColorRGBA | ColorRGBX]) {.raises: [].} = ## Converts an image to premultiplied alpha from straight alpha. var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: # When supported, SIMD convert as much as possible let alphaMask = mm_set1_epi32(cast[int32](0xff000000)) @@ -140,7 +142,7 @@ proc isOpaque*(data: var seq[ColorRGBX], start, len: int): bool = result = true var i = start - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let vec255 = mm_set1_epi32(cast[int32](uint32.high)) colorMask = mm_set1_epi32(cast[int32]([255.uint8, 255, 255, 0])) @@ -161,7 +163,7 @@ proc isOpaque*(data: var seq[ColorRGBX], start, len: int): bool = if data[j].a != 255: return false -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: proc packAlphaValues*(v: M128i): M128i {.inline, raises: [].} = ## Shuffle the alpha values for these 4 colors to the first 4 bytes let mask = mm_set1_epi32(cast[int32](0xff000000)) diff --git a/src/pixie/masks.nim b/src/pixie/masks.nim index 4ff81ac..3af7cf7 100644 --- a/src/pixie/masks.nim +++ b/src/pixie/masks.nim @@ -1,6 +1,6 @@ import common, internal, vmath -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 type @@ -87,7 +87,7 @@ proc minifyBy2*(mask: Mask, power = 1): Mask {.raises: [PixieError].} = result = newMask(src.width div 2, src.height div 2) for y in 0 ..< result.height: var x: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let oddMask = mm_set1_epi16(cast[int16](0xff00)) firstByte = cast[M128i]( @@ -169,7 +169,7 @@ proc magnifyBy2*(mask: Mask, power = 1): Mask {.raises: [PixieError].} = for y in 0 ..< mask.height: # Write one row of values duplicated by scale var x: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if scale == 2: while x <= mask.width - 16: let @@ -236,7 +236,7 @@ proc getValueSmooth*(mask: Mask, x, y: float32): uint8 {.raises: [].} = proc invert*(mask: Mask) {.raises: [].} = ## Inverts all of the values - creates a negative of the mask. var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let vec255 = mm_set1_epi8(cast[int8](255)) let byteLen = mask.data.len for _ in 0 ..< byteLen div 16: @@ -312,7 +312,7 @@ proc spread*(mask: Mask, spread: float32) {.raises: [PixieError].} = proc ceil*(mask: Mask) {.raises: [].} = ## A value of 0 stays 0. Anything else turns into 255. var i: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let zeroVec = mm_setzero_si128() vec255 = mm_set1_epi32(cast[int32](uint32.high)) diff --git a/src/pixie/paints.nim b/src/pixie/paints.nim index 1b6aa7d..133a367 100644 --- a/src/pixie/paints.nim +++ b/src/pixie/paints.nim @@ -1,6 +1,6 @@ -import chroma, common, images, vmath +import chroma, common, images, internal, vmath -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 type @@ -122,7 +122,7 @@ proc fillGradientLinear(image: Image, paint: Paint) = if at.y == to.y: # Horizontal gradient var x: int while x < image.width: - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if x + 4 <= image.width: var colors: array[4, ColorRGBX] for i in 0 ..< 4: @@ -153,7 +153,7 @@ proc fillGradientLinear(image: Image, paint: Paint) = t = toLineSpace(at, to, xy) rgbx = paint.gradientColor(t) var x: int - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let colorVec = mm_set1_epi32(cast[int32](rgbx)) for _ in 0 ..< image.width div 4: mm_storeu_si128(image.data[image.dataIndex(x, y)].addr, colorVec) diff --git a/src/pixie/paths.nim b/src/pixie/paths.nim index 414a619..50f6054 100644 --- a/src/pixie/paths.nim +++ b/src/pixie/paths.nim @@ -1,7 +1,7 @@ import blends, bumpy, chroma, common, fenv, images, internal, masks, paints, strutils, vmath -when defined(amd64) and not defined(pixieNoSimd): +when defined(amd64) and allowSimd: import nimsimd/sse2 type @@ -1296,7 +1296,7 @@ proc computeCoverage( let fillLen = at.int - fillStart if fillLen > 0: var i = fillStart - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: let sampleCoverageVec = mm_set1_epi8(cast[int8](sampleCoverage)) for _ in 0 ..< fillLen div 16: var coverageVec = mm_loadu_si128(coverages[i - startX].addr) @@ -1326,7 +1326,7 @@ proc fillCoverage( blendMode: BlendMode ) = var x = startX - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if blendMode.hasSimdBlender(): # When supported, SIMD blend as much as possible let @@ -1445,7 +1445,7 @@ proc fillCoverage( blendMode: BlendMode ) = var x = startX - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if blendMode.hasSimdMasker(): let maskerSimd = blendMode.maskerSimd() @@ -1511,7 +1511,7 @@ proc fillHits( continue var x = fillStart - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if blendMode.hasSimdBlender(): # When supported, SIMD blend as much as possible let colorVec = mm_set1_epi32(cast[int32](rgbx)) @@ -1573,7 +1573,7 @@ proc fillHits( continue var x = fillStart - when defined(amd64) and not defined(pixieNoSimd): + when defined(amd64) and allowSimd: if blendMode.hasSimdMasker(): let maskerSimd = blendMode.maskerSimd() diff --git a/tests/benchmark_paths.nim b/tests/benchmark_paths.nim index cab240a..84c5f32 100644 --- a/tests/benchmark_paths.nim +++ b/tests/benchmark_paths.nim @@ -16,12 +16,101 @@ block: timeIt "fillOverlaps": doAssert path.fillOverlaps(vec2(1, 1)) == false -timeIt "roundedRect": - const radius = 20 +const + width = 500 + height = 300 + radius = 20 - let path = newPath() - path.roundedRect(0.5, 0.5, 499, 299, radius, radius, radius, radius) - # path.roundedRect(0, 0, 500, 300, radius, radius, radius, radius) +let paint = newPaint(SolidPaint) +paint.color = color(0, 0, 0, 0.5) - let image = newImage(500, 300) - image.fillPath(path, rgba(0, 0, 0, 255)) +let rect = newPath() +rect.rect(10, 10, 480, 280) + +let roundedRect = newPath() +roundedRect.roundedRect(10.5, 10.5, 479, 279, radius, radius, radius, radius) +# roundedRect.roundedRect(10, 10, 480, 280, radius, radius, radius, radius) + +timeIt "rect Image OverwriteBlend": + paint.blendMode = OverwriteBlend + + let image = newImage(width, height) + image.fillPath(rect, paint) + +timeIt "rect Image NormalBlend": + paint.blendMode = NormalBlend + + let image = newImage(width, height) + image.fillPath(rect, paint) + +timeIt "rect Image MaskBlend": + paint.blendMode = MaskBlend + + let image = newImage(width, height) + image.fill(rgbx(255, 255, 255, 255)) + image.fillPath(rect, paint) + +timeIt "roundedRect Image OverwriteBlend": + paint.blendMode = OverwriteBlend + + let image = newImage(width, height) + image.fillPath(roundedRect, paint) + +timeIt "roundedRect Image NormalBlend": + paint.blendMode = NormalBlend + + let image = newImage(width, height) + image.fillPath(roundedRect, paint) + +timeIt "roundedRect Image MaskBlend": + paint.blendMode = MaskBlend + + let image = newImage(width, height) + image.fill(rgbx(255, 255, 255, 255)) + image.fillPath(roundedRect, paint) + +timeIt "rect Mask OverwriteBlend": + let mask = newMask(width, height) + mask.fillPath(roundedRect, blendMode = OverwriteBlend) + +timeIt "rect Mask NormalBlend": + let mask = newMask(width, height) + mask.fillPath(rect, blendMode = NormalBlend) + +timeIt "rect Mask MaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(rect, blendMode = MaskBlend) + +timeIt "rect Mask SubtractMaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(rect, blendMode = SubtractMaskBlend) + +timeIt "rect Mask ExcludeMaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(rect, blendMode = ExcludeMaskBlend) + +timeIt "roundedRect Mask OverwriteBlend": + let mask = newMask(width, height) + mask.fillPath(roundedRect, blendMode = OverwriteBlend) + +timeIt "roundedRect Mask NormalBlend": + let mask = newMask(width, height) + mask.fillPath(roundedRect, blendMode = NormalBlend) + +timeIt "roundedRect Mask MaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(roundedRect, blendMode = MaskBlend) + +timeIt "roundedRect Mask SubtractMaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(roundedRect, blendMode = SubtractMaskBlend) + +timeIt "roundedRect Mask ExcludeMaskBlend": + let mask = newMask(width, height) + mask.fill(255) + mask.fillPath(roundedRect, blendMode = ExcludeMaskBlend) diff --git a/tests/benchmark_png.nim b/tests/benchmark_png.nim index 29ad32a..81fcf76 100644 --- a/tests/benchmark_png.nim +++ b/tests/benchmark_png.nim @@ -5,38 +5,42 @@ let filePath = "tests/fileformats/png/lenna.png" data = readFile(filePath) -timeIt "pixie decode": - keep decodePngRaw(data) +block: + let + decodedPng = decodePng(data) + decodedImage = newImage(decodedPng) -timeIt "pixie encode": - let decoded = decodePngRaw(data) - keep encodePng(decoded).len + timeIt "pixie decode": + discard decodePng(data) -timeIt "pixie decode + alpha": - keep decodePng(data) + timeIt "pixie encode": + discard encodePng(decodedPng) -timeIt "pixie encode + alpha": - let decoded = decodePng(data) - keep encodePng(decoded).len + timeIt "pixie decode + alpha": + discard newImage(decodePng(data)) -timeIt "nimPNG decode": - keep decodePNG32(data) + timeIt "pixie encode + alpha": + discard encodePng(decodedImage) + +block: + timeIt "nimPNG decode": + discard decodePNG32(data) -timeIt "nimPNG encode": let decoded = decodePNG32(data) - keep encodePNG32(decoded.data, decoded.width, decoded.height).pixels.len + timeIt "nimPNG encode": + discard encodePNG32(decoded.data, decoded.width, decoded.height) -timeIt "stb_image decode": - var width, height, channels: int - keep loadFromMemory( - cast[seq[byte]](data), - width, - height, - channels, - stbi.RGBA - ) +block: + timeIt "stb_image decode": + var width, height, channels: int + discard loadFromMemory( + cast[seq[byte]](data), + width, + height, + channels, + stbi.RGBA + ) -timeIt "stb_image encode": var width, height, channels: int let decoded = loadFromMemory( cast[seq[byte]](data), @@ -45,16 +49,17 @@ timeIt "stb_image encode": channels, stbi.RGBA ) - keep writePNG(width, height, channels, decoded).len -timeIt "cairo decode": - keep imageSurfaceCreateFromPng(filePath) + timeIt "stb_image encode": + discard writePNG(width, height, channels, decoded).len + +block: + timeIt "cairo decode": + discard imageSurfaceCreateFromPng(filePath) -timeIt "cairo encode": let decoded = imageSurfaceCreateFromPng(filePath) - - var write: WriteFunc = - proc(closure: pointer, data: cstring, len: int32): Status {.cdecl.} = - StatusSuccess - - discard decoded.writeToPng(write, nil) + timeIt "cairo encode": + var write: WriteFunc = + proc(closure: pointer, data: cstring, len: int32): Status {.cdecl.} = + StatusSuccess + discard decoded.writeToPng(write, nil) diff --git a/tests/test_fonts.nim b/tests/test_fonts.nim index 0ffff65..ce6ee67 100644 --- a/tests/test_fonts.nim +++ b/tests/test_fonts.nim @@ -1,4 +1,4 @@ -import pixie, pixie/fileformats/png, strformat, unicode, os +import os, pixie, pixie/fileformats/png, strformat, unicode proc wh(image: Image): Vec2 = ## Return with and height as a size vector. diff --git a/tests/test_images.nim b/tests/test_images.nim index 8b2a756..2e0fcd6 100644 --- a/tests/test_images.nim +++ b/tests/test_images.nim @@ -23,10 +23,10 @@ block: doAssert image[9, 9] == rgba(128, 0, 0, 128) block: - let image = newImage(10, 10) - image.fill(rgba(128, 0, 0, 128)) - image.data.toStraightAlpha() - doAssert image[9, 9] == rgba(254, 0, 0, 128) + var data = newSeq[ColorRGBX](100) + fillUnsafe(data, rgbx(100, 0, 0, 128), 0, data.len) + data.toStraightAlpha() + doAssert data[10] == rgbx(199, 0, 0, 128) block: let image = newImage(100, 100) diff --git a/tests/test_svg.nim b/tests/test_svg.nim index e31e449..9bc7683 100644 --- a/tests/test_svg.nim +++ b/tests/test_svg.nim @@ -29,5 +29,8 @@ for file in files: doDiff(readImage(&"tests/fileformats/svg/{file}.svg"), file) block: - let svg = parseSvg(readFile("tests/fileformats/svg/accessibility-outline.svg"), 512, 512) + let svg = parseSvg( + readFile("tests/fileformats/svg/accessibility-outline.svg"), + 512, 512 + ) doDiff(newImage(svg), "accessibility-outline")