some small things

This commit is contained in:
Ryan Oldenburg 2020-12-03 00:06:57 -06:00
parent 3cb35ddee1
commit 37c5869b0c
6 changed files with 46 additions and 55 deletions

View file

@ -248,7 +248,7 @@ proc mix*(blendMode: BlendMode, target, blend: Color): Color =
result.b /= result.a result.b /= result.a
proc mix*(blendMode: BlendMode, dest, src: ColorRGBA): ColorRGBA {.inline.} = proc mix*(blendMode: BlendMode, dest, src: ColorRGBA): ColorRGBA {.inline.} =
return blendMode.mix(dest.color, src.color).rgba blendMode.mix(dest.color, src.color).rgba
# TODO: Fix fast paths # TODO: Fix fast paths
# if blendMode == Normal: # if blendMode == Normal:
@ -271,15 +271,14 @@ proc mix*(blendMode: BlendMode, dest, src: ColorRGBA): ColorRGBA {.inline.} =
# else: # else:
# return blendMode.mix(target.color, blend.color).rgba # return blendMode.mix(target.color, blend.color).rgba
proc multiply(Cb, Cs: float32): float32 {.inline.} =
Cb * Cs
proc screen(Cb, Cs: float32): float32 {.inline.} = proc screen(Cb, Cs: float32): float32 {.inline.} =
1 - (1 - Cb) * (1 - Cs) 1 - (1 - Cb) * (1 - Cs)
proc hardLight(Cb, Cs: float32): float32 {.inline.} = proc hardLight(Cb, Cs: float32): float32 {.inline.} =
if Cs <= 0.5: multiply(Cb, 2 * Cs) if Cs <= 0.5:
else: screen(Cb, 2 * Cs - 1) Cb * 2 * Cs
else:
screen(Cb, 2 * Cs - 1)
proc softLight(a, b: float32): float32 {.inline.} = proc softLight(a, b: float32): float32 {.inline.} =
## Pegtop ## Pegtop
@ -365,7 +364,7 @@ proc blendDarken(Cb, Cs: float32): float32 {.inline.} =
min(Cb, Cs) min(Cb, Cs)
proc blendMultiply(Cb, Cs: float32): float32 {.inline.} = proc blendMultiply(Cb, Cs: float32): float32 {.inline.} =
multiply(Cb, Cs) Cb * Cs
proc blendLinearBurn(Cb, Cs: float32): float32 {.inline.} = proc blendLinearBurn(Cb, Cs: float32): float32 {.inline.} =
Cb + Cs - 1 Cb + Cs - 1
@ -675,7 +674,6 @@ proc mix2*(blendMode: BlendMode, dest, src: ColorRGBA): ColorRGBA {.inline.} =
of bmIntersectMask: blendIntersectMask(dest, src) of bmIntersectMask: blendIntersectMask(dest, src)
of bmExcludeMask: blendExcludeMask(dest, src) of bmExcludeMask: blendExcludeMask(dest, src)
proc mixStatic*(blendMode: static[BlendMode], dest, src: ColorRGBA): ColorRGBA {.inline.} = proc mixStatic*(blendMode: static[BlendMode], dest, src: ColorRGBA): ColorRGBA {.inline.} =
when blendMOde == bmNormal: blendNormal(dest, src) when blendMOde == bmNormal: blendNormal(dest, src)
elif blendMOde == bmDarken: blendDarken(dest, src) elif blendMOde == bmDarken: blendDarken(dest, src)

View file

@ -6,8 +6,8 @@ type
width*, height*: int width*, height*: int
data*: seq[ColorRGBA] data*: seq[ColorRGBA]
proc draw*(a: Image, b: Image, mat: Mat3, blendMode = bmNormal) proc draw*(a, b: Image, mat: Mat3, blendMode = bmNormal)
proc draw*(a: Image, b: Image, pos = vec2(0, 0), blendMode = bmNormal) proc draw*(a, b: Image, pos = vec2(0, 0), blendMode = bmNormal) {.inline.}
proc newImage*(width, height: int): Image = proc newImage*(width, height: int): Image =
## Creates a new image with appropriate dimensions. ## Creates a new image with appropriate dimensions.
@ -47,8 +47,7 @@ proc `$`*(image: Image): string =
proc inside*(image: Image, x, y: int): bool {.inline.} = proc inside*(image: Image, x, y: int): bool {.inline.} =
## Returns true if (x, y) is inside the image. ## Returns true if (x, y) is inside the image.
x >= 0 and x < image.width and x >= 0 and x < image.width and y >= 0 and y < image.height
y >= 0 and y < image.height
proc inside1px*(image: Image, x, y: float): bool {.inline.} = proc inside1px*(image: Image, x, y: float): bool {.inline.} =
## Returns true if (x, y) is inside the image. ## Returns true if (x, y) is inside the image.
@ -66,7 +65,7 @@ proc getRgbaUnsafe*(image: Image, x, y: int): ColorRGBA {.inline.} =
proc getAddr*(image: Image, x, y: int): pointer {.inline.} = proc getAddr*(image: Image, x, y: int): pointer {.inline.} =
## Gets a address of the color from (x, y) coordinates. ## Gets a address of the color from (x, y) coordinates.
## Unsafe make sure x, y are in bounds. ## Unsafe make sure x, y are in bounds.
addr image.data[image.width * y + x] image.data[image.width * y + x].addr
proc `[]`*(image: Image, x, y: int): ColorRGBA {.inline.} = proc `[]`*(image: Image, x, y: int): ColorRGBA {.inline.} =
## Gets a pixel at (x, y) or returns transparent black if outside of bounds. ## Gets a pixel at (x, y) or returns transparent black if outside of bounds.
@ -85,19 +84,17 @@ proc `[]=`*(image: Image, x, y: int, rgba: ColorRGBA) {.inline.} =
if image.inside(x, y): if image.inside(x, y):
image.setRgbaUnsafe(x, y, rgba) image.setRgbaUnsafe(x, y, rgba)
proc newImageFill*(width, height: int, rgba: ColorRgba): Image =
## Fills the image with a solid color.
result = newImageNoInit(width, height)
for y in 0 ..< result.height:
for x in 0 ..< result.width:
result.setRgbaUnsafe(x, y, rgba)
proc fill*(image: Image, rgba: ColorRgba) = proc fill*(image: Image, rgba: ColorRgba) =
## Fills the image with a solid color. ## Fills the image with a solid color.
for y in 0 ..< image.height: for y in 0 ..< image.height:
for x in 0 ..< image.width: for x in 0 ..< image.width:
image.setRgbaUnsafe(x, y, rgba) image.setRgbaUnsafe(x, y, rgba)
proc newImageFill*(width, height: int, rgba: ColorRgba): Image =
## Fills the image with a solid color.
result = newImageNoInit(width, height)
result.fill(rgba)
proc invert*(image: Image) = proc invert*(image: Image) =
## Inverts all of the colors and alpha. ## Inverts all of the colors and alpha.
for y in 0 ..< image.height: for y in 0 ..< image.height:
@ -222,25 +219,27 @@ proc getRgbaSmooth*(image: Image, x, y: float32): ColorRGBA {.inline.} =
return finalMix.fromAlphy().rgba() return finalMix.fromAlphy().rgba()
proc hasEffect*(blendMode: BlendMode, rgba: ColorRGBA): bool = # @andre: unused, delete?
## Returns true if applying rgba with current blend mode has effect.
case blendMode
of bmMask:
rgba.a != 255
of bmOverwrite:
true
of bmIntersectMask:
true
else:
rgba.a > 0
proc allowCopy*(blendMode: BlendMode): bool = # proc hasEffect*(blendMode: BlendMode, rgba: ColorRGBA): bool =
## Returns true if applying rgba with current blend mode has effect. # ## Returns true if applying rgba with current blend mode has effect.
case blendMode # case blendMode
of bmIntersectMask: # of bmMask:
false # rgba.a != 255
else: # of bmOverwrite:
true # true
# of bmIntersectMask:
# true
# else:
# rgba.a > 0
# proc allowCopy*(blendMode: BlendMode): bool =
# ## Returns true if applying rgba with current blend mode has effect.
# case blendMode
# of bmIntersectMask:
# false
# else:
# true
proc resize*(srcImage: Image, width, height: int): Image = proc resize*(srcImage: Image, width, height: int): Image =
result = newImage(width, height) result = newImage(width, height)
@ -252,7 +251,6 @@ proc resize*(srcImage: Image, width, height: int): Image =
)) ))
) )
proc blur*(image: Image, radius: float32): Image = proc blur*(image: Image, radius: float32): Image =
## Applies Gaussian blur to the image given a radius. ## Applies Gaussian blur to the image given a radius.
let radius = (radius).int let radius = (radius).int
@ -369,21 +367,18 @@ proc spread*(image: Image, spread: float32): Image =
var maxAlpha = 0.uint8 var maxAlpha = 0.uint8
for bx in -spread.int .. spread.int: for bx in -spread.int .. spread.int:
for by in -spread.int .. spread.int: for by in -spread.int .. spread.int:
#if vec2(bx.float32, by.float32).length < spread: # if vec2(bx.float32, by.float32).length < spread:
let alpha = image[x + bx, y + by].a let alpha = image[x + bx, y + by].a
if alpha > maxAlpha: if alpha > maxAlpha:
maxAlpha = alpha maxAlpha = alpha
if maxAlpha == 255: if maxAlpha == 255:
break break
if maxAlpha == 255:
break
result[x, y] = rgba(0, 0, 0, maxAlpha) result[x, y] = rgba(0, 0, 0, maxAlpha)
proc shadow*( proc shadow*(
mask: Image, mask: Image,
offset: Vec2, offset: Vec2,
spread: float32, spread, blur: float32,
blur: float32,
color: ColorRGBA color: ColorRGBA
): Image = ): Image =
## Create a shadow of the image with the offset, spread and blur. ## Create a shadow of the image with the offset, spread and blur.
@ -400,11 +395,11 @@ proc shadow*(
proc applyOpacity*(image: Image, opacity: float32): Image = proc applyOpacity*(image: Image, opacity: float32): Image =
## Multiplies alpha of the image by opacity. ## Multiplies alpha of the image by opacity.
result = newImageNoInit(image.width, image.height) result = newImageNoInit(image.width, image.height)
let op = (255 * opacity).uint8 let op = (255 * opacity).uint32
for y in 0 ..< image.height: for y in 0 ..< image.height:
for x in 0 ..< image.width: for x in 0 ..< image.width:
var rgba = image.getRgbaUnsafe(x, y) var rgba = image.getRgbaUnsafe(x, y)
rgba.a = ((rgba.a.uint32 * op.uint32) div 255).clamp(0, 255).uint8 rgba.a = ((rgba.a.uint32 * op) div 255).clamp(0, 255).uint8
result.setRgbaUnsafe(x, y, rgba) result.setRgbaUnsafe(x, y, rgba)
proc sharpOpacity*(image: Image): Image = proc sharpOpacity*(image: Image): Image =
@ -459,7 +454,6 @@ proc drawUberStatic(
inPlace: static[bool], inPlace: static[bool],
smooth: static[bool], smooth: static[bool],
) = ) =
for y in 0 ..< a.height: for y in 0 ..< a.height:
var var
xMin = 0 xMin = 0
@ -513,13 +507,12 @@ proc drawUberStatic(
zeroMem(c.getAddr(xMax, y), 4*(a.width - xMax)) zeroMem(c.getAddr(xMax, y), 4*(a.width - xMax))
else: else:
when not inPlace: when not inPlace:
#for x in xMax ..< a.width: # for x in xMax ..< a.width:
# result.setRgbaUnsafe(x, y, a.getRgbaUnsafe(x, y)) # result.setRgbaUnsafe(x, y, a.getRgbaUnsafe(x, y))
if a.width - xMax > 0: if a.width - xMax > 0:
copyMem(c.getAddr(xMax, y), a.getAddr(xMax, y), 4*(a.width - xMax)) copyMem(c.getAddr(xMax, y), a.getAddr(xMax, y), 4*(a.width - xMax))
proc draw*(a, b: Image, mat: Mat3, blendMode: BlendMode) =
proc draw*(a: Image, b: Image, mat: Mat3, blendMode: BlendMode) =
## Draws one image onto another using matrix with color blending. ## Draws one image onto another using matrix with color blending.
var var
@ -558,7 +551,6 @@ proc draw*(a: Image, b: Image, mat: Mat3, blendMode: BlendMode) =
mat[2, 0].fractional == 0.0 and mat[2, 1].fractional == 0.0) mat[2, 0].fractional == 0.0 and mat[2, 1].fractional == 0.0)
if not smooth: if not smooth:
#echo "draw ", b.width, "x", b.height, " fast"
case blendMode case blendMode
of bmNormal: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmNormal, true, false) of bmNormal: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmNormal, true, false)
of bmDarken: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmDarken, true, false) of bmDarken: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmDarken, true, false)
@ -584,7 +576,6 @@ proc draw*(a: Image, b: Image, mat: Mat3, blendMode: BlendMode) =
of bmIntersectMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmIntersectMask, true, false) of bmIntersectMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmIntersectMask, true, false)
of bmExcludeMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmExcludeMask, true, false) of bmExcludeMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmExcludeMask, true, false)
else: else:
#echo "draw ", b.width, "x", b.height, " smooth"
case blendMode case blendMode
of bmNormal: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmNormal, true, true) of bmNormal: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmNormal, true, true)
of bmDarken: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmDarken, true, true) of bmDarken: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmDarken, true, true)
@ -610,5 +601,7 @@ proc draw*(a: Image, b: Image, mat: Mat3, blendMode: BlendMode) =
of bmIntersectMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmIntersectMask, true, true) of bmIntersectMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmIntersectMask, true, true)
of bmExcludeMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmExcludeMask, true, true) of bmExcludeMask: drawUberStatic(a, b, c, start, stepX, stepY, lines, bmExcludeMask, true, true)
proc draw*(a: Image, b: Image, pos = vec2(0, 0), blendMode = bmNormal) = proc draw*(
a, b: Image, pos = vec2(0, 0), blendMode = bmNormal
) {.inline.} =
a.draw(b, translate(pos), blendMode) a.draw(b, translate(pos), blendMode)

View file

@ -50,7 +50,7 @@ proc `[]=`*(mask: Mask, x, y: int, value: uint8) {.inline.} =
mask.setUnsafe(x, y, value) mask.setUnsafe(x, y, value)
proc fill*(mask: Mask, value: uint8) = proc fill*(mask: Mask, value: uint8) =
## Fills the mask with the paramter value. ## Fills the mask with the parameter value.
for i in 0 ..< mask.data.len: for i in 0 ..< mask.data.len:
mask.data[i] = value mask.data[i] = value

View file

@ -132,7 +132,7 @@ proc parsePath*(path: string): Path =
number = $c number = $c
of ' ', ',', '\r', '\n', '\t': of ' ', ',', '\r', '\n', '\t':
finishDigit() finishDigit()
else: # TODO: ? else: # TODO: still needed?
if command == Move and numbers.len == 2: if command == Move and numbers.len == 2:
finishCommand() finishCommand()
command = Line command = Line