|
@ -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.
|
||||
|
|
BIN
tests/fonts/diffs/fallback.png
Normal file
After Width: | Height: | Size: 16 KiB |
BIN
tests/fonts/diffs/fallback2.png
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
tests/fonts/diffs/tofu_advance.png
Normal file
After Width: | Height: | Size: 4.7 KiB |
BIN
tests/fonts/masters/fallback.png
Normal file
After Width: | Height: | Size: 7.8 KiB |
BIN
tests/fonts/masters/fallback2.png
Normal file
After Width: | Height: | Size: 7.2 KiB |
BIN
tests/fonts/masters/tofu_advance.png
Normal file
After Width: | Height: | Size: 2.7 KiB |
BIN
tests/fonts/rendered/fallback.png
Normal file
After Width: | Height: | Size: 16 KiB |
BIN
tests/fonts/rendered/fallback2.png
Normal file
After Width: | Height: | Size: 16 KiB |
BIN
tests/fonts/rendered/tofu_advance.png
Normal file
After Width: | Height: | Size: 5.6 KiB |
|
@ -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")
|
||||
|
|