Merge pull request #150 from guzba/master

1.0.5 converters, morepretty
This commit is contained in:
treeform 2021-03-03 20:14:38 -08:00 committed by GitHub
commit 4a9d1297db
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 63 additions and 55 deletions

View file

@ -24,7 +24,7 @@ var a = newImage(1000, 1000)
a.fill(rgba(0, 0, 0, 255)) a.fill(rgba(0, 0, 0, 255))
timeIt "pixie": timeIt "pixie":
var p: paths.Path var p: pixie.Path
p.moveTo(0, 0) p.moveTo(0, 0)
p.lineTo(500, 0) p.lineTo(500, 0)
p.lineTo(500, 500) p.lineTo(500, 500)

View file

@ -492,7 +492,7 @@ proc decodeSvg*(data: string, width = 0, height = 0): Image =
let let
bgra = pixels[result.dataIndex(x, y)] bgra = pixels[result.dataIndex(x, y)]
rgba = rgba(bgra[2], bgra[1], bgra[0], bgra[3]) rgba = rgba(bgra[2], bgra[1], bgra[0], bgra[3])
result.setRgbaUnsafe(x, y, rgba) result.setRgbaUnsafe(x, y, rgba.rgbx())
except PixieError as e: except PixieError as e:
raise e raise e
except: except:

View file

@ -1,4 +1,4 @@
version = "1.0.4" version = "1.0.5"
author = "Andre von Houck and Ryan Oldenburg" author = "Andre von Houck and Ryan Oldenburg"
description = "Full-featured 2d graphics library for Nim." description = "Full-featured 2d graphics library for Nim."
license = "MIT" license = "MIT"
@ -7,7 +7,7 @@ srcDir = "src"
requires "nim >= 1.2.6" requires "nim >= 1.2.6"
requires "vmath >= 0.4.0" requires "vmath >= 0.4.0"
requires "chroma >= 0.2.3" requires "chroma >= 0.2.4"
requires "zippy >= 0.3.5" requires "zippy >= 0.3.5"
requires "flatty >= 0.1.3" requires "flatty >= 0.1.3"
requires "nimsimd >= 1.0.0" requires "nimsimd >= 1.0.0"

View file

@ -1,6 +1,7 @@
import bumpy, chroma, flatty/binny, os, pixie/blends, pixie/common, import bumpy, chroma, flatty/binny, os, pixie/blends, pixie/common,
pixie/fileformats/bmp, pixie/fileformats/gif, pixie/fileformats/jpg, pixie/fileformats/png, pixie/fileformats/bmp, pixie/fileformats/gif, pixie/fileformats/jpg,
pixie/fileformats/svg, pixie/images, pixie/masks, pixie/paints, pixie/paths, vmath pixie/fileformats/png, pixie/fileformats/svg, pixie/images, pixie/masks,
pixie/paints, pixie/paths, vmath
export blends, bumpy, chroma, common, images, masks, paints, paths, vmath export blends, bumpy, chroma, common, images, masks, paints, paths, vmath
@ -8,6 +9,14 @@ type
FileFormat* = enum FileFormat* = enum
ffPng, ffBmp, ffJpg, ffGif ffPng, ffBmp, ffJpg, ffGif
converter autoStraightAlpha*(c: ColorRGBX): ColorRGBA {.inline.} =
## Convert a paremultiplied alpha RGBA to a straight alpha RGBA.
c.rgba()
converter autoPremultipliedAlpha*(c: ColorRGBA): ColorRGBX {.inline.} =
## Convert a straight alpha RGBA to a premultiplied alpha RGBA.
c.rgbx()
proc decodeImage*(data: string | seq[uint8]): Image = proc decodeImage*(data: string | seq[uint8]): Image =
## Loads an image from a memory. ## Loads an image from a memory.
if data.len > 8 and data.readUint64(0) == cast[uint64](pngSignature): if data.len > 8 and data.readUint64(0) == cast[uint64](pngSignature):

View file

@ -221,8 +221,8 @@ proc blendMultiply(backdrop, source: ColorRGBX): ColorRGBX =
proc blendColorBurn(backdrop, source: ColorRGBX): ColorRGBX = proc blendColorBurn(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha() backdrop = backdrop.rgba()
source = source.toStraightAlpha() source = source.rgba()
proc blend(backdrop, source: uint32): uint8 {.inline.} = proc blend(backdrop, source: uint32): uint8 {.inline.} =
if backdrop == 255: if backdrop == 255:
255.uint8 255.uint8
@ -234,7 +234,7 @@ proc blendColorBurn(backdrop, source: ColorRGBX): ColorRGBX =
blended.r = blend(backdrop.r, source.r) blended.r = blend(backdrop.r, source.r)
blended.g = blend(backdrop.g, source.g) blended.g = blend(backdrop.g, source.g)
blended.b = blend(backdrop.b, source.b) blended.b = blend(backdrop.b, source.b)
result = alphaFix(backdrop, source, blended).toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgbx()
proc blendLighten(backdrop, source: ColorRGBX): ColorRGBX = proc blendLighten(backdrop, source: ColorRGBX): ColorRGBX =
proc blend( proc blend(
@ -268,8 +268,8 @@ proc blendScreen(backdrop, source: ColorRGBX): ColorRGBX =
proc blendColorDodge(backdrop, source: ColorRGBX): ColorRGBX = proc blendColorDodge(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha() backdrop = backdrop.rgba()
source = source.toStraightAlpha() source = source.rgba()
proc blend(backdrop, source: uint32): uint8 {.inline.} = proc blend(backdrop, source: uint32): uint8 {.inline.} =
if backdrop == 0: if backdrop == 0:
0.uint8 0.uint8
@ -281,7 +281,7 @@ proc blendColorDodge(backdrop, source: ColorRGBX): ColorRGBX =
blended.r = blend(backdrop.r, source.r) blended.r = blend(backdrop.r, source.r)
blended.g = blend(backdrop.g, source.g) blended.g = blend(backdrop.g, source.g)
blended.b = blend(backdrop.b, source.b) blended.b = blend(backdrop.b, source.b)
result = alphaFix(backdrop, source, blended).toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgbx()
proc blendOverlay(backdrop, source: ColorRGBX): ColorRGBX = proc blendOverlay(backdrop, source: ColorRGBX): ColorRGBX =
result.r = hardLight(source.r, source.a, backdrop.r, backdrop.a) result.r = hardLight(source.r, source.a, backdrop.r, backdrop.a)
@ -298,8 +298,8 @@ proc blendSoftLight(backdrop, source: ColorRGBX): ColorRGBX =
# ).uint8 # ).uint8
let let
backdrop = backdrop.toStraightAlpha() backdrop = backdrop.rgba()
source = source.toStraightAlpha() source = source.rgba()
var rgba: ColorRGBA var rgba: ColorRGBA
when defined(amd64) and not defined(pixieNoSimd): when defined(amd64) and not defined(pixieNoSimd):
@ -361,7 +361,7 @@ proc blendSoftLight(backdrop, source: ColorRGBX): ColorRGBX =
blended = alphaFix(b, s, blended) blended = alphaFix(b, s, blended)
rgba = blended.rgba rgba = blended.rgba
result = rgba result = rgba.rgbx()
proc blendHardLight(backdrop, source: ColorRGBX): ColorRGBX = proc blendHardLight(backdrop, source: ColorRGBX): ColorRGBX =
result.r = hardLight(backdrop.r, backdrop.a, source.r, source.a) result.r = hardLight(backdrop.r, backdrop.a, source.r, source.a)
@ -396,31 +396,31 @@ proc blendExclusion(backdrop, source: ColorRGBX): ColorRGBX =
proc blendColor(backdrop, source: ColorRGBX): ColorRGBX = proc blendColor(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha().color backdrop = backdrop.rgba().color
source = source.toStraightAlpha().color source = source.rgba().color
blended = SetLum(source, Lum(backdrop)) blended = SetLum(source, Lum(backdrop))
result = alphaFix(backdrop, source, blended).rgba.toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgba.rgbx()
proc blendLuminosity(backdrop, source: ColorRGBX): ColorRGBX = proc blendLuminosity(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha().color backdrop = backdrop.rgba().color
source = source.toStraightAlpha().color source = source.rgba().color
blended = SetLum(backdrop, Lum(source)) blended = SetLum(backdrop, Lum(source))
result = alphaFix(backdrop, source, blended).rgba.toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgba.rgbx()
proc blendHue(backdrop, source: ColorRGBX): ColorRGBX = proc blendHue(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha().color backdrop = backdrop.rgba().color
source = source.toStraightAlpha().color source = source.rgba().color
blended = SetLum(SetSat(source, Sat(backdrop)), Lum(backdrop)) blended = SetLum(SetSat(source, Sat(backdrop)), Lum(backdrop))
result = alphaFix(backdrop, source, blended).rgba.toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgba.rgbx()
proc blendSaturation(backdrop, source: ColorRGBX): ColorRGBX = proc blendSaturation(backdrop, source: ColorRGBX): ColorRGBX =
let let
backdrop = backdrop.toStraightAlpha().color backdrop = backdrop.rgba().color
source = source.toStraightAlpha().color source = source.rgba().color
blended = SetLum(SetSat(backdrop, Sat(source)), Lum(backdrop)) blended = SetLum(SetSat(backdrop, Sat(source)), Lum(backdrop))
result = alphaFix(backdrop, source, blended).rgba.toPremultipliedAlpha() result = alphaFix(backdrop, source, blended).rgba.rgbx()
proc blendMask(backdrop, source: ColorRGBX): ColorRGBX = proc blendMask(backdrop, source: ColorRGBX): ColorRGBX =
let k = source.a.uint32 let k = source.a.uint32

View file

@ -46,7 +46,7 @@ proc decodeBmp*(data: string): Image =
rgba.b = data.readUint8(offset + 0) rgba.b = data.readUint8(offset + 0)
rgba.a = 255 rgba.a = 255
offset += 3 offset += 3
result[x, result.height - y - 1] = rgba result[x, result.height - y - 1] = rgba.rgbx()
proc decodeBmp*(data: seq[uint8]): Image {.inline.} = proc decodeBmp*(data: seq[uint8]): Image {.inline.} =
## Decodes bitmap data into an Image. ## Decodes bitmap data into an Image.
@ -84,7 +84,7 @@ proc encodeBmp*(image: Image): string =
for y in 0 ..< image.height: for y in 0 ..< image.height:
for x in 0 ..< image.width: for x in 0 ..< image.width:
let rgba = image[x, image.height - y - 1].toStraightAlpha() let rgba = image[x, image.height - y - 1].rgba()
result.addUint8(rgba.r) result.addUint8(rgba.r)
result.addUint8(rgba.g) result.addUint8(rgba.g)
result.addUint8(rgba.b) result.addUint8(rgba.b)

View file

@ -1,4 +1,4 @@
import chroma, flatty/binny, pixie/common, pixie/images, math, zippy/bitstreams import chroma, flatty/binny, math, pixie/common, pixie/images, zippy/bitstreams
const gifSignatures* = @["GIF87a", "GIF89a"] const gifSignatures* = @["GIF87a", "GIF89a"]
@ -158,7 +158,7 @@ proc decodeGif*(data: string): Image =
# Convert color indexes into real colors. # Convert color indexes into real colors.
for j, idx in colorIndexes: for j, idx in colorIndexes:
if idx >= colors.len or j >= result.data.len: failInvalid() if idx >= colors.len or j >= result.data.len: failInvalid()
result.data[j] = colors[idx] result.data[j] = colors[idx].rgbx()
of 0x21: # Read EXTENSION block. of 0x21: # Read EXTENSION block.
# Skip over all extensions (mostly animation information). # Skip over all extensions (mostly animation information).

View file

@ -1,5 +1,5 @@
import chroma, flatty/binny, math, pixie/common, pixie/images, pixie/masks, import chroma, flatty/binny, math, pixie/common, pixie/images, pixie/internal,
zippy, zippy/crc, pixie/internal pixie/masks, zippy, zippy/crc
# See http://www.libpng.org/pub/png/spec/1.2/PNG-Contents.html # See http://www.libpng.org/pub/png/spec/1.2/PNG-Contents.html

View file

@ -9,7 +9,7 @@ const
type Ctx = object type Ctx = object
fillRule: WindingRule fillRule: WindingRule
fill, stroke: ColorRGBA fill, stroke: ColorRGBX
strokeWidth: float32 strokeWidth: float32
strokeLineCap: LineCap strokeLineCap: LineCap
strokeLineJoin: LineJoin strokeLineJoin: LineJoin
@ -25,8 +25,8 @@ proc attrOrDefault(node: XmlNode, name, default: string): string =
result = default result = default
proc initCtx(): Ctx = proc initCtx(): Ctx =
result.fill = parseHtmlColor("black").rgba result.fill = parseHtmlColor("black").rgbx
result.stroke = parseHtmlColor("black").rgba result.stroke = parseHtmlColor("black").rgbx
result.strokeWidth = 1 result.strokeWidth = 1
result.transform = mat3() result.transform = mat3()
@ -79,18 +79,18 @@ proc decodeCtx(inherited: Ctx, node: XmlNode): Ctx =
if fill == "" or fill == "currentColor": if fill == "" or fill == "currentColor":
discard # Inherit discard # Inherit
elif fill == "none": elif fill == "none":
result.fill = ColorRGBA() result.fill = ColorRGBX()
else: else:
result.fill = parseHtmlColor(fill).rgba result.fill = parseHtmlColor(fill).rgbx
if stroke == "": if stroke == "":
discard # Inherit discard # Inherit
elif stroke == "currentColor": elif stroke == "currentColor":
result.shouldStroke = true result.shouldStroke = true
elif stroke == "none": elif stroke == "none":
result.stroke = ColorRGBA() result.stroke = ColorRGBX()
else: else:
result.stroke = parseHtmlColor(stroke).rgba result.stroke = parseHtmlColor(stroke).rgbx
result.shouldStroke = true result.shouldStroke = true
if strokeWidth == "": if strokeWidth == "":
@ -101,7 +101,7 @@ proc decodeCtx(inherited: Ctx, node: XmlNode): Ctx =
result.strokeWidth = parseFloat(strokeWidth) result.strokeWidth = parseFloat(strokeWidth)
result.shouldStroke = true result.shouldStroke = true
if result.stroke == ColorRGBA() or result.strokeWidth <= 0: if result.stroke == ColorRGBX() or result.strokeWidth <= 0:
result.shouldStroke = false result.shouldStroke = false
if strokeLineCap == "": if strokeLineCap == "":
@ -216,7 +216,7 @@ proc draw(img: Image, node: XmlNode, ctxStack: var seq[Ctx]) =
d = node.attr("d") d = node.attr("d")
ctx = decodeCtx(ctxStack[^1], node) ctx = decodeCtx(ctxStack[^1], node)
path = parsePath(d) path = parsePath(d)
if ctx.fill != ColorRGBA(): if ctx.fill != ColorRGBX():
img.fillPath(path, ctx.fill, ctx.transform, ctx.fillRule) img.fillPath(path, ctx.fill, ctx.transform, ctx.fillRule)
if ctx.shouldStroke: if ctx.shouldStroke:
img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth) img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth)
@ -234,7 +234,7 @@ proc draw(img: Image, node: XmlNode, ctxStack: var seq[Ctx]) =
path.lineTo(x2, y2) path.lineTo(x2, y2)
path.closePath() path.closePath()
if ctx.fill != ColorRGBA(): if ctx.fill != ColorRGBX():
img.fillPath(path, ctx.fill, ctx.transform) img.fillPath(path, ctx.fill, ctx.transform)
if ctx.shouldStroke: if ctx.shouldStroke:
img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth) img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth)
@ -270,7 +270,7 @@ proc draw(img: Image, node: XmlNode, ctxStack: var seq[Ctx]) =
if node.tag == "polygon": if node.tag == "polygon":
path.closePath() path.closePath()
if ctx.fill != ColorRGBA(): if ctx.fill != ColorRGBX():
img.fillPath(path, ctx.fill, ctx.transform) img.fillPath(path, ctx.fill, ctx.transform)
if ctx.shouldStroke: if ctx.shouldStroke:
img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth) img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth)
@ -308,7 +308,7 @@ proc draw(img: Image, node: XmlNode, ctxStack: var seq[Ctx]) =
else: else:
path.rect(x, y, width, height) path.rect(x, y, width, height)
if ctx.fill != ColorRGBA(): if ctx.fill != ColorRGBX():
img.fillPath(path, ctx.fill, ctx.transform) img.fillPath(path, ctx.fill, ctx.transform)
if ctx.shouldStroke: if ctx.shouldStroke:
img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth) img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth)
@ -330,7 +330,7 @@ proc draw(img: Image, node: XmlNode, ctxStack: var seq[Ctx]) =
var path: Path var path: Path
path.ellipse(cx, cy, rx, ry) path.ellipse(cx, cy, rx, ry)
if ctx.fill != ColorRGBA(): if ctx.fill != ColorRGBX():
img.fillPath(path, ctx.fill, ctx.transform) img.fillPath(path, ctx.fill, ctx.transform)
if ctx.shouldStroke: if ctx.shouldStroke:
img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth) img.strokePath(path, ctx.stroke, ctx.transform, ctx.strokeWidth)

View file

@ -31,13 +31,12 @@ proc toLineSpace(at, to, point: Vec2): float32 =
## Convert position on to where it would fall on a line between at and to. ## Convert position on to where it would fall on a line between at and to.
let let
d = to - at d = to - at
det = d.x*d.x + d.y*d.y det = d.x * d.x + d.y * d.y
return (d.y*(point.y-at.y)+d.x*(point.x-at.x))/det return (d.y * (point.y - at.y) + d.x * (point.x - at.x)) / det
proc gradientPut(image: Image, x, y: int, a: float32, stops: seq[ColorStop]) = proc gradientPut(image: Image, x, y: int, a: float32, stops: seq[ColorStop]) =
## Put an gradient color based on the "a" - were are we related to a line. ## Put an gradient color based on the "a" - were are we related to a line.
var var index = -1
index = -1
for i, stop in stops: for i, stop in stops:
if stop.position < a: if stop.position < a:
index = i index = i
@ -59,7 +58,7 @@ proc gradientPut(image: Image, x, y: int, a: float32, stops: seq[ColorStop]) =
gs2.color.color, gs2.color.color,
(a - gs1.position) / (gs2.position - gs1.position) (a - gs1.position) / (gs2.position - gs1.position)
) )
image.setRgbaUnsafe(x, y, color.rgba.toPremultipliedAlpha()) image.setRgbaUnsafe(x, y, color.rgba.rgbx())
proc fillLinearGradient*( proc fillLinearGradient*(
image: Image, image: Image,

View file

@ -24,8 +24,8 @@ timeIt "subImage":
reset() reset()
# timeIt "superImage": timeIt "superImage":
# discard keep image.superImage(-10, -10, 2580, 1460)
reset() reset()

View file

@ -1,4 +1,4 @@
import pixie/fileformats/gif, pixie import pixie, pixie/fileformats/gif
var img = decodeGIF(readFile("tests/images/gif/3x5.gif")) var img = decodeGIF(readFile("tests/images/gif/3x5.gif"))
img.writeFile("tests/images/gif/3x5.png") img.writeFile("tests/images/gif/3x5.png")

View file

@ -1,4 +1,4 @@
import chroma, pixie, vmath, pixie/internal import chroma, pixie, pixie/internal, vmath
block: block:
let image = newImage(10, 10) let image = newImage(10, 10)