aboutsummaryrefslogtreecommitdiff
path: root/Tests/voltLib/volttofea_test.py
diff options
context:
space:
mode:
Diffstat (limited to 'Tests/voltLib/volttofea_test.py')
-rw-r--r--Tests/voltLib/volttofea_test.py1253
1 files changed, 1253 insertions, 0 deletions
diff --git a/Tests/voltLib/volttofea_test.py b/Tests/voltLib/volttofea_test.py
new file mode 100644
index 00000000..0d8d8d28
--- /dev/null
+++ b/Tests/voltLib/volttofea_test.py
@@ -0,0 +1,1253 @@
+import pathlib
+import shutil
+import tempfile
+import unittest
+from io import StringIO
+
+from fontTools.voltLib.voltToFea import VoltToFea
+
+DATADIR = pathlib.Path(__file__).parent / "data"
+
+
+class ToFeaTest(unittest.TestCase):
+ @classmethod
+ def setup_class(cls):
+ cls.tempdir = None
+ cls.num_tempfiles = 0
+
+ @classmethod
+ def teardown_class(cls):
+ if cls.tempdir:
+ shutil.rmtree(cls.tempdir, ignore_errors=True)
+
+ @classmethod
+ def temp_path(cls):
+ if not cls.tempdir:
+ cls.tempdir = pathlib.Path(tempfile.mkdtemp())
+ cls.num_tempfiles += 1
+ return cls.tempdir / f"tmp{cls.num_tempfiles}"
+
+ def test_def_glyph_base(self):
+ fea = self.parse('DEF_GLYPH ".notdef" ID 0 TYPE BASE END_GLYPH')
+ self.assertEqual(
+ fea,
+ "@GDEF_base = [.notdef];\n"
+ "table GDEF {\n"
+ " GlyphClassDef @GDEF_base, , , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_glyph_base_2_components(self):
+ fea = self.parse(
+ 'DEF_GLYPH "glyphBase" ID 320 TYPE BASE COMPONENTS 2 END_GLYPH'
+ )
+ self.assertEqual(
+ fea,
+ "@GDEF_base = [glyphBase];\n"
+ "table GDEF {\n"
+ " GlyphClassDef @GDEF_base, , , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_glyph_ligature_2_components(self):
+ fea = self.parse('DEF_GLYPH "f_f" ID 320 TYPE LIGATURE COMPONENTS 2 END_GLYPH')
+ self.assertEqual(
+ fea,
+ "@GDEF_ligature = [f_f];\n"
+ "table GDEF {\n"
+ " GlyphClassDef , @GDEF_ligature, , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_glyph_mark(self):
+ fea = self.parse('DEF_GLYPH "brevecomb" ID 320 TYPE MARK END_GLYPH')
+ self.assertEqual(
+ fea,
+ "@GDEF_mark = [brevecomb];\n"
+ "table GDEF {\n"
+ " GlyphClassDef , , @GDEF_mark, ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_glyph_component(self):
+ fea = self.parse('DEF_GLYPH "f.f_f" ID 320 TYPE COMPONENT END_GLYPH')
+ self.assertEqual(
+ fea,
+ "@GDEF_component = [f.f_f];\n"
+ "table GDEF {\n"
+ " GlyphClassDef , , , @GDEF_component;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_glyph_no_type(self):
+ fea = self.parse('DEF_GLYPH "glyph20" ID 20 END_GLYPH')
+ self.assertEqual(fea, "")
+
+ def test_def_glyph_case_sensitive(self):
+ fea = self.parse(
+ 'DEF_GLYPH "A" ID 3 UNICODE 65 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "a" ID 4 UNICODE 97 TYPE BASE END_GLYPH\n'
+ )
+ self.assertEqual(
+ fea,
+ "@GDEF_base = [A a];\n"
+ "table GDEF {\n"
+ " GlyphClassDef @GDEF_base, , , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_def_group_glyphs(self):
+ fea = self.parse(
+ 'DEF_GROUP "aaccented"\n'
+ 'ENUM GLYPH "aacute" GLYPH "abreve" GLYPH "acircumflex" '
+ 'GLYPH "adieresis" GLYPH "ae" GLYPH "agrave" GLYPH "amacron" '
+ 'GLYPH "aogonek" GLYPH "aring" GLYPH "atilde" END_ENUM\n'
+ "END_GROUP\n"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@aaccented = [aacute abreve acircumflex adieresis ae"
+ " agrave amacron aogonek aring atilde];",
+ )
+
+ def test_def_group_groups(self):
+ fea = self.parse(
+ 'DEF_GROUP "Group1"\n'
+ 'ENUM GLYPH "a" GLYPH "b" GLYPH "c" GLYPH "d" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "Group2"\n'
+ 'ENUM GLYPH "e" GLYPH "f" GLYPH "g" GLYPH "h" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "TestGroup"\n'
+ 'ENUM GROUP "Group1" GROUP "Group2" END_ENUM\n'
+ "END_GROUP\n"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@Group1 = [a b c d];\n"
+ "@Group2 = [e f g h];\n"
+ "@TestGroup = [@Group1 @Group2];",
+ )
+
+ def test_def_group_groups_not_yet_defined(self):
+ fea = self.parse(
+ 'DEF_GROUP "Group1"\n'
+ 'ENUM GLYPH "a" GLYPH "b" GLYPH "c" GLYPH "d" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "TestGroup1"\n'
+ 'ENUM GROUP "Group1" GROUP "Group2" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "TestGroup2"\n'
+ 'ENUM GROUP "Group2" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "TestGroup3"\n'
+ 'ENUM GROUP "Group2" GROUP "Group1" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "Group2"\n'
+ 'ENUM GLYPH "e" GLYPH "f" GLYPH "g" GLYPH "h" END_ENUM\n'
+ "END_GROUP\n"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@Group1 = [a b c d];\n"
+ "@Group2 = [e f g h];\n"
+ "@TestGroup1 = [@Group1 @Group2];\n"
+ "@TestGroup2 = [@Group2];\n"
+ "@TestGroup3 = [@Group2 @Group1];",
+ )
+
+ def test_def_group_glyphs_and_group(self):
+ fea = self.parse(
+ 'DEF_GROUP "aaccented"\n'
+ 'ENUM GLYPH "aacute" GLYPH "abreve" GLYPH "acircumflex" '
+ 'GLYPH "adieresis" GLYPH "ae" GLYPH "agrave" GLYPH "amacron" '
+ 'GLYPH "aogonek" GLYPH "aring" GLYPH "atilde" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "KERN_lc_a_2ND"\n'
+ 'ENUM GLYPH "a" GROUP "aaccented" END_ENUM\n'
+ "END_GROUP"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@aaccented = [aacute abreve acircumflex adieresis ae"
+ " agrave amacron aogonek aring atilde];\n"
+ "@KERN_lc_a_2ND = [a @aaccented];",
+ )
+
+ def test_def_group_range(self):
+ fea = self.parse(
+ 'DEF_GLYPH "a" ID 163 UNICODE 97 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "agrave" ID 194 UNICODE 224 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "aacute" ID 195 UNICODE 225 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "acircumflex" ID 196 UNICODE 226 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "atilde" ID 197 UNICODE 227 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "c" ID 165 UNICODE 99 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "ccaron" ID 209 UNICODE 269 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "ccedilla" ID 210 UNICODE 231 TYPE BASE END_GLYPH\n'
+ 'DEF_GLYPH "cdotaccent" ID 210 UNICODE 267 TYPE BASE END_GLYPH\n'
+ 'DEF_GROUP "KERN_lc_a_2ND"\n'
+ 'ENUM RANGE "a" TO "atilde" GLYPH "b" RANGE "c" TO "cdotaccent" '
+ "END_ENUM\n"
+ "END_GROUP"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@KERN_lc_a_2ND = [a - atilde b c - cdotaccent];\n"
+ "@GDEF_base = [a agrave aacute acircumflex atilde c"
+ " ccaron ccedilla cdotaccent];\n"
+ "table GDEF {\n"
+ " GlyphClassDef @GDEF_base, , , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_script_without_langsys(self):
+ fea = self.parse('DEF_SCRIPT NAME "Latin" TAG "latn"\n' "END_SCRIPT")
+ self.assertEqual(fea, "")
+
+ def test_langsys_normal(self):
+ fea = self.parse(
+ 'DEF_SCRIPT NAME "Latin" TAG "latn"\n'
+ 'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n'
+ "END_LANGSYS\n"
+ 'DEF_LANGSYS NAME "Moldavian" TAG "MOL "\n'
+ "END_LANGSYS\n"
+ "END_SCRIPT"
+ )
+ self.assertEqual(fea, "")
+
+ def test_langsys_no_script_name(self):
+ fea = self.parse(
+ 'DEF_SCRIPT TAG "latn"\n'
+ 'DEF_LANGSYS NAME "Default" TAG "dflt"\n'
+ "END_LANGSYS\n"
+ "END_SCRIPT"
+ )
+ self.assertEqual(fea, "")
+
+ def test_langsys_lang_in_separate_scripts(self):
+ fea = self.parse(
+ 'DEF_SCRIPT NAME "Default" TAG "DFLT"\n'
+ 'DEF_LANGSYS NAME "Default" TAG "dflt"\n'
+ "END_LANGSYS\n"
+ 'DEF_LANGSYS NAME "Default" TAG "ROM "\n'
+ "END_LANGSYS\n"
+ "END_SCRIPT\n"
+ 'DEF_SCRIPT NAME "Latin" TAG "latn"\n'
+ 'DEF_LANGSYS NAME "Default" TAG "dflt"\n'
+ "END_LANGSYS\n"
+ 'DEF_LANGSYS NAME "Default" TAG "ROM "\n'
+ "END_LANGSYS\n"
+ "END_SCRIPT"
+ )
+ self.assertEqual(fea, "")
+
+ def test_langsys_no_lang_name(self):
+ fea = self.parse(
+ 'DEF_SCRIPT NAME "Latin" TAG "latn"\n'
+ 'DEF_LANGSYS TAG "dflt"\n'
+ "END_LANGSYS\n"
+ "END_SCRIPT"
+ )
+ self.assertEqual(fea, "")
+
+ def test_feature(self):
+ fea = self.parse(
+ 'DEF_SCRIPT NAME "Latin" TAG "latn"\n'
+ 'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n'
+ 'DEF_FEATURE NAME "Fractions" TAG "frac"\n'
+ 'LOOKUP "fraclookup"\n'
+ "END_FEATURE\n"
+ "END_LANGSYS\n"
+ "END_SCRIPT\n"
+ 'DEF_LOOKUP "fraclookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "one" GLYPH "slash" GLYPH "two"\n'
+ 'WITH GLYPH "one_slash_two.frac"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup fraclookup {\n"
+ " sub one slash two by one_slash_two.frac;\n"
+ "} fraclookup;\n"
+ "\n"
+ "# Features\n"
+ "feature frac {\n"
+ " script latn;\n"
+ " language ROM exclude_dflt;\n"
+ " lookup fraclookup;\n"
+ "} frac;\n",
+ )
+
+ def test_feature_sub_lookups(self):
+ fea = self.parse(
+ 'DEF_SCRIPT NAME "Latin" TAG "latn"\n'
+ 'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n'
+ 'DEF_FEATURE NAME "Fractions" TAG "frac"\n'
+ 'LOOKUP "fraclookup\\1"\n'
+ 'LOOKUP "fraclookup\\1"\n'
+ "END_FEATURE\n"
+ "END_LANGSYS\n"
+ "END_SCRIPT\n"
+ 'DEF_LOOKUP "fraclookup\\1" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "one" GLYPH "slash" GLYPH "two"\n'
+ 'WITH GLYPH "one_slash_two.frac"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION\n"
+ 'DEF_LOOKUP "fraclookup\\2" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "one" GLYPH "slash" GLYPH "three"\n'
+ 'WITH GLYPH "one_slash_three.frac"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup fraclookup {\n"
+ " lookupflag RightToLeft;\n"
+ " # fraclookup\\1\n"
+ " sub one slash two by one_slash_two.frac;\n"
+ " subtable;\n"
+ " # fraclookup\\2\n"
+ " sub one slash three by one_slash_three.frac;\n"
+ "} fraclookup;\n"
+ "\n"
+ "# Features\n"
+ "feature frac {\n"
+ " script latn;\n"
+ " language ROM exclude_dflt;\n"
+ " lookup fraclookup;\n"
+ "} frac;\n",
+ )
+
+ def test_lookup_comment(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "smcp" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ 'COMMENTS "Smallcaps lookup for testing"\n'
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "a"\n'
+ 'WITH GLYPH "a.sc"\n'
+ "END_SUB\n"
+ 'SUB GLYPH "b"\n'
+ 'WITH GLYPH "b.sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup smcp {\n"
+ " # Smallcaps lookup for testing\n"
+ " sub a by a.sc;\n"
+ " sub b by b.sc;\n"
+ "} smcp;\n",
+ )
+
+ def test_substitution_single(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "smcp" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "a"\n'
+ 'WITH GLYPH "a.sc"\n'
+ "END_SUB\n"
+ 'SUB GLYPH "b"\n'
+ 'WITH GLYPH "b.sc"\n'
+ "END_SUB\n"
+ "SUB WITH\n" # Empty substitution, will be ignored
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup smcp {\n"
+ " sub a by a.sc;\n"
+ " sub b by b.sc;\n"
+ "} smcp;\n",
+ )
+
+ def test_substitution_single_in_context(self):
+ fea = self.parse(
+ 'DEF_GROUP "Denominators" ENUM GLYPH "one.dnom" GLYPH "two.dnom" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "fracdnom" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ 'IN_CONTEXT LEFT ENUM GROUP "Denominators" GLYPH "fraction" '
+ "END_ENUM\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "one"\n'
+ 'WITH GLYPH "one.dnom"\n'
+ "END_SUB\n"
+ 'SUB GLYPH "two"\n'
+ 'WITH GLYPH "two.dnom"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@Denominators = [one.dnom two.dnom];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup fracdnom {\n"
+ " sub [@Denominators fraction] one' by one.dnom;\n"
+ " sub [@Denominators fraction] two' by two.dnom;\n"
+ "} fracdnom;\n",
+ )
+
+ def test_substitution_single_in_contexts(self):
+ fea = self.parse(
+ 'DEF_GROUP "Hebrew" ENUM GLYPH "uni05D0" GLYPH "uni05D1" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "HebrewCurrency" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ 'RIGHT GROUP "Hebrew"\n'
+ 'RIGHT GLYPH "one.Hebr"\n'
+ "END_CONTEXT\n"
+ "IN_CONTEXT\n"
+ 'LEFT GROUP "Hebrew"\n'
+ 'LEFT GLYPH "one.Hebr"\n'
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "dollar"\n'
+ 'WITH GLYPH "dollar.Hebr"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@Hebrew = [uni05D0 uni05D1];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup HebrewCurrency {\n"
+ " sub dollar' @Hebrew one.Hebr by dollar.Hebr;\n"
+ " sub @Hebrew one.Hebr dollar' by dollar.Hebr;\n"
+ "} HebrewCurrency;\n",
+ )
+
+ def test_substitution_single_except_context(self):
+ fea = self.parse(
+ 'DEF_GROUP "Hebrew" ENUM GLYPH "uni05D0" GLYPH "uni05D1" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "HebrewCurrency" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "EXCEPT_CONTEXT\n"
+ 'RIGHT GROUP "Hebrew"\n'
+ 'RIGHT GLYPH "one.Hebr"\n'
+ "END_CONTEXT\n"
+ "IN_CONTEXT\n"
+ 'LEFT GROUP "Hebrew"\n'
+ 'LEFT GLYPH "one.Hebr"\n'
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "dollar"\n'
+ 'WITH GLYPH "dollar.Hebr"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@Hebrew = [uni05D0 uni05D1];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup HebrewCurrency {\n"
+ " ignore sub dollar' @Hebrew one.Hebr;\n"
+ " sub @Hebrew one.Hebr dollar' by dollar.Hebr;\n"
+ "} HebrewCurrency;\n",
+ )
+
+ def test_substitution_skip_base(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" SKIP_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [marka markb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag IgnoreBaseGlyphs;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_process_base(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [marka markb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_process_marks_all(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS "ALL"'
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [marka markb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_process_marks_none(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS "NONE"'
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [marka markb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag IgnoreMarks;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_skip_marks(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE SKIP_MARKS '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [marka markb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag IgnoreMarks;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_mark_attachment(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "acutecmb" GLYPH "gravecmb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE '
+ 'PROCESS_MARKS "SomeMarks" \n'
+ "DIRECTION RTL\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [acutecmb gravecmb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag RightToLeft MarkAttachmentType"
+ " @SomeMarks;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_mark_glyph_set(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "acutecmb" GLYPH "gravecmb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE '
+ 'PROCESS_MARKS MARK_GLYPH_SET "SomeMarks" \n'
+ "DIRECTION RTL\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [acutecmb gravecmb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag RightToLeft UseMarkFilteringSet"
+ " @SomeMarks;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_process_all_marks(self):
+ fea = self.parse(
+ 'DEF_GROUP "SomeMarks" ENUM GLYPH "acutecmb" GLYPH "gravecmb" '
+ "END_ENUM END_GROUP\n"
+ 'DEF_LOOKUP "SomeSub" PROCESS_BASE '
+ "PROCESS_MARKS ALL \n"
+ "DIRECTION RTL\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "A"\n'
+ 'WITH GLYPH "A.c2sc"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@SomeMarks = [acutecmb gravecmb];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup SomeSub {\n"
+ " lookupflag RightToLeft;\n"
+ " sub A by A.c2sc;\n"
+ "} SomeSub;\n",
+ )
+
+ def test_substitution_no_reversal(self):
+ # TODO: check right context with no reversal
+ fea = self.parse(
+ 'DEF_LOOKUP "Lookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ 'RIGHT ENUM GLYPH "a" GLYPH "b" END_ENUM\n'
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "a"\n'
+ 'WITH GLYPH "a.alt"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup Lookup {\n"
+ " sub a' [a b] by a.alt;\n"
+ "} Lookup;\n",
+ )
+
+ def test_substitution_reversal(self):
+ fea = self.parse(
+ 'DEF_GROUP "DFLT_Num_standardFigures"\n'
+ 'ENUM GLYPH "zero" GLYPH "one" GLYPH "two" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "DFLT_Num_numerators"\n'
+ 'ENUM GLYPH "zero.numr" GLYPH "one.numr" GLYPH "two.numr" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_LOOKUP "RevLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR REVERSAL\n"
+ "IN_CONTEXT\n"
+ 'RIGHT ENUM GLYPH "a" GLYPH "b" END_ENUM\n'
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GROUP "DFLT_Num_standardFigures"\n'
+ 'WITH GROUP "DFLT_Num_numerators"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@DFLT_Num_standardFigures = [zero one two];\n"
+ "@DFLT_Num_numerators = [zero.numr one.numr two.numr];\n"
+ "\n"
+ "# Lookups\n"
+ "lookup RevLookup {\n"
+ " rsub @DFLT_Num_standardFigures' [a b] by @DFLT_Num_numerators;\n"
+ "} RevLookup;\n",
+ )
+
+ def test_substitution_single_to_multiple(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "ccmp" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "aacute"\n'
+ 'WITH GLYPH "a" GLYPH "acutecomb"\n'
+ "END_SUB\n"
+ 'SUB GLYPH "agrave"\n'
+ 'WITH GLYPH "a" GLYPH "gravecomb"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup ccmp {\n"
+ " sub aacute by a acutecomb;\n"
+ " sub agrave by a gravecomb;\n"
+ "} ccmp;\n",
+ )
+
+ def test_substitution_multiple_to_single(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "liga" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB GLYPH "f" GLYPH "i"\n'
+ 'WITH GLYPH "f_i"\n'
+ "END_SUB\n"
+ 'SUB GLYPH "f" GLYPH "t"\n'
+ 'WITH GLYPH "f_t"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup liga {\n"
+ " sub f i by f_i;\n"
+ " sub f t by f_t;\n"
+ "} liga;\n",
+ )
+
+ def test_substitution_reverse_chaining_single(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "numr" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR REVERSAL\n"
+ "IN_CONTEXT\n"
+ "RIGHT ENUM "
+ 'GLYPH "fraction" '
+ 'RANGE "zero.numr" TO "nine.numr" '
+ "END_ENUM\n"
+ "END_CONTEXT\n"
+ "AS_SUBSTITUTION\n"
+ 'SUB RANGE "zero" TO "nine"\n'
+ 'WITH RANGE "zero.numr" TO "nine.numr"\n'
+ "END_SUB\n"
+ "END_SUBSTITUTION"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup numr {\n"
+ " rsub zero - nine' [fraction zero.numr - nine.numr] by zero.numr - nine.numr;\n"
+ "} numr;\n",
+ )
+
+ # GPOS
+ # ATTACH_CURSIVE
+ # ATTACH
+ # ADJUST_PAIR
+ # ADJUST_SINGLE
+ def test_position_attach(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "anchor_top" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ 'ATTACH GLYPH "a" GLYPH "e"\n'
+ 'TO GLYPH "acutecomb" AT ANCHOR "top" '
+ 'GLYPH "gravecomb" AT ANCHOR "top"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_ANCHOR "MARK_top" ON 120 GLYPH acutecomb COMPONENT 1 '
+ "AT POS DX 0 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "MARK_top" ON 121 GLYPH gravecomb COMPONENT 1 '
+ "AT POS DX 0 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "top" ON 31 GLYPH a COMPONENT 1 '
+ "AT POS DX 210 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "top" ON 35 GLYPH e COMPONENT 1 '
+ "AT POS DX 215 DY 450 END_POS END_ANCHOR\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "markClass acutecomb <anchor 0 450> @top;\n"
+ "markClass gravecomb <anchor 0 450> @top;\n"
+ "\n"
+ "# Lookups\n"
+ "lookup anchor_top {\n"
+ " lookupflag RightToLeft;\n"
+ " pos base a\n"
+ " <anchor 210 450> mark @top;\n"
+ " pos base e\n"
+ " <anchor 215 450> mark @top;\n"
+ "} anchor_top;\n",
+ )
+
+ def test_position_attach_mkmk(self):
+ fea = self.parse(
+ 'DEF_GLYPH "brevecomb" ID 1 TYPE MARK END_GLYPH\n'
+ 'DEF_GLYPH "gravecomb" ID 2 TYPE MARK END_GLYPH\n'
+ 'DEF_LOOKUP "anchor_top" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ 'ATTACH GLYPH "gravecomb"\n'
+ 'TO GLYPH "acutecomb" AT ANCHOR "top"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_ANCHOR "MARK_top" ON 1 GLYPH acutecomb COMPONENT 1 '
+ "AT POS DX 0 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "top" ON 2 GLYPH gravecomb COMPONENT 1 '
+ "AT POS DX 210 DY 450 END_POS END_ANCHOR\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "markClass acutecomb <anchor 0 450> @top;\n"
+ "\n"
+ "# Lookups\n"
+ "lookup anchor_top {\n"
+ " lookupflag RightToLeft;\n"
+ " pos mark gravecomb\n"
+ " <anchor 210 450> mark @top;\n"
+ "} anchor_top;\n"
+ "\n"
+ "@GDEF_mark = [brevecomb gravecomb];\n"
+ "table GDEF {\n"
+ " GlyphClassDef , , @GDEF_mark, ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_position_attach_in_context(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "test" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ 'EXCEPT_CONTEXT LEFT GLYPH "a" END_CONTEXT\n'
+ "AS_POSITION\n"
+ 'ATTACH GLYPH "a"\n'
+ 'TO GLYPH "acutecomb" AT ANCHOR "top" '
+ 'GLYPH "gravecomb" AT ANCHOR "top"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_ANCHOR "MARK_top" ON 120 GLYPH acutecomb COMPONENT 1 '
+ "AT POS DX 0 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "MARK_top" ON 121 GLYPH gravecomb COMPONENT 1 '
+ "AT POS DX 0 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "top" ON 31 GLYPH a COMPONENT 1 '
+ "AT POS DX 210 DY 450 END_POS END_ANCHOR\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "markClass acutecomb <anchor 0 450> @top;\n"
+ "markClass gravecomb <anchor 0 450> @top;\n"
+ "\n"
+ "# Lookups\n"
+ "lookup test_target {\n"
+ " pos base a\n"
+ " <anchor 210 450> mark @top;\n"
+ "} test_target;\n"
+ "\n"
+ "lookup test {\n"
+ " lookupflag RightToLeft;\n"
+ " ignore pos a [acutecomb gravecomb]';\n"
+ " pos [acutecomb gravecomb]' lookup test_target;\n"
+ "} test;\n",
+ )
+
+ def test_position_attach_cursive(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "SomeLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ 'ATTACH_CURSIVE EXIT GLYPH "a" GLYPH "b" '
+ 'ENTER GLYPH "a" GLYPH "c"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_ANCHOR "exit" ON 1 GLYPH a COMPONENT 1 AT POS END_POS END_ANCHOR\n'
+ 'DEF_ANCHOR "entry" ON 1 GLYPH a COMPONENT 1 AT POS END_POS END_ANCHOR\n'
+ 'DEF_ANCHOR "exit" ON 2 GLYPH b COMPONENT 1 AT POS END_POS END_ANCHOR\n'
+ 'DEF_ANCHOR "entry" ON 3 GLYPH c COMPONENT 1 AT POS END_POS END_ANCHOR\n'
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup SomeLookup {\n"
+ " lookupflag RightToLeft;\n"
+ " pos cursive a <anchor 0 0> <anchor 0 0>;\n"
+ " pos cursive c <anchor 0 0> <anchor NULL>;\n"
+ " pos cursive b <anchor NULL> <anchor 0 0>;\n"
+ "} SomeLookup;\n",
+ )
+
+ def test_position_adjust_pair(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "kern1" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION RTL\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ "ADJUST_PAIR\n"
+ ' FIRST GLYPH "A" FIRST GLYPH "V"\n'
+ ' SECOND GLYPH "A" SECOND GLYPH "V"\n'
+ " 1 2 BY POS ADV -30 END_POS POS END_POS\n"
+ " 2 1 BY POS ADV -25 END_POS POS END_POS\n"
+ "END_ADJUST\n"
+ "END_POSITION\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup kern1 {\n"
+ " lookupflag RightToLeft;\n"
+ " enum pos A V -30;\n"
+ " enum pos V A -25;\n"
+ "} kern1;\n",
+ )
+
+ def test_position_adjust_pair_in_context(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "kern1" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ 'EXCEPT_CONTEXT LEFT GLYPH "A" END_CONTEXT\n'
+ "AS_POSITION\n"
+ "ADJUST_PAIR\n"
+ ' FIRST GLYPH "A" FIRST GLYPH "V"\n'
+ ' SECOND GLYPH "A" SECOND GLYPH "V"\n'
+ " 2 1 BY POS ADV -25 END_POS POS END_POS\n"
+ "END_ADJUST\n"
+ "END_POSITION\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup kern1_target {\n"
+ " enum pos V A -25;\n"
+ "} kern1_target;\n"
+ "\n"
+ "lookup kern1 {\n"
+ " ignore pos A V' A';\n"
+ " pos V' lookup kern1_target A' lookup kern1_target;\n"
+ "} kern1;\n",
+ )
+
+ def test_position_adjust_single(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "TestLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ "ADJUST_SINGLE"
+ ' GLYPH "glyph1" BY POS ADV 0 DX 123 END_POS\n'
+ ' GLYPH "glyph2" BY POS ADV 0 DX 456 END_POS\n'
+ "END_ADJUST\n"
+ "END_POSITION\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup TestLookup {\n"
+ " pos glyph1 <123 0 0 0>;\n"
+ " pos glyph2 <456 0 0 0>;\n"
+ "} TestLookup;\n",
+ )
+
+ def test_position_adjust_single_in_context(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "TestLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "EXCEPT_CONTEXT\n"
+ 'LEFT GLYPH "leftGlyph"\n'
+ 'RIGHT GLYPH "rightGlyph"\n'
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ "ADJUST_SINGLE"
+ ' GLYPH "glyph1" BY POS ADV 0 DX 123 END_POS\n'
+ ' GLYPH "glyph2" BY POS ADV 0 DX 456 END_POS\n'
+ "END_ADJUST\n"
+ "END_POSITION\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup TestLookup_target {\n"
+ " pos glyph1 <123 0 0 0>;\n"
+ " pos glyph2 <456 0 0 0>;\n"
+ "} TestLookup_target;\n"
+ "\n"
+ "lookup TestLookup {\n"
+ " ignore pos leftGlyph [glyph1 glyph2]' rightGlyph;\n"
+ " pos [glyph1 glyph2]' lookup TestLookup_target;\n"
+ "} TestLookup;\n",
+ )
+
+ def test_def_anchor(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "TestLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ 'ATTACH GLYPH "a"\n'
+ 'TO GLYPH "acutecomb" AT ANCHOR "top"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_ANCHOR "top" ON 120 GLYPH a '
+ "COMPONENT 1 AT POS DX 250 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "MARK_top" ON 120 GLYPH acutecomb '
+ "COMPONENT 1 AT POS DX 0 DY 450 END_POS END_ANCHOR"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "markClass acutecomb <anchor 0 450> @top;\n"
+ "\n"
+ "# Lookups\n"
+ "lookup TestLookup {\n"
+ " pos base a\n"
+ " <anchor 250 450> mark @top;\n"
+ "} TestLookup;\n",
+ )
+
+ def test_def_anchor_multi_component(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "TestLookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ 'ATTACH GLYPH "f_f"\n'
+ 'TO GLYPH "acutecomb" AT ANCHOR "top"\n'
+ "END_ATTACH\n"
+ "END_POSITION\n"
+ 'DEF_GLYPH "f_f" ID 120 TYPE LIGATURE COMPONENTS 2 END_GLYPH\n'
+ 'DEF_ANCHOR "top" ON 120 GLYPH f_f '
+ "COMPONENT 1 AT POS DX 250 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "top" ON 120 GLYPH f_f '
+ "COMPONENT 2 AT POS DX 450 DY 450 END_POS END_ANCHOR\n"
+ 'DEF_ANCHOR "MARK_top" ON 120 GLYPH acutecomb '
+ "COMPONENT 1 AT POS END_POS END_ANCHOR"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "markClass acutecomb <anchor 0 0> @top;\n"
+ "\n"
+ "# Lookups\n"
+ "lookup TestLookup {\n"
+ " pos ligature f_f\n"
+ " <anchor 250 450> mark @top\n"
+ " ligComponent\n"
+ " <anchor 450 450> mark @top;\n"
+ "} TestLookup;\n"
+ "\n"
+ "@GDEF_ligature = [f_f];\n"
+ "table GDEF {\n"
+ " GlyphClassDef , @GDEF_ligature, , ;\n"
+ "} GDEF;\n",
+ )
+
+ def test_anchor_adjust_device(self):
+ fea = self.parse(
+ 'DEF_ANCHOR "MARK_top" ON 123 GLYPH diacglyph '
+ "COMPONENT 1 AT POS DX 0 DY 456 ADJUST_BY 12 AT 34 "
+ "ADJUST_BY 56 AT 78 END_POS END_ANCHOR"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Mark classes\n"
+ "#markClass diacglyph <anchor 0 456 <device NULL>"
+ " <device 34 12, 78 56>> @top;",
+ )
+
+ def test_use_extension(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "kern1" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR\n"
+ "IN_CONTEXT\n"
+ "END_CONTEXT\n"
+ "AS_POSITION\n"
+ "ADJUST_PAIR\n"
+ ' FIRST GLYPH "A" FIRST GLYPH "V"\n'
+ ' SECOND GLYPH "A" SECOND GLYPH "V"\n'
+ " 1 2 BY POS ADV -30 END_POS POS END_POS\n"
+ " 2 1 BY POS ADV -25 END_POS POS END_POS\n"
+ "END_ADJUST\n"
+ "END_POSITION\n"
+ "COMPILER_USEEXTENSIONLOOKUPS\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup kern1 useExtension {\n"
+ " enum pos A V -30;\n"
+ " enum pos V A -25;\n"
+ "} kern1;\n",
+ )
+
+ def test_unsupported_compiler_flags(self):
+ with self.assertLogs(level="WARNING") as logs:
+ fea = self.parse("CMAP_FORMAT 0 3 4")
+ self.assertEqual(fea, "")
+ self.assertEqual(
+ logs.output,
+ [
+ "WARNING:fontTools.voltLib.voltToFea:Unsupported setting ignored: CMAP_FORMAT"
+ ],
+ )
+
+ def test_sanitize_lookup_name(self):
+ fea = self.parse(
+ 'DEF_LOOKUP "Test Lookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR IN_CONTEXT END_CONTEXT\n"
+ "AS_POSITION ADJUST_PAIR END_ADJUST END_POSITION\n"
+ 'DEF_LOOKUP "Test-Lookup" PROCESS_BASE PROCESS_MARKS ALL '
+ "DIRECTION LTR IN_CONTEXT END_CONTEXT\n"
+ "AS_POSITION ADJUST_PAIR END_ADJUST END_POSITION\n"
+ )
+ self.assertEqual(
+ fea,
+ "\n# Lookups\n"
+ "lookup Test_Lookup {\n"
+ " \n"
+ "} Test_Lookup;\n"
+ "\n"
+ "lookup Test_Lookup_ {\n"
+ " \n"
+ "} Test_Lookup_;\n",
+ )
+
+ def test_sanitize_group_name(self):
+ fea = self.parse(
+ 'DEF_GROUP "aaccented glyphs"\n'
+ 'ENUM GLYPH "aacute" GLYPH "abreve" END_ENUM\n'
+ "END_GROUP\n"
+ 'DEF_GROUP "aaccented+glyphs"\n'
+ 'ENUM GLYPH "aacute" GLYPH "abreve" END_ENUM\n'
+ "END_GROUP\n"
+ )
+ self.assertEqual(
+ fea,
+ "# Glyph classes\n"
+ "@aaccented_glyphs = [aacute abreve];\n"
+ "@aaccented_glyphs_ = [aacute abreve];",
+ )
+
+ def test_cli_vtp(self):
+ vtp = DATADIR / "Nutso.vtp"
+ fea = DATADIR / "Nutso.fea"
+ self.cli(vtp, fea)
+
+ def test_group_order(self):
+ vtp = DATADIR / "NamdhinggoSIL1006.vtp"
+ fea = DATADIR / "NamdhinggoSIL1006.fea"
+ self.cli(vtp, fea)
+
+ def test_cli_ttf(self):
+ ttf = DATADIR / "Nutso.ttf"
+ fea = DATADIR / "Nutso.fea"
+ self.cli(ttf, fea)
+
+ def test_cli_ttf_no_TSIV(self):
+ from fontTools.voltLib.voltToFea import main as cli
+
+ ttf = DATADIR / "Empty.ttf"
+ temp = self.temp_path()
+ self.assertEqual(1, cli([str(ttf), str(temp)]))
+
+ def cli(self, source, fea):
+ from fontTools.voltLib.voltToFea import main as cli
+
+ temp = self.temp_path()
+ cli([str(source), str(temp)])
+ with temp.open() as f:
+ res = f.read()
+ with fea.open() as f:
+ ref = f.read()
+ self.assertEqual(ref, res)
+
+ def parse(self, text):
+ return VoltToFea(StringIO(text)).convert()
+
+
+if __name__ == "__main__":
+ import sys
+
+ sys.exit(unittest.main())