diff --git a/src/pixie/contexts.nim b/src/pixie/contexts.nim index 7ba9e4a..e1a9f2b 100644 --- a/src/pixie/contexts.nim +++ b/src/pixie/contexts.nim @@ -7,6 +7,15 @@ import bumpy, chroma, pixie/common, pixie/fonts, pixie/images, pixie/masks, ## https://developer.mozilla.org/en-US/docs/Web/API/ContextRenderingContext2D type + + BaselineAlignment* = enum + TopBaseline + HangingBaseline + MiddleBaseline + AlphabeticBaseline + IdeographicBaseline + BottomBaseline + Context* = ref object image*: Image fillStyle*, strokeStyle*: Paint @@ -18,6 +27,7 @@ type font*: string ## File path to a .ttf or .otf file. fontSize*: float32 textAlign*: HorizontalAlignment + textBaseline*: BaselineAlignment lineDash: seq[float32] path: Path mat: Mat3 @@ -58,6 +68,7 @@ proc newContext*(image: Image): Context {.raises: [].} = result.strokeStyle = newPaint(SolidPaint) result.strokeStyle.color = color(0, 0, 0, 1) result.fontSize = 12 + result.textBaseline = AlphabeticBaseline proc newContext*(width, height: int): Context {.inline, raises: [PixieError].} = ## Create a new Context that will draw to a new image of width and height. @@ -185,7 +196,22 @@ proc fillText(ctx: Context, image: Image, text: string, at: Vec2) = # Canvas positions text relative to the alphabetic baseline by default var at = at - at.y -= round(font.typeface.ascent * font.scale) + + case ctx.textBaseline: + of TopBaseline: + discard + of HangingBaseline: + # TODO: make accurate (Used by Tibetan and other Indic scripts.) + discard + of MiddleBaseline: + at.y -= round((font.typeface.ascent - font.typeface.descent) / 2 * font.scale) + of AlphabeticBaseline: + at.y -= round(font.typeface.ascent * font.scale) + of IdeographicBaseline: + # TODO: make accurate (Used by Chinese, Japanese, and Korean scripts.) + at.y -= round((font.typeface.ascent - font.typeface.descent) * font.scale) + of BottomBaseline: + at.y -= round((font.typeface.ascent - font.typeface.descent) * font.scale) font.paint = ctx.fillStyle diff --git a/src/pixie/images.nim b/src/pixie/images.nim index b3b4cf8..1a81499 100644 --- a/src/pixie/images.nim +++ b/src/pixie/images.nim @@ -781,12 +781,12 @@ proc drawUber( when type(a) is Image and type(b) is Image: if blendMode in {NormalBlend, OverwriteBlend} and isOpaque(b.data, b.dataIndex(sx, sy), xStop - xStart): - copyMem( - a.data[a.dataIndex(x, y)].addr, - b.data[b.dataIndex(sx, sy)].addr, - (xStop - xStart) * 4 - ) - continue + copyMem( + a.data[a.dataIndex(x, y)].addr, + b.data[b.dataIndex(sx, sy)].addr, + (xStop - xStart) * 4 + ) + continue when defined(amd64) and not defined(pixieNoSimd): case blendMode: diff --git a/tests/contexts/textBaseline_1.png b/tests/contexts/textBaseline_1.png new file mode 100644 index 0000000..d7333c7 Binary files /dev/null and b/tests/contexts/textBaseline_1.png differ diff --git a/tests/test_contexts.nim b/tests/test_contexts.nim index ae4bac4..8366133 100644 --- a/tests/test_contexts.nim +++ b/tests/test_contexts.nim @@ -644,3 +644,34 @@ block: ctx.restore() ctx.fillRect(0, 0, ctx.image.width.float32, ctx.image.height.float32) ctx.image.writeFile("tests/contexts/paintSaveRestore.png") + +block: + # From https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/textBaseline + let + image = newImage(550, 500) + ctx = newContext(image) + image.fill(rgba(255, 255, 255, 255)) + + const baselines = @[ + TopBaseline, + HangingBaseline, + MiddleBaseline, + AlphabeticBaseline, + IdeographicBaseline, + BottomBaseline, + ] + + ctx.font = "tests/fonts/Roboto-Regular_1.ttf" + ctx.fontSize = 28 + ctx.strokeStyle = "red" + + for index, baseline in baselines: + ctx.textBaseline = baseline + let y = (75 + index * 75).float32 + ctx.beginPath() + ctx.moveTo(0, y + 0.5) + ctx.lineTo(550, y + 0.5) + ctx.stroke() + ctx.fillText("Abcdefghijklmnop (" & $baseline & ")", 0, y) + + ctx.image.writeFile("tests/contexts/textBaseline_1.png")