inter/misc/tools/gen-glyphinfo.py

143 lines
3.6 KiB
Python
Raw Normal View History

2017-08-22 09:05:20 +02:00
#!/usr/bin/env python
# encoding: utf8
#
# Grab http://www.unicode.org/Public/UCD/latest/ucd/UnicodeData.txt
#
2018-09-04 01:06:04 +02:00
import os, sys
from os.path import dirname, basename, abspath, relpath, join as pjoin
sys.path.append(abspath(pjoin(dirname(__file__), 'tools')))
from common import BASEDIR, getVersion, getLocalTimeZoneOffset
import json, re
import time
2017-08-22 09:05:20 +02:00
from argparse import ArgumentParser
from collections import OrderedDict
2019-01-02 05:47:16 +01:00
from configparser import RawConfigParser
2018-09-04 01:06:04 +02:00
from unicode_util import parseUnicodeDataFile
from defcon import Font
2017-08-22 09:05:20 +02:00
def rgbaToCSSColor(r=0, g=0, b=0, a=1):
R,G,B = int(r * 255), int(g * 255), int(b * 255)
if a == 1:
return '#%02x%02x%02x' % (R,G,B)
else:
2018-09-04 01:06:04 +02:00
return 'rgba(%d,%d,%d,%g)' % (R,G,B,a)
2017-08-22 09:05:20 +02:00
def unicodeName(cp):
if cp is not None and len(cp.name):
if cp.name[0] == '<':
return '[' + cp.categoryName + ']'
elif len(cp.name):
return cp.name
return None
2018-09-04 01:06:04 +02:00
# YYYY/MM/DD HH:mm:ss => YYYY-MM-DDTHH:mm:ssZ (local time -> UTC)
def localDateTimeToUTCStr(localstr, pattern='%Y/%m/%d %H:%M:%S'):
t = time.strptime(localstr, pattern)
ts = time.mktime(t) - getLocalTimeZoneOffset()
return time.strftime('%Y-%m-%dT%H:%M:%SZ', time.localtime(ts))
def processGlyph(g, ucd, seenGlyphnames):
name = g.name
if name in seenGlyphnames:
return None
seenGlyphnames.add(name)
# not exported?
if 'com.schriftgestaltung.Glyphs.Export' in g.lib:
if not g.lib['com.schriftgestaltung.Glyphs.Export']:
return None
# color
color = None
if 'public.markColor' in g.lib:
rgba = [float(c.strip()) for c in g.lib['public.markColor'].strip().split(',')]
color = rgbaToCSSColor(*rgba)
isEmpty = 0
if not g.bounds or g.bounds[3] == 0:
isEmpty = 1
# name, isEmpty, unicode, unicodeName, color
glyph = None
ucs = g.unicodes
if len(ucs):
for uc in ucs:
ucName = unicodeName(ucd.get(uc))
# if not ucName and uc >= 0xE000 and uc <= 0xF8FF:
# ucName = '[private use %04X]' % uc
ucstr = '%04X' % uc
if color:
glyph = [name, isEmpty, ucstr, ucName, color]
elif ucName:
glyph = [name, isEmpty, ucstr, ucName]
else:
glyph = [name, isEmpty, ucstr]
else:
if color:
glyph = [name, isEmpty, None, None, color]
else:
glyph = [name, isEmpty]
return glyph
2021-04-01 04:09:32 +02:00
def glyphSortFun(g):
if len(g) > 2 and g[2] is not None:
return g[2]
elif len(g) > 0:
return g[0]
else:
return ""
2018-09-04 01:06:04 +02:00
2017-08-22 09:05:20 +02:00
def main():
argparser = ArgumentParser(
description='Generate info on name, unicodes and color mark for all glyphs')
argparser.add_argument(
'-ucd', dest='ucdFile', metavar='<file>', type=str,
help='UnicodeData.txt file from http://www.unicode.org/')
argparser.add_argument(
'fontPath', metavar='<ufofile>', type=str)
2017-08-22 09:05:20 +02:00
args = argparser.parse_args()
font = Font(args.fontPath)
2017-08-22 09:05:20 +02:00
ucd = {}
if args.ucdFile:
ucd = parseUnicodeDataFile(args.ucdFile)
2018-09-04 01:06:04 +02:00
glyphs = [] # contains final glyph data printed as JSON
seenGlyphnames = set()
for name in font.lib['public.glyphOrder']:
g = font[name]
glyph = processGlyph(g, ucd, seenGlyphnames)
if glyph is not None:
glyphs.append(glyph)
unorderedGlyphs = []
for g in font:
glyph = processGlyph(g, ucd, seenGlyphnames)
if glyph is not None:
unorderedGlyphs.append(glyph)
if unorderedGlyphs:
# sort by unicode
2021-04-01 04:09:32 +02:00
glyphs = glyphs + sorted(unorderedGlyphs, key=glyphSortFun)
2017-08-22 09:05:20 +02:00
print('{"glyphs":[')
prefix = ' '
for g in glyphs:
print(prefix + json.dumps(g))
if prefix == ' ':
prefix = ', '
print(']}')
if __name__ == '__main__':
main()