more binding prep
This commit is contained in:
parent
5868034f08
commit
412baf6c19
6 changed files with 72 additions and 79 deletions
|
@ -505,9 +505,9 @@ proc circle*(ctx: Context, cx, cy, r: float32) {.inline.} =
|
|||
## Adds a circle to the current path.
|
||||
ctx.path.circle(cx, cy, r)
|
||||
|
||||
proc circle*(ctx: Context, center: Vec2, r: float32) {.inline.} =
|
||||
proc circle*(ctx: Context, circle: Circle) {.inline.} =
|
||||
## Adds a circle to the current path.
|
||||
ctx.path.circle(center, r)
|
||||
ctx.path.circle(circle)
|
||||
|
||||
proc polygon*(ctx: Context, x, y, size: float32, sides: int) {.inline.} =
|
||||
## Adds an n-sided regular polygon at (x, y) of size to the current path.
|
||||
|
@ -563,7 +563,7 @@ proc strokeEllipse*(ctx: Context, center: Vec2, rx, ry: float32) =
|
|||
proc fillCircle*(ctx: Context, circle: Circle) =
|
||||
## Draws a circle that is filled according to the current fillStyle
|
||||
let path = newPath()
|
||||
path.circle(circle.pos, circle.radius)
|
||||
path.circle(circle)
|
||||
ctx.fill(path)
|
||||
|
||||
proc fillCircle*(ctx: Context, center: Vec2, radius: float32) =
|
||||
|
@ -576,7 +576,7 @@ proc strokeCircle*(ctx: Context, circle: Circle) =
|
|||
## Draws a circle that is stroked (outlined) according to the current
|
||||
## strokeStyle and other context settings.
|
||||
let path = newPath()
|
||||
path.circle(circle.pos, circle.radius)
|
||||
path.circle(circle)
|
||||
ctx.stroke(path)
|
||||
|
||||
proc strokeCircle*(ctx: Context, center: Vec2, radius: float32) =
|
||||
|
|
|
@ -437,7 +437,7 @@ proc parseSvgFont*(buf: string): Typeface =
|
|||
proc textUber(
|
||||
target: Image | Mask,
|
||||
arrangement: Arrangement,
|
||||
transform: Vec2 | Mat3 = vec2(0, 0),
|
||||
transform = mat3(),
|
||||
strokeWidth = 1.0,
|
||||
lineCap = lcButt,
|
||||
lineJoin = ljMiter,
|
||||
|
@ -518,7 +518,7 @@ proc textUber(
|
|||
proc fillText*(
|
||||
target: Image | Mask,
|
||||
arrangement: Arrangement,
|
||||
transform: Vec2 | Mat3 = vec2(0, 0)
|
||||
transform = mat3()
|
||||
) {.inline.} =
|
||||
## Fills the text arrangement.
|
||||
textUber(
|
||||
|
@ -531,7 +531,7 @@ proc fillText*(
|
|||
target: Image | Mask,
|
||||
font: Font,
|
||||
text: string,
|
||||
transform: Vec2 | Mat3 = vec2(0, 0),
|
||||
transform = mat3(),
|
||||
bounds = vec2(0, 0),
|
||||
hAlign = haLeft,
|
||||
vAlign = vaTop
|
||||
|
@ -546,7 +546,7 @@ proc fillText*(
|
|||
proc strokeText*(
|
||||
target: Image | Mask,
|
||||
arrangement: Arrangement,
|
||||
transform: Vec2 | Mat3 = vec2(0, 0),
|
||||
transform = mat3(),
|
||||
strokeWidth = 1.0,
|
||||
lineCap = lcButt,
|
||||
lineJoin = ljMiter,
|
||||
|
@ -570,7 +570,7 @@ proc strokeText*(
|
|||
target: Image | Mask,
|
||||
font: Font,
|
||||
text: string,
|
||||
transform: Vec2 | Mat3 = vec2(0, 0),
|
||||
transform = mat3(),
|
||||
strokeWidth = 1.0,
|
||||
bounds = vec2(0, 0),
|
||||
hAlign = haLeft,
|
||||
|
|
|
@ -815,7 +815,7 @@ proc drawUber(a, b: Image | Mask, mat = mat3(), blendMode = bmNormal) =
|
|||
zeroMem(a.data[a.dataIndex(xMax, y)].addr, 4 * (a.width - xMax))
|
||||
|
||||
proc draw*(
|
||||
a, b: Image, transform: Mat3 = mat3(), blendMode = bmNormal
|
||||
a, b: Image, transform = mat3(), blendMode = bmNormal
|
||||
) {.inline.} =
|
||||
## Draws one image onto another using matrix with color blending.
|
||||
when type(transform) is Vec2:
|
||||
|
@ -824,7 +824,7 @@ proc draw*(
|
|||
a.drawUber(b, transform, blendMode)
|
||||
|
||||
proc draw*(
|
||||
a, b: Mask, transform: Mat3 = mat3(), blendMode = bmMask
|
||||
a, b: Mask, transform = mat3(), blendMode = bmMask
|
||||
) {.inline.} =
|
||||
## Draws a mask onto a mask using a matrix with color blending.
|
||||
when type(transform) is Vec2:
|
||||
|
@ -833,7 +833,7 @@ proc draw*(
|
|||
a.drawUber(b, transform, blendMode)
|
||||
|
||||
proc draw*(
|
||||
image: Image, mask: Mask, transform: Mat3 = mat3(), blendMode = bmMask
|
||||
image: Image, mask: Mask, transform = mat3(), blendMode = bmMask
|
||||
) {.inline.} =
|
||||
## Draws a mask onto an image using a matrix with color blending.
|
||||
when type(transform) is Vec2:
|
||||
|
@ -842,7 +842,7 @@ proc draw*(
|
|||
image.drawUber(mask, transform, blendMode)
|
||||
|
||||
proc draw*(
|
||||
mask: Mask, image: Image, transform: Mat3 = mat3(), blendMode = bmMask
|
||||
mask: Mask, image: Image, transform = mat3(), blendMode = bmMask
|
||||
) {.inline.} =
|
||||
## Draws a image onto a mask using a matrix with color blending.
|
||||
when type(transform) is Vec2:
|
||||
|
|
|
@ -17,13 +17,13 @@ type
|
|||
## Line join type for strokes.
|
||||
ljMiter, ljRound, ljBevel
|
||||
|
||||
PathCommandKind* = enum
|
||||
PathCommandKind = enum
|
||||
## Type of path commands
|
||||
Close,
|
||||
Move, Line, HLine, VLine, Cubic, SCubic, Quad, TQuad, Arc,
|
||||
RMove, RLine, RHLine, RVLine, RCubic, RSCubic, RQuad, RTQuad, RArc
|
||||
|
||||
PathCommand* = object
|
||||
PathCommand = object
|
||||
## Binary version of an SVG command.
|
||||
kind: PathCommandKind
|
||||
numbers: seq[float32]
|
||||
|
@ -50,15 +50,12 @@ proc newPath*(): Path =
|
|||
## Create a new Path.
|
||||
Path()
|
||||
|
||||
proc pixelScale(transform: Vec2 | Mat3): float32 =
|
||||
proc pixelScale(transform: Mat3): float32 =
|
||||
## What is the largest scale factor of this transform?
|
||||
when type(transform) is Vec2:
|
||||
return 1.0
|
||||
else:
|
||||
max(
|
||||
vec2(transform[0, 0], transform[0, 1]).length,
|
||||
vec2(transform[1, 0], transform[1, 1]).length
|
||||
)
|
||||
max(
|
||||
vec2(transform[0, 0], transform[0, 1]).length,
|
||||
vec2(transform[1, 0], transform[1, 1]).length
|
||||
)
|
||||
|
||||
proc isRelative(kind: PathCommandKind): bool =
|
||||
kind in {
|
||||
|
@ -507,12 +504,6 @@ proc rect*(path: Path, x, y, w, h: float32, clockwise = true) =
|
|||
path.lineTo(x + w, y)
|
||||
path.closePath()
|
||||
|
||||
proc rect*(path: Path, pos: Vec2, wh: Vec2, clockwise = true) {.inline.} =
|
||||
## Adds a rectangle.
|
||||
## Clockwise param can be used to subtract a rect from a path when using
|
||||
## even-odd winding rule.
|
||||
path.rect(pos.x, pos.y, wh.x, wh.y, clockwise)
|
||||
|
||||
proc rect*(path: Path, rect: Rect, clockwise = true) {.inline.} =
|
||||
## Adds a rectangle.
|
||||
## Clockwise param can be used to subtract a rect from a path when using
|
||||
|
@ -590,14 +581,6 @@ proc roundedRect*(
|
|||
|
||||
path.closePath()
|
||||
|
||||
proc roundedRect*(
|
||||
path: Path, pos, wh: Vec2, nw, ne, se, sw: float32, clockwise = true
|
||||
) {.inline.} =
|
||||
## Adds a rounded rectangle.
|
||||
## Clockwise param can be used to subtract a rect from a path when using
|
||||
## even-odd winding rule.
|
||||
path.roundedRect(pos.x, pos.y, wh.x, wh.y, nw, ne, se, sw, clockwise)
|
||||
|
||||
proc roundedRect*(
|
||||
path: Path, rect: Rect, nw, ne, se, sw: float32, clockwise = true
|
||||
) {.inline.} =
|
||||
|
@ -627,10 +610,6 @@ proc circle*(path: Path, cx, cy, r: float32) {.inline.} =
|
|||
## Adds a circle.
|
||||
path.ellipse(cx, cy, r, r)
|
||||
|
||||
proc circle*(path: Path, center: Vec2, r: float32) {.inline.} =
|
||||
## Adds a circle.
|
||||
path.ellipse(center.x, center.y, r, r)
|
||||
|
||||
proc circle*(path: Path, circle: Circle) {.inline.} =
|
||||
## Adds a circle.
|
||||
path.ellipse(circle.pos.x, circle.pos.y, circle.radius, circle.radius)
|
||||
|
@ -648,7 +627,7 @@ proc polygon*(path: Path, pos: Vec2, size: float32, sides: int) {.inline.} =
|
|||
## Adds a n-sided regular polygon at (x, y) with the parameter size.
|
||||
path.polygon(pos.x, pos.y, size, sides)
|
||||
|
||||
proc commandsToShapes*(
|
||||
proc commandsToShapes(
|
||||
path: Path, closeSubpaths = false, pixelScale: float32 = 1.0
|
||||
): seq[seq[Vec2]] =
|
||||
## Converts SVG-like commands to sequences of vectors.
|
||||
|
@ -1750,22 +1729,16 @@ proc parseSomePath(
|
|||
elif type(path) is Path:
|
||||
path.commandsToShapes(closeSubpaths, pixelScale)
|
||||
|
||||
proc transform(shapes: var seq[seq[Vec2]], transform: Vec2 | Mat3) =
|
||||
when type(transform) is Vec2:
|
||||
if transform != vec2():
|
||||
for shape in shapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment += transform
|
||||
else:
|
||||
if transform != mat3():
|
||||
for shape in shapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment = transform * segment
|
||||
proc transform(shapes: var seq[seq[Vec2]], transform: Mat3) =
|
||||
if transform != mat3():
|
||||
for shape in shapes.mitems:
|
||||
for segment in shape.mitems:
|
||||
segment = transform * segment
|
||||
|
||||
proc fillPath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
transform: Vec2 | Mat3 = vec2(),
|
||||
transform = mat3(),
|
||||
windingRule = wrNonZero,
|
||||
blendMode = bmNormal
|
||||
) =
|
||||
|
@ -1778,7 +1751,7 @@ proc fillPath*(
|
|||
image: Image,
|
||||
path: SomePath,
|
||||
paint: Paint,
|
||||
transform: Vec2 | Mat3 = vec2(),
|
||||
transform = mat3(),
|
||||
windingRule = wrNonZero
|
||||
) =
|
||||
## Fills a path.
|
||||
|
@ -1827,7 +1800,7 @@ proc fillPath*(
|
|||
proc strokePath*(
|
||||
mask: Mask,
|
||||
path: SomePath,
|
||||
transform: Vec2 | Mat3 = vec2(),
|
||||
transform = mat3(),
|
||||
strokeWidth = 1.0,
|
||||
lineCap = lcButt,
|
||||
lineJoin = ljMiter,
|
||||
|
@ -1851,7 +1824,7 @@ proc strokePath*(
|
|||
image: Image,
|
||||
path: SomePath,
|
||||
paint: Paint,
|
||||
transform: Vec2 | Mat3 = vec2(),
|
||||
transform = mat3(),
|
||||
strokeWidth = 1.0,
|
||||
lineCap = lcButt,
|
||||
lineJoin = ljMiter,
|
||||
|
@ -1948,7 +1921,7 @@ proc overlaps(
|
|||
proc fillOverlaps*(
|
||||
path: Path,
|
||||
test: Vec2,
|
||||
transform: Vec2 | Mat3 = vec2(), ## Applied to the path, not the test point.
|
||||
transform = mat3(), ## Applied to the path, not the test point.
|
||||
windingRule = wrNonZero
|
||||
): bool =
|
||||
## Returns whether or not the specified point is contained in the current path.
|
||||
|
@ -1959,7 +1932,7 @@ proc fillOverlaps*(
|
|||
proc strokeOverlaps*(
|
||||
path: Path,
|
||||
test: Vec2,
|
||||
transform: Vec2 | Mat3 = vec2(), ## Applied to the path, not the test point.
|
||||
transform = mat3(), ## Applied to the path, not the test point.
|
||||
strokeWidth = 1.0,
|
||||
lineCap = lcButt,
|
||||
lineJoin = ljMiter,
|
||||
|
|
|
@ -154,7 +154,7 @@ block:
|
|||
let image = newImage(200, 100)
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.fillText(font, "First line")
|
||||
image.fillText(font, "Second line", vec2(0, font.defaultLineHeight))
|
||||
image.fillText(font, "Second line", translate(vec2(0, font.defaultLineHeight)))
|
||||
|
||||
doDiff(image, "basic7")
|
||||
|
||||
|
@ -826,7 +826,7 @@ block:
|
|||
|
||||
let arrangement = typeset(spans, bounds = vec2(360, 360))
|
||||
|
||||
image.fillText(arrangement, vec2(20, 20))
|
||||
image.fillText(arrangement, translate(vec2(20, 20)))
|
||||
|
||||
doDiff(image, "spans5")
|
||||
|
||||
|
@ -965,7 +965,7 @@ block:
|
|||
|
||||
let arrangement = typeset(spans, bounds = vec2(360, 360))
|
||||
|
||||
image.fillText(arrangement, vec2(20, 20))
|
||||
image.fillText(arrangement, translate(vec2(20, 20)))
|
||||
|
||||
doDiff(image, "spans6")
|
||||
|
||||
|
|
|
@ -213,7 +213,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20 Z")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcRound, ljRound)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcRound, ljRound
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/boxRound.png")
|
||||
|
||||
|
@ -222,7 +224,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20 Z")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcRound, ljBevel)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcRound, ljBevel
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/boxBevel.png")
|
||||
|
||||
|
@ -231,7 +235,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20 Z")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcRound, ljMiter)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcRound, ljMiter
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/boxMiter.png")
|
||||
|
||||
|
@ -240,7 +246,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcButt, ljBevel)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcButt, ljBevel
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/lcButt.png")
|
||||
|
||||
|
@ -249,7 +257,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcRound, ljBevel)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcRound, ljBevel
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/lcRound.png")
|
||||
|
||||
|
@ -258,7 +268,9 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 20 3 L 20 20 L 3 20")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcSquare, ljBevel)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcSquare, ljBevel
|
||||
)
|
||||
|
||||
image.writeFile("tests/images/paths/lcSquare.png")
|
||||
|
||||
|
@ -269,31 +281,31 @@ block:
|
|||
image.fill(rgba(255, 255, 255, 255))
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 5), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 5)), 10, lcButt, ljBevel,
|
||||
)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 25), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 25)), 10, lcButt, ljBevel,
|
||||
dashes = @[2.float32, 2]
|
||||
)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 45), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 45)), 10, lcButt, ljBevel,
|
||||
dashes = @[4.float32, 4]
|
||||
)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 65), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 65)), 10, lcButt, ljBevel,
|
||||
dashes = @[2.float32, 4, 6, 2]
|
||||
)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 85), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 85)), 10, lcButt, ljBevel,
|
||||
dashes = @[1.float32]
|
||||
)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(5, 105), 10, lcButt, ljBevel,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(5, 105)), 10, lcButt, ljBevel,
|
||||
dashes = @[1.float32, 2, 3, 4, 5, 6, 7, 8, 9]
|
||||
)
|
||||
|
||||
|
@ -311,7 +323,7 @@ block:
|
|||
path.lineTo(sin(th)*20, cos(th)*20)
|
||||
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), vec2(30, 30), 8, lcButt, ljMiter,
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(30, 30)), 8, lcButt, ljMiter,
|
||||
miterLimit = limit
|
||||
)
|
||||
image.writeFile(&"tests/images/paths/miterLimit_{angle.int}deg_{limit:0.2f}num.png")
|
||||
|
@ -340,28 +352,36 @@ block:
|
|||
image = newImage(60, 60)
|
||||
path = parsePath("M 3 3 L 3 3 L 3 3")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcSquare, ljMiter)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcSquare, ljMiter
|
||||
)
|
||||
|
||||
block:
|
||||
let
|
||||
image = newImage(60, 60)
|
||||
path = parsePath("L 0 0 L 0 0")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcSquare, ljMiter)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcSquare, ljMiter
|
||||
)
|
||||
|
||||
block:
|
||||
let
|
||||
image = newImage(60, 60)
|
||||
path = parsePath("L 1 1")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcSquare, ljMiter)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcSquare, ljMiter
|
||||
)
|
||||
|
||||
block:
|
||||
let
|
||||
image = newImage(60, 60)
|
||||
path = parsePath("L 0 0")
|
||||
image.fill(rgba(255, 255, 255, 255))
|
||||
image.strokePath(path, rgba(0, 0, 0, 255), vec2(10, 10), 10, lcSquare, ljMiter)
|
||||
image.strokePath(
|
||||
path, rgba(0, 0, 0, 255), translate(vec2(10, 10)), 10, lcSquare, ljMiter
|
||||
)
|
||||
|
||||
block:
|
||||
let image = newImage(100, 100)
|
||||
|
|
Loading…
Reference in a new issue