summaryrefslogtreecommitdiffstats
path: root/examples/psfgenftml.py
diff options
context:
space:
mode:
Diffstat (limited to 'examples/psfgenftml.py')
-rw-r--r--examples/psfgenftml.py189
1 files changed, 189 insertions, 0 deletions
diff --git a/examples/psfgenftml.py b/examples/psfgenftml.py
new file mode 100644
index 0000000..2807e45
--- /dev/null
+++ b/examples/psfgenftml.py
@@ -0,0 +1,189 @@
+#!/usr/bin/env python3
+'''
+Example script to generate ftml document from glyph_data.csv and UFO.
+
+To try this with the Harmattan font project:
+ 1) clone and build Harmattan:
+ clone https://github.com/silnrsi/font-harmattan
+ cd font-harmattan
+ smith configure
+ smith build ftml
+ 2) run psfgenftml as follows:
+ python3 psfgenftml.py \
+ -t "AllChars" \
+ --ap "_?dia[AB]$" \
+ --xsl ../tools/lib/ftml.xsl \
+ --scale 200 \
+ -i source/glyph_data.csv \
+ -s "url(../references/Harmattan-Regular-v1.ttf)=ver 1" \
+ -s "url(../results/Harmattan-Regular.ttf)=Reg-GR" \
+ -s "url(../results/tests/ftml/fonts/Harmattan-Regular_ot_arab.ttf)=Reg-OT" \
+ source/Harmattan-Regular.ufo tests/AllChars-dev.ftml
+ 3) launch resulting output file, tests/AllChars-dev.ftml, in a browser.
+ (see https://silnrsi.github.io/FDBP/en-US/Browsers%20as%20a%20font%20test%20platform.html)
+ NB: Using Firefox will allow simultaneous display of both Graphite and OpenType rendering
+ 4) As above but substitute:
+ -t "Diac Test" for the -t parameter
+ tests/DiacTest-dev.ftml for the final parameter
+ and launch tests/DiacTest-dev.ftml in a browser.
+'''
+__url__ = 'https://github.com/silnrsi/pysilfont'
+__copyright__ = 'Copyright (c) 2018,2021 SIL International (https://www.sil.org)'
+__license__ = 'Released under the MIT License (https://opensource.org/licenses/MIT)'
+__author__ = 'Bob Hallissy'
+
+import re
+from silfont.core import execute
+import silfont.ftml_builder as FB
+
+argspec = [
+ ('ifont', {'help': 'Input UFO'}, {'type': 'infont'}),
+ ('output', {'help': 'Output file ftml in XML format', 'nargs': '?'}, {'type': 'outfile', 'def': '_out.ftml'}),
+ ('-i','--input', {'help': 'Glyph info csv file'}, {'type': 'incsv', 'def': 'glyph_data.csv'}),
+ ('-f','--fontcode', {'help': 'letter to filter for glyph_data'},{}),
+ ('-l','--log', {'help': 'Set log file name'}, {'type': 'outfile', 'def': '_ftml.log'}),
+ ('--langs', {'help':'List of bcp47 language tags', 'default': None}, {}),
+ ('--rtl', {'help': 'enable right-to-left features', 'action': 'store_true'}, {}),
+ ('--norendercheck', {'help': 'do not include the RenderingUnknown check', 'action': 'store_true'}, {}),
+ ('-t', '--test', {'help': 'name of the test to generate', 'default': None}, {}),
+ ('-s','--fontsrc', {'help': 'font source: "url()" or "local()" optionally followed by "=label"', 'action': 'append'}, {}),
+ ('--scale', {'help': 'percentage to scale rendered text (default 100)'}, {}),
+ ('--ap', {'help': 'regular expression describing APs to examine', 'default': '.'}, {}),
+ ('-w', '--width', {'help': 'total width of all <string> column (default automatic)'}, {}),
+ ('--xsl', {'help': 'XSL stylesheet to use'}, {}),
+]
+
+
+def doit(args):
+ logger = args.logger
+
+ # Read input csv
+ builder = FB.FTMLBuilder(logger, incsv=args.input, fontcode=args.fontcode, font=args.ifont, ap=args.ap,
+ rtlenable=True, langs=args.langs)
+
+ # Override default base (25CC) for displaying combining marks:
+ builder.diacBase = 0x0628 # beh
+
+ # Initialize FTML document:
+ # Default name for test: AllChars or something based on the csvdata file:
+ test = args.test or 'AllChars (NG)'
+ widths = None
+ if args.width:
+ try:
+ width, units = re.match(r'(\d+)(.*)$', args.width).groups()
+ if len(args.fontsrc):
+ width = int(round(int(width)/len(args.fontsrc)))
+ widths = {'string': f'{width}{units}'}
+ logger.log(f'width: {args.width} --> {widths["string"]}', 'I')
+ except:
+ logger.log(f'Unable to parse width argument "{args.width}"', 'W')
+ # split labels from fontsource parameter
+ fontsrc = []
+ labels = []
+ for sl in args.fontsrc:
+ try:
+ s, l = sl.split('=',1)
+ fontsrc.append(s)
+ labels.append(l)
+ except ValueError:
+ fontsrc.append(sl)
+ labels.append(None)
+ ftml = FB.FTML(test, logger, rendercheck=not args.norendercheck, fontscale=args.scale,
+ widths=widths, xslfn=args.xsl, fontsrc=fontsrc, fontlabel=labels, defaultrtl=args.rtl)
+
+ if test.lower().startswith("allchars"):
+ # all chars that should be in the font:
+ ftml.startTestGroup('Encoded characters')
+ for uid in sorted(builder.uids()):
+ if uid < 32: continue
+ c = builder.char(uid)
+ # iterate over all permutations of feature settings that might affect this character:
+ for featlist in builder.permuteFeatures(uids = (uid,)):
+ ftml.setFeatures(featlist)
+ builder.render((uid,), ftml)
+ # Don't close test -- collect consecutive encoded chars in a single row
+ ftml.clearFeatures()
+ for langID in sorted(c.langs):
+ ftml.setLang(langID)
+ builder.render((uid,), ftml)
+ ftml.clearLang()
+
+ # Add unencoded specials and ligatures -- i.e., things with a sequence of USVs in the glyph_data:
+ ftml.startTestGroup('Specials & ligatures from glyph_data')
+ for basename in sorted(builder.specials()):
+ special = builder.special(basename)
+ # iterate over all permutations of feature settings that might affect this special
+ for featlist in builder.permuteFeatures(uids = special.uids):
+ ftml.setFeatures(featlist)
+ builder.render(special.uids, ftml)
+ # close test so each special is on its own row:
+ ftml.closeTest()
+ ftml.clearFeatures()
+ if len(special.langs):
+ for langID in sorted(special.langs):
+ ftml.setLang(langID)
+ builder.render(special.uids, ftml)
+ ftml.closeTest()
+ ftml.clearLang()
+
+ # Add Lam-Alef data manually
+ ftml.startTestGroup('Lam-Alef')
+ # generate list of lam and alef characters that should be in the font:
+ lamlist = list(filter(lambda x: x in builder.uids(), (0x0644, 0x06B5, 0x06B6, 0x06B7, 0x06B8, 0x076A, 0x08A6)))
+ aleflist = list(filter(lambda x: x in builder.uids(), (0x0627, 0x0622, 0x0623, 0x0625, 0x0671, 0x0672, 0x0673, 0x0675, 0x0773, 0x0774)))
+ # iterate over all combinations:
+ for lam in lamlist:
+ for alef in aleflist:
+ for featlist in builder.permuteFeatures(uids = (lam, alef)):
+ ftml.setFeatures(featlist)
+ builder.render((lam,alef), ftml)
+ # close test so each combination is on its own row:
+ ftml.closeTest()
+ ftml.clearFeatures()
+
+ if test.lower().startswith("diac"):
+ # Diac attachment:
+
+ # Representative base and diac chars:
+ repDiac = list(filter(lambda x: x in builder.uids(), (0x064E, 0x0650, 0x065E, 0x0670, 0x0616, 0x06E3, 0x08F0, 0x08F2)))
+ repBase = list(filter(lambda x: x in builder.uids(), (0x0627, 0x0628, 0x062B, 0x0647, 0x064A, 0x77F, 0x08AC)))
+
+ ftml.startTestGroup('Representative diacritics on all bases that take diacritics')
+ for uid in sorted(builder.uids()):
+ # ignore some I don't care about:
+ if uid < 32 or uid in (0xAA, 0xBA): continue
+ c = builder.char(uid)
+ # Always process Lo, but others only if that take marks:
+ if c.general == 'Lo' or c.isBase:
+ for diac in repDiac:
+ for featlist in builder.permuteFeatures(uids = (uid,diac)):
+ ftml.setFeatures(featlist)
+ # Don't automatically separate connecting or mirrored forms into separate lines:
+ builder.render((uid,diac), ftml, addBreaks = False)
+ ftml.clearFeatures()
+ ftml.closeTest()
+
+ ftml.startTestGroup('All diacritics on representative bases')
+ for uid in sorted(builder.uids()):
+ # ignore non-ABS marks
+ if uid < 0x600 or uid in range(0xFE00, 0xFE10): continue
+ c = builder.char(uid)
+ if c.general == 'Mn':
+ for base in repBase:
+ for featlist in builder.permuteFeatures(uids = (uid,base)):
+ ftml.setFeatures(featlist)
+ builder.render((base,uid), ftml, keyUID = uid, addBreaks = False)
+ ftml.clearFeatures()
+ ftml.closeTest()
+
+ ftml.startTestGroup('Special cases')
+ builder.render((0x064A, 0x065E), ftml, comment="Yeh + Fatha should keep dots")
+ builder.render((0x064A, 0x0654), ftml, comment="Yeh + Hamza should loose dots")
+ ftml.closeTest()
+
+ # Write the output ftml file
+ ftml.writeFile(args.output)
+
+
+def cmd() : execute("UFO",doit,argspec)
+if __name__ == "__main__": cmd()