2016-03-26 22:46:18 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
import freetype
|
|
|
|
|
2016-03-27 11:36:27 +00:00
|
|
|
MIN_GLYPH = ord(' ')
|
|
|
|
MAX_GLYPH = ord('~')
|
2016-03-26 22:46:18 +00:00
|
|
|
|
2016-10-18 13:05:55 +00:00
|
|
|
# metrics explanation: https://www.freetype.org/freetype2/docs/glyphs/metrics.png
|
|
|
|
|
2017-06-13 14:50:03 +00:00
|
|
|
|
2016-03-26 22:46:18 +00:00
|
|
|
def process_face(name, style, size):
|
2016-10-18 13:05:55 +00:00
|
|
|
print('Processing ... %s %s %s' % (name, style, size))
|
2016-03-26 22:46:18 +00:00
|
|
|
face = freetype.Face('/usr/share/fonts/truetype/%s-%s.ttf' % (name, style))
|
|
|
|
face.set_pixel_sizes(0, size)
|
2016-10-18 13:05:55 +00:00
|
|
|
fontname = '%s_%s_%d' % (name.lower(), style.lower(), size)
|
|
|
|
with open('font_%s.h' % fontname, 'wt') as f:
|
|
|
|
f.write('#include <stdint.h>\n\n')
|
2017-02-22 13:26:30 +00:00
|
|
|
f.write('extern const uint8_t * const Font_%s_%s_%d[%d + 1 - %d];\n' % (name, style, size, MAX_GLYPH, MIN_GLYPH))
|
2016-10-18 13:05:55 +00:00
|
|
|
with open('font_%s.c' % fontname, 'wt') as f:
|
|
|
|
f.write('#include "font_%s.h"\n\n' % fontname)
|
2016-03-26 22:46:18 +00:00
|
|
|
f.write('// first two bytes are width and height of the glyph\n')
|
|
|
|
f.write('// third, fourth and fifth bytes are advance, bearingX and bearingY of the horizontal metrics of the glyph\n')
|
|
|
|
f.write('// rest is packed 4-bit glyph data\n\n')
|
|
|
|
for i in range(MIN_GLYPH, MAX_GLYPH + 1):
|
|
|
|
c = chr(i)
|
|
|
|
face.load_char(c, freetype.FT_LOAD_RENDER | freetype.FT_LOAD_TARGET_NORMAL)
|
|
|
|
bitmap = face.glyph.bitmap
|
|
|
|
metrics = face.glyph.metrics
|
2017-06-13 14:50:03 +00:00
|
|
|
assert metrics.width // 64 == bitmap.width
|
|
|
|
assert metrics.height // 64 == bitmap.rows
|
|
|
|
assert metrics.horiAdvance % 64 == 0
|
|
|
|
assert metrics.horiBearingX % 64 == 0
|
|
|
|
assert metrics.horiBearingY % 64 == 0
|
|
|
|
assert bitmap.width == bitmap.pitch
|
|
|
|
assert len(bitmap.buffer) == bitmap.pitch * bitmap.rows
|
2016-03-27 11:36:27 +00:00
|
|
|
print('Loaded glyph "%c" ... %d x %d @ %d grays (%d bytes)' % (c, bitmap.width, bitmap.rows, bitmap.num_grays, len(bitmap.buffer)))
|
2016-03-27 21:30:35 +00:00
|
|
|
f.write('/* %c */ static const uint8_t Font_%s_%s_%d_glyph_%d[] = { %d, %d, %d, %d, %d' % (c, name, style, size, i, bitmap.width, bitmap.rows, metrics.horiAdvance // 64, metrics.horiBearingX // 64, metrics.horiBearingY // 64))
|
2016-03-26 22:46:18 +00:00
|
|
|
buf = list(bitmap.buffer)
|
2016-03-27 21:30:35 +00:00
|
|
|
if len(buf) > 0:
|
|
|
|
if len(buf) % 2 > 0:
|
|
|
|
buf.append(0)
|
2017-06-13 14:50:03 +00:00
|
|
|
buf = [((a & 0xF0) | (b >> 4)) for a, b in [buf[i:i + 2] for i in range(0, len(buf), 2)]]
|
2016-03-27 21:30:35 +00:00
|
|
|
f.write(', ' + ', '.join(['%d' % x for x in buf]))
|
2016-03-26 22:46:18 +00:00
|
|
|
f.write(' };\n')
|
2016-03-27 21:30:35 +00:00
|
|
|
f.write('\nconst uint8_t * const Font_%s_%s_%d[%d + 1 - %d] = {\n' % (name, style, size, MAX_GLYPH, MIN_GLYPH))
|
2016-03-26 22:46:18 +00:00
|
|
|
for i in range(MIN_GLYPH, MAX_GLYPH + 1):
|
2016-03-27 21:30:35 +00:00
|
|
|
f.write(' Font_%s_%s_%d_glyph_%d,\n' % (name, style, size, i))
|
2016-03-26 22:46:18 +00:00
|
|
|
f.write('};\n')
|
|
|
|
|
2017-09-05 21:15:47 +00:00
|
|
|
|
2016-03-30 16:30:22 +00:00
|
|
|
process_face('Roboto', 'Regular', 20)
|
2016-10-18 13:05:55 +00:00
|
|
|
process_face('Roboto', 'Bold', 20)
|
|
|
|
process_face('RobotoMono', 'Regular', 20)
|