From 33b032550ce4e2cbbf2604874baa8c226ae7f91f Mon Sep 17 00:00:00 2001 From: Ryan Oldenburg Date: Mon, 8 Feb 2021 19:51:15 -0600 Subject: [PATCH] draw mask on mask --- src/pixie/images.nim | 60 +++++++++++++++++------- src/pixie/masks.nim | 2 +- src/pixie/paths.nim | 17 +++++++ tests/images/{ => masks}/circleMask.png | Bin tests/images/masks/maskedMask.png | Bin 0 -> 885 bytes tests/test_images.nim | 14 ------ tests/test_masks.nim | 29 +++++++++++- 7 files changed, 88 insertions(+), 34 deletions(-) rename tests/images/{ => masks}/circleMask.png (100%) create mode 100644 tests/images/masks/maskedMask.png diff --git a/src/pixie/images.nim b/src/pixie/images.nim index 993b3b4..f7b8bd5 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -292,18 +292,31 @@ proc getRgbaSmooth*(image: Image, x, y: float32): ColorRGBA = lerp(bottomMix, topMix, diffY) proc drawCorrect( - a: Image, b: Image | Mask, mat = mat3(), blendMode = bmNormal + a: Image | Mask, b: Image | Mask, mat = mat3(), blendMode = bmNormal ) = ## Draws one image onto another using matrix with color blending. - when type(b) is Image: - let blender = blendMode.blender() - else: + + proc validateMaskBlendMode() = if blendMode notin {bmMask}: raise newException( PixieError, "Blend mode " & $blendMode & " not supported for masks" ) + when type(a) is Image: + when type(b) is Image: + let blender = blendMode.blenderPremultiplied() + else: # b is a Mask + validateMaskBlendMode() + else: # a is a Mask + when type(b) is Image: + raise newException( + PixieError, + "Drawing an image onto a mask is not supported yet" + ) + else: # b is a Mask + validateMaskBlendMode() + var matInv = mat.inverse() b = b @@ -325,28 +338,39 @@ proc drawCorrect( samplePos = matInv * vec2(x.float32 + h, y.float32 + h) xFloat = samplePos.x - h yFloat = samplePos.y - h - rgba = a.getRgbaUnsafe(x, y) - var blended: ColorRGBA - when type(b) is Image: - let sample = b.getRgbaSmooth(xFloat, yFloat) - blended = blender(rgba, sample) - else: - let sample = b.getValueSmooth(xFloat, yFloat).uint32 - blended = rgba( - ((rgba.r * sample) div 255).uint8, - ((rgba.g * sample) div 255).uint8, - ((rgba.b * sample) div 255).uint8, - ((rgba.a * sample) div 255).uint8 - ) + when type(a) is Image: + let rgba = a.getRgbaUnsafe(x, y) + var blended: ColorRGBA + when type(b) is Image: + let sample = b.getRgbaSmooth(xFloat, yFloat) + blended = blender(rgba, sample) + else: # b is a Mask + let sample = b.getValueSmooth(xFloat, yFloat).uint32 + blended = rgba( + ((rgba.r * sample) div 255).uint8, + ((rgba.g * sample) div 255).uint8, + ((rgba.b * sample) div 255).uint8, + ((rgba.a * sample) div 255).uint8 + ) + a.setRgbaUnsafe(x, y, blended) + else: # a is a Mask, b must be a mask + let + value = a.getValueUnsafe(x, y) + sample = b.getValueSmooth(xFloat, yFloat).uint32 + a.setValueUnsafe(x, y, ((value * sample) div 255).uint8) - a.setRgbaUnsafe(x, y, blended) +proc draw*(image: Image, mask: Mask, mat: Mat3, blendMode = bmMask) = + image.drawCorrect(mask, mat, blendMode) proc draw*( image: Image, mask: Mask, pos = vec2(0, 0), blendMode = bmMask ) {.inline.} = image.drawCorrect(mask, translate(pos), blendMode) +proc draw*(a, b: Mask, mat = mat3(), blendMode = bmMask) = + a.drawCorrect(b, mat, blendMode) + when defined(release): {.pop.} diff --git a/src/pixie/masks.nim b/src/pixie/masks.nim index c600409..93cf1af 100644 --- a/src/pixie/masks.nim +++ b/src/pixie/masks.nim @@ -1,4 +1,4 @@ -import common, vmath, system/memory +import blends, common, vmath, system/memory type Mask* = ref object diff --git a/src/pixie/paths.nim b/src/pixie/paths.nim index d0895b6..71034d8 100644 --- a/src/pixie/paths.nim +++ b/src/pixie/paths.nim @@ -363,6 +363,23 @@ proc rect*(path: var Path, x, y, w, h: float32) = proc rect*(path: var Path, pos: Vec2, wh: Vec2) {.inline.} = path.rect(pos.x, pos.y, wh.x, wh.y) +proc roundedRect*( + path: var Path, pos, wh: Vec2, nw, ne, se, sw: float32, clockwise = true +) = + let + maxRadius = min(wh.x / 2, wh.y / 2) + nw = min(nw, maxRadius) + ne = min(ne, maxRadius) + se = min(se, maxRadius) + sw = min(sw, maxRadius) + + path.moveTo(pos.x + nw, pos.y) + path.arcTo(pos.x + wh.x, pos.y, pos.x + wh.x, pos.y + wh.y, ne) + path.arcTo(pos.x + wh.x, pos.y + wh.y, pos.x, pos.y + wh.y, se) + path.arcTo(pos.x, pos.y + wh.y, pos.x, pos.y, sw) + path.arcTo(pos.x, pos.y, pos.x + wh.x, pos.y, nw) + path.closePath() + proc ellipse*(path: var Path, cx, cy, rx, ry: float32) = let magicX = (4.0 * (-1.0 + sqrt(2.0)) / 3) * rx diff --git a/tests/images/circleMask.png b/tests/images/masks/circleMask.png similarity index 100% rename from tests/images/circleMask.png rename to tests/images/masks/circleMask.png diff --git a/tests/images/masks/maskedMask.png b/tests/images/masks/maskedMask.png new file mode 100644 index 0000000000000000000000000000000000000000..eb13fae891b2101bbdada95316a620525090c47f GIT binary patch literal 885 zcmV-*1B(2KP)2)5ht$#uYSWmaFl{YVlE@Z@Ry10W`4dI8kj555Ons95;`s=bX-d4xRlUwDWMXVtF-pmP2cnS(&{{K(v%_Zk(Rj6-fphU%T04T z9PYH-yo#&)woXMQJ(7~Nrl8jo$Br52IqKgzET@r@xNZgW9jsvqM@~&jL?$&EiT7Qx zb7@#B&Vmc06V{vSk&s;LNM7G>g``un;g!!QLP=u4&q-{~SR*buQ-JkB z_<|l4Tyj1e84P0*Et9yUHVYdKVY(A(AS89$u|5dKQ>1~A+)c##AUN+K4TPjlRbtqX zoKuZ!(nefmstZ42W8sTbw{Y0T7T&9R82fD;#%<-;WOW)vl89E^zH&MK2^#6;hwJ=SsV<5oC%B@MOa4@H(r=7 zts95;`s=bX-dOKXLs5-Hoq){E