diff --git a/src/pixie/fonts.nim b/src/pixie/fonts.nim index f2d019a..0b42065 100644 --- a/src/pixie/fonts.nim +++ b/src/pixie/fonts.nim @@ -12,6 +12,7 @@ type opentype: OpenType svgFont: SvgFont filePath*: string + fallbacks*: seq[Typeface] Font* = ref object typeface*: Typeface @@ -53,6 +54,13 @@ type # tcSmallCaps # tcSmallCapsForced +proc scale*(typeface: Typeface): float32 {.inline, raises: [].} = + ## The scale factor to transform font units into pixels. + if typeface.opentype != nil: + typeface.opentype.head.unitsPerEm.float32 + else: + typeface.svgFont.unitsPerEm + proc ascent*(typeface: Typeface): float32 {.raises: [].} = ## The font ascender value in font units. if typeface.opentype != nil: @@ -97,49 +105,77 @@ proc isCCW(typeface: Typeface): bool {.inline.} = if typeface.opentype != nil: return typeface.opentype.isCCW() +proc hasGlyph*(typeface: Typeface, rune: Rune): bool {.inline.} = + ## Returns if there is a glyph for this rune. + if typeface.opentype != nil: + typeface.opentype.hasGlyph(rune) + else: + typeface.svgFont.hasGlyph(rune) + +proc fallbackTypeface*(typeface: Typeface, rune: Rune): Typeface = + ## Looks through fallback typefaces to find one that has the glyph. + if typeface.hasGlyph(rune): + return typeface + for fallback in typeface.fallbacks: + let typeface = fallback.fallbackTypeface(rune) + if typeface != nil: + return typeface + proc getGlyphPath*( typeface: Typeface, rune: Rune ): Path {.inline, raises: [PixieError].} = ## The glyph path for the rune. result = newPath() - if rune.uint32 > SP.uint32: # Empty paths for control runes (not tofu) - if typeface.opentype != nil: - result.addPath(typeface.opentype.getGlyphPath(rune)) - else: - result.addPath(typeface.svgFont.getGlyphPath(rune)) -proc hasGlyph*(typeface: Typeface, rune: Rune): bool {.inline.} = - ## Returns if there is a glyph for this rune. - if rune.uint32 > SP.uint32: # Empty paths for control runes (not tofu) - if typeface.opentype != nil: - typeface.opentype.hasGlyph(rune) - else: - typeface.svgFont.hasGlyph(rune) + let typeface2 = typeface.fallbackTypeface(rune) + if typeface2 == nil: + return + + if typeface2.opentype != nil: + result.addPath(typeface2.opentype.getGlyphPath(rune)) else: - false + result.addPath(typeface2.svgFont.getGlyphPath(rune)) + + # Apply typeface ratio. + let ratio = typeface.scale / typeface2.scale + if ratio != 1.0: + result.transform(scale(vec2(ratio, ratio))) proc getAdvance*(typeface: Typeface, rune: Rune): float32 {.inline, raises: [].} = ## The advance for the rune in pixels. - if typeface.opentype != nil: - typeface.opentype.getAdvance(rune) + var typeface2 = typeface.fallbackTypeface(rune) + if typeface2 == nil: + # Get tofu advance, see tofu_advance test. + typeface2 = typeface + + if typeface2.opentype != nil: + result = typeface2.opentype.getAdvance(rune) else: - typeface.svgFont.getAdvance(rune) + result = typeface2.svgFont.getAdvance(rune) + + # Apply typeface ratio. + result *= typeface.scale / typeface2.scale proc getKerningAdjustment*( typeface: Typeface, left, right: Rune ): float32 {.inline, raises: [].} = ## The kerning adjustment for the rune pair, in pixels. - if typeface.opentype != nil: - typeface.opentype.getKerningAdjustment(left, right) - else: - typeface.svgfont.getKerningAdjustment(left, right) + let + typefaceRight = typeface.fallbackTypeface(right) + typefaceLeft = typeface.fallbackTypeface(left) + # Only do kerning if both typefaces are the same. + if typefaceRight == typefaceLeft: + if typefaceRight.opentype != nil: + result = typefaceRight.opentype.getKerningAdjustment(left, right) + else: + result = typefaceRight.svgfont.getKerningAdjustment(left, right) + + # Apply typeface ratio. + result *= typeface.scale / typefaceRight.scale proc scale*(font: Font): float32 {.inline, raises: [].} = ## The scale factor to transform font units into pixels. - if font.typeface.opentype != nil: - font.size / font.typeface.opentype.head.unitsPerEm.float32 - else: - font.size / font.typeface.svgFont.unitsPerEm + font.size / font.typeface.scale proc defaultLineHeight*(font: Font): float32 {.inline, raises: [].} = ## The default line height in pixels for the current font size. diff --git a/tests/fonts/diffs/fallback.png b/tests/fonts/diffs/fallback.png new file mode 100644 index 0000000..80016e6 Binary files /dev/null and b/tests/fonts/diffs/fallback.png differ diff --git a/tests/fonts/diffs/fallback2.png b/tests/fonts/diffs/fallback2.png new file mode 100644 index 0000000..35a9e3a Binary files /dev/null and b/tests/fonts/diffs/fallback2.png differ diff --git a/tests/fonts/diffs/tofu_advance.png b/tests/fonts/diffs/tofu_advance.png new file mode 100644 index 0000000..2b30fd2 Binary files /dev/null and b/tests/fonts/diffs/tofu_advance.png differ diff --git a/tests/fonts/masters/fallback.png b/tests/fonts/masters/fallback.png new file mode 100644 index 0000000..c1e51c1 Binary files /dev/null and b/tests/fonts/masters/fallback.png differ diff --git a/tests/fonts/masters/fallback2.png b/tests/fonts/masters/fallback2.png new file mode 100644 index 0000000..11351fc Binary files /dev/null and b/tests/fonts/masters/fallback2.png differ diff --git a/tests/fonts/masters/tofu_advance.png b/tests/fonts/masters/tofu_advance.png new file mode 100644 index 0000000..adee2e3 Binary files /dev/null and b/tests/fonts/masters/tofu_advance.png differ diff --git a/tests/fonts/rendered/fallback.png b/tests/fonts/rendered/fallback.png new file mode 100644 index 0000000..a200ca0 Binary files /dev/null and b/tests/fonts/rendered/fallback.png differ diff --git a/tests/fonts/rendered/fallback2.png b/tests/fonts/rendered/fallback2.png new file mode 100644 index 0000000..a200ca0 Binary files /dev/null and b/tests/fonts/rendered/fallback2.png differ diff --git a/tests/fonts/rendered/tofu_advance.png b/tests/fonts/rendered/tofu_advance.png new file mode 100644 index 0000000..a1deb57 Binary files /dev/null and b/tests/fonts/rendered/tofu_advance.png differ diff --git a/tests/test_fonts.nim b/tests/test_fonts.nim index b4a34e3..6b78f7c 100644 --- a/tests/test_fonts.nim +++ b/tests/test_fonts.nim @@ -1057,3 +1057,47 @@ block: image.fillText(font, "Grumpy wizards make toxic brew for the evil Queen and Jack.") doDiff(image, "cff_strikethrough") + +block: + var font = readFont("tests/fonts/Inter-Regular.ttf") + var typeface = readTypeface("tests/fonts/NotoSansJP-Regular.ttf") + font.typeface.fallbacks.add(typeface) + font.size = 26 + let image = newImage(800, 100) + image.fill(rgba(255, 255, 255, 255)) + image.fillText(font, "Grumpy ウィザード make 有毒な醸造 for the 悪い女王 and Jack.") + + doDiff(image, "fallback") + +block: + let + font = readFont("tests/fonts/Inter-Regular.ttf") + typeface1 = readTypeface("tests/fonts/Aclonica-Regular_1.ttf") + typeface2 = readTypeface("tests/fonts/Ubuntu-Regular_1.ttf") + typeface3 = readTypeface("tests/fonts/NotoSansJP-Regular.ttf") + + # font + # |.... typeface1 + # |.... typeface2 + # |.... typeface3 (with JP) + + font.typeface.fallbacks.add(typeface1) + typeface1.fallbacks.add(typeface2) + typeface1.fallbacks.add(typeface3) + + font.size = 26 + let image = newImage(800, 100) + image.fill(rgba(255, 255, 255, 255)) + image.fillText(font, "Grumpy ウィザード make 有毒な醸造 for the 悪い女王 and Jack.") + + doDiff(image, "fallback2") + +block: + var font = readFont("tests/fonts/Inter-Regular.ttf") + + font.size = 26 + let image = newImage(800, 100) + image.fill(rgba(255, 255, 255, 255)) + image.fillText(font, "This[]Advance!") + + doDiff(image, "tofu_advance")