blob: abc02d3b0d4a4105e4dc9b635237180799ac71b1 [file] [log] [blame]
from fontTools.voltLib import ast
from fontTools.voltLib.error import VoltLibError
from fontTools.voltLib.parser import Parser
from io import StringIO
import unittest
class ParserTest(unittest.TestCase):
def __init__(self, methodName):
unittest.TestCase.__init__(self, methodName)
# Python 3 renamed assertRaisesRegexp to assertRaisesRegex,
# and fires deprecation warnings if a program uses the old name.
if not hasattr(self, "assertRaisesRegex"):
self.assertRaisesRegex = self.assertRaisesRegexp
def assertSubEqual(self, sub, glyph_ref, replacement_ref):
glyphs = [[g.glyph for g in v] for v in sub.mapping.keys()]
replacement = [[g.glyph for g in v] for v in sub.mapping.values()]
self.assertEqual(glyphs, glyph_ref)
self.assertEqual(replacement, replacement_ref)
def test_def_glyph_base(self):
[def_glyph] = self.parse(
'DEF_GLYPH ".notdef" ID 0 TYPE BASE END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
(".notdef", 0, None, "BASE", None),
)
def test_def_glyph_base_with_unicode(self):
[def_glyph] = self.parse(
'DEF_GLYPH "space" ID 3 UNICODE 32 TYPE BASE END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("space", 3, [0x0020], "BASE", None),
)
def test_def_glyph_base_with_unicodevalues(self):
[def_glyph] = self.parse_(
'DEF_GLYPH "CR" ID 2 UNICODEVALUES "U+0009" ' "TYPE BASE END_GLYPH"
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("CR", 2, [0x0009], "BASE", None),
)
def test_def_glyph_base_with_mult_unicodevalues(self):
[def_glyph] = self.parse(
'DEF_GLYPH "CR" ID 2 UNICODEVALUES "U+0009,U+000D" ' "TYPE BASE END_GLYPH"
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("CR", 2, [0x0009, 0x000D], "BASE", None),
)
def test_def_glyph_base_with_empty_unicodevalues(self):
[def_glyph] = self.parse_(
'DEF_GLYPH "i.locl" ID 269 UNICODEVALUES "" ' "TYPE BASE END_GLYPH"
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("i.locl", 269, None, "BASE", None),
)
def test_def_glyph_base_2_components(self):
[def_glyph] = self.parse(
'DEF_GLYPH "glyphBase" ID 320 TYPE BASE COMPONENTS 2 END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("glyphBase", 320, None, "BASE", 2),
)
def test_def_glyph_ligature_2_components(self):
[def_glyph] = self.parse(
'DEF_GLYPH "f_f" ID 320 TYPE LIGATURE COMPONENTS 2 END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("f_f", 320, None, "LIGATURE", 2),
)
def test_def_glyph_mark(self):
[def_glyph] = self.parse(
'DEF_GLYPH "brevecomb" ID 320 TYPE MARK END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("brevecomb", 320, None, "MARK", None),
)
def test_def_glyph_component(self):
[def_glyph] = self.parse(
'DEF_GLYPH "f.f_f" ID 320 TYPE COMPONENT END_GLYPH'
).statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("f.f_f", 320, None, "COMPONENT", None),
)
def test_def_glyph_no_type(self):
[def_glyph] = self.parse('DEF_GLYPH "glyph20" ID 20 END_GLYPH').statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
("glyph20", 20, None, None, None),
)
def test_def_glyph_case_sensitive(self):
def_glyphs = 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'
).statements
self.assertEqual(
(
def_glyphs[0].name,
def_glyphs[0].id,
def_glyphs[0].unicode,
def_glyphs[0].type,
def_glyphs[0].components,
),
("A", 3, [0x41], "BASE", None),
)
self.assertEqual(
(
def_glyphs[1].name,
def_glyphs[1].id,
def_glyphs[1].unicode,
def_glyphs[1].type,
def_glyphs[1].components,
),
("a", 4, [0x61], "BASE", None),
)
def test_def_group_glyphs(self):
[def_group] = 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"
).statements
self.assertEqual(
(def_group.name, def_group.enum.glyphSet()),
(
"aaccented",
(
"aacute",
"abreve",
"acircumflex",
"adieresis",
"ae",
"agrave",
"amacron",
"aogonek",
"aring",
"atilde",
),
),
)
def test_def_group_groups(self):
[group1, group2, test_group] = 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"
).statements
groups = [g.group for g in test_group.enum.enum]
self.assertEqual((test_group.name, groups), ("TestGroup", ["Group1", "Group2"]))
def test_def_group_groups_not_yet_defined(self):
[group1, test_group1, test_group2, test_group3, group2] = 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"
).statements
groups = [g.group for g in test_group1.enum.enum]
self.assertEqual(
(test_group1.name, groups), ("TestGroup1", ["Group1", "Group2"])
)
groups = [g.group for g in test_group2.enum.enum]
self.assertEqual((test_group2.name, groups), ("TestGroup2", ["Group2"]))
groups = [g.group for g in test_group3.enum.enum]
self.assertEqual(
(test_group3.name, groups), ("TestGroup3", ["Group2", "Group1"])
)
# def test_def_group_groups_undefined(self):
# with self.assertRaisesRegex(
# VoltLibError,
# r'Group "Group2" is used but undefined.'):
# [group1, test_group, group2] = self.parse(
# 'DEF_GROUP "Group1"\n'
# 'ENUM GLYPH "a" GLYPH "b" GLYPH "c" GLYPH "d" END_ENUM\n'
# 'END_GROUP\n'
# 'DEF_GROUP "TestGroup"\n'
# 'ENUM GROUP "Group1" GROUP "Group2" END_ENUM\n'
# 'END_GROUP\n'
# ).statements
def test_def_group_glyphs_and_group(self):
[def_group1, def_group2] = 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"
).statements
items = def_group2.enum.enum
self.assertEqual(
(def_group2.name, items[0].glyphSet(), items[1].group),
("KERN_lc_a_2ND", ("a",), "aaccented"),
)
def test_def_group_range(self):
def_group = 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"
).statements[-1]
self.assertEqual(
(def_group.name, def_group.enum.glyphSet()),
(
"KERN_lc_a_2ND",
(
"a",
"agrave",
"aacute",
"acircumflex",
"atilde",
"b",
"c",
"ccaron",
"ccedilla",
"cdotaccent",
),
),
)
def test_group_duplicate(self):
self.assertRaisesRegex(
VoltLibError,
'Glyph group "dupe" already defined, ' "group names are case insensitive",
self.parse,
'DEF_GROUP "dupe"\n'
'ENUM GLYPH "a" GLYPH "b" END_ENUM\n'
"END_GROUP\n"
'DEF_GROUP "dupe"\n'
'ENUM GLYPH "x" END_ENUM\n'
"END_GROUP",
)
def test_group_duplicate_case_insensitive(self):
self.assertRaisesRegex(
VoltLibError,
'Glyph group "Dupe" already defined, ' "group names are case insensitive",
self.parse,
'DEF_GROUP "dupe"\n'
'ENUM GLYPH "a" GLYPH "b" END_ENUM\n'
"END_GROUP\n"
'DEF_GROUP "Dupe"\n'
'ENUM GLYPH "x" END_ENUM\n'
"END_GROUP",
)
def test_script_without_langsys(self):
[script] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n' "END_SCRIPT"
).statements
self.assertEqual((script.name, script.tag, script.langs), ("Latin", "latn", []))
def test_langsys_normal(self):
[def_script] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n\n'
"END_LANGSYS\n"
'DEF_LANGSYS NAME "Moldavian" TAG "MOL "\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
self.assertEqual((def_script.name, def_script.tag), ("Latin", "latn"))
def_lang = def_script.langs[0]
self.assertEqual((def_lang.name, def_lang.tag), ("Romanian", "ROM "))
def_lang = def_script.langs[1]
self.assertEqual((def_lang.name, def_lang.tag), ("Moldavian", "MOL "))
def test_langsys_no_script_name(self):
[langsys] = self.parse(
'DEF_SCRIPT TAG "latn"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
self.assertEqual((langsys.name, langsys.tag), (None, "latn"))
lang = langsys.langs[0]
self.assertEqual((lang.name, lang.tag), ("Default", "dflt"))
def test_langsys_no_script_tag_fails(self):
with self.assertRaisesRegex(VoltLibError, r'.*Expected "TAG"'):
[langsys] = self.parse(
'DEF_SCRIPT NAME "Latin"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def test_langsys_duplicate_script(self):
with self.assertRaisesRegex(
VoltLibError,
'Script "DFLT" already defined, ' "script tags are case insensitive",
):
[langsys1, langsys2] = self.parse(
'DEF_SCRIPT NAME "Default" TAG "DFLT"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
"END_SCRIPT\n"
'DEF_SCRIPT TAG "DFLT"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def test_langsys_duplicate_lang(self):
with self.assertRaisesRegex(
VoltLibError,
'Language "dflt" already defined in script "DFLT", '
"language tags are case insensitive",
):
[langsys] = self.parse(
'DEF_SCRIPT NAME "Default" TAG "DFLT"\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n'
"END_LANGSYS\n"
'DEF_LANGSYS NAME "Default" TAG "dflt"\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def test_langsys_lang_in_separate_scripts(self):
[langsys1, langsys2] = self.parse(
'DEF_SCRIPT NAME "Default" TAG "DFLT"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
'DEF_LANGSYS NAME "Default" TAG "ROM "\n\n'
"END_LANGSYS\n"
"END_SCRIPT\n"
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS NAME "Default" TAG "dflt"\n\n'
"END_LANGSYS\n"
'DEF_LANGSYS NAME "Default" TAG "ROM "\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
self.assertEqual(
(langsys1.langs[0].tag, langsys1.langs[1].tag), ("dflt", "ROM ")
)
self.assertEqual(
(langsys2.langs[0].tag, langsys2.langs[1].tag), ("dflt", "ROM ")
)
def test_langsys_no_lang_name(self):
[langsys] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS TAG "dflt"\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
self.assertEqual((langsys.name, langsys.tag), ("Latin", "latn"))
lang = langsys.langs[0]
self.assertEqual((lang.name, lang.tag), (None, "dflt"))
def test_langsys_no_langsys_tag_fails(self):
with self.assertRaisesRegex(VoltLibError, r'.*Expected "TAG"'):
[langsys] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS NAME "Default"\n\n'
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def test_feature(self):
[def_script] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n\n'
'DEF_FEATURE NAME "Fractions" TAG "frac"\n'
' LOOKUP "fraclookup"\n'
"END_FEATURE\n"
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def_feature = def_script.langs[0].features[0]
self.assertEqual(
(def_feature.name, def_feature.tag, def_feature.lookups),
("Fractions", "frac", ["fraclookup"]),
)
[def_script] = self.parse(
'DEF_SCRIPT NAME "Latin" TAG "latn"\n\n'
'DEF_LANGSYS NAME "Romanian" TAG "ROM "\n\n'
'DEF_FEATURE NAME "Kerning" TAG "kern"\n'
' LOOKUP "kern1" LOOKUP "kern2"\n'
"END_FEATURE\n"
"END_LANGSYS\n"
"END_SCRIPT"
).statements
def_feature = def_script.langs[0].features[0]
self.assertEqual(
(def_feature.name, def_feature.tag, def_feature.lookups),
("Kerning", "kern", ["kern1", "kern2"]),
)
def test_lookup_duplicate(self):
with self.assertRaisesRegex(
VoltLibError,
'Lookup "dupe" already defined, ' "lookup names are case insensitive",
):
[lookup1, lookup2] = self.parse(
'DEF_LOOKUP "dupe"\n'
"AS_SUBSTITUTION\n"
'SUB GLYPH "a"\n'
'WITH GLYPH "a.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION\n"
'DEF_LOOKUP "dupe"\n'
"AS_SUBSTITUTION\n"
'SUB GLYPH "b"\n'
'WITH GLYPH "b.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION\n"
).statements
def test_lookup_duplicate_insensitive_case(self):
with self.assertRaisesRegex(
VoltLibError,
'Lookup "Dupe" already defined, ' "lookup names are case insensitive",
):
[lookup1, lookup2] = self.parse(
'DEF_LOOKUP "dupe"\n'
"AS_SUBSTITUTION\n"
'SUB GLYPH "a"\n'
'WITH GLYPH "a.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION\n"
'DEF_LOOKUP "Dupe"\n'
"AS_SUBSTITUTION\n"
'SUB GLYPH "b"\n'
'WITH GLYPH "b.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION\n"
).statements
def test_lookup_name_starts_with_letter(self):
with self.assertRaisesRegex(
VoltLibError, r'Lookup name "\\lookupname" must start with a letter'
):
[lookup] = self.parse(
'DEF_LOOKUP "\\lookupname"\n'
"AS_SUBSTITUTION\n"
'SUB GLYPH "a"\n'
'WITH GLYPH "a.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION\n"
).statements
def test_lookup_comments(self):
[lookup] = self.parse(
'DEF_LOOKUP "test" PROCESS_BASE PROCESS_MARKS ALL DIRECTION LTR\n'
'COMMENTS "Hello\\nWorld"\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "a"\n'
'WITH GLYPH "b"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual(lookup.name, "test")
self.assertEqual(lookup.comments, "Hello\nWorld")
def test_substitution_empty(self):
with self.assertRaisesRegex(VoltLibError, r"Expected SUB"):
[lookup] = self.parse(
'DEF_LOOKUP "empty_substitution" PROCESS_BASE PROCESS_MARKS '
"ALL DIRECTION LTR\n"
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
"END_SUBSTITUTION"
).statements
def test_substitution_invalid_many_to_many(self):
with self.assertRaisesRegex(VoltLibError, r"Invalid substitution type"):
[lookup] = self.parse(
'DEF_LOOKUP "invalid_substitution" 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.alt" GLYPH "i.alt"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
def test_substitution_invalid_reverse_chaining_single(self):
with self.assertRaisesRegex(VoltLibError, r"Invalid substitution type"):
[lookup] = self.parse(
'DEF_LOOKUP "invalid_substitution" PROCESS_BASE PROCESS_MARKS '
"ALL DIRECTION LTR REVERSAL\n"
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "f" GLYPH "i"\n'
'WITH GLYPH "f_i"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
def test_substitution_invalid_mixed(self):
with self.assertRaisesRegex(VoltLibError, r"Invalid substitution type"):
[lookup] = self.parse(
'DEF_LOOKUP "invalid_substitution" PROCESS_BASE PROCESS_MARKS '
"ALL DIRECTION LTR\n"
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "fi"\n'
'WITH GLYPH "f" GLYPH "i"\n'
"END_SUB\n"
'SUB GLYPH "f" GLYPH "l"\n'
'WITH GLYPH "f_l"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
def test_substitution_single(self):
[lookup] = 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"
"END_SUBSTITUTION"
).statements
self.assertEqual(lookup.name, "smcp")
self.assertSubEqual(lookup.sub, [["a"], ["b"]], [["a.sc"], ["b.sc"]])
def test_substitution_single_in_context(self):
[group, lookup] = self.parse(
'DEF_GROUP "Denominators"\n'
' ENUM GLYPH "one.dnom" GLYPH "two.dnom" END_ENUM\n'
"END_GROUP\n"
'DEF_LOOKUP "fracdnom" PROCESS_BASE PROCESS_MARKS ALL '
"DIRECTION LTR\n"
"IN_CONTEXT\n"
' 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"
).statements
context = lookup.context[0]
self.assertEqual(lookup.name, "fracdnom")
self.assertEqual(context.ex_or_in, "IN_CONTEXT")
self.assertEqual(len(context.left), 1)
self.assertEqual(len(context.left[0]), 1)
self.assertEqual(len(context.left[0][0].enum), 2)
self.assertEqual(context.left[0][0].enum[0].group, "Denominators")
self.assertEqual(context.left[0][0].enum[1].glyph, "fraction")
self.assertEqual(context.right, [])
self.assertSubEqual(
lookup.sub, [["one"], ["two"]], [["one.dnom"], ["two.dnom"]]
)
def test_substitution_single_in_contexts(self):
[group, lookup] = self.parse(
'DEF_GROUP "Hebrew"\n'
' ENUM GLYPH "uni05D0" GLYPH "uni05D1" END_ENUM\n'
"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"
).statements
context1 = lookup.context[0]
context2 = lookup.context[1]
self.assertEqual(lookup.name, "HebrewCurrency")
self.assertEqual(context1.ex_or_in, "IN_CONTEXT")
self.assertEqual(context1.left, [])
self.assertEqual(len(context1.right), 2)
self.assertEqual(len(context1.right[0]), 1)
self.assertEqual(len(context1.right[1]), 1)
self.assertEqual(context1.right[0][0].group, "Hebrew")
self.assertEqual(context1.right[1][0].glyph, "one.Hebr")
self.assertEqual(context2.ex_or_in, "IN_CONTEXT")
self.assertEqual(len(context2.left), 2)
self.assertEqual(len(context2.left[0]), 1)
self.assertEqual(len(context2.left[1]), 1)
self.assertEqual(context2.left[0][0].group, "Hebrew")
self.assertEqual(context2.left[1][0].glyph, "one.Hebr")
self.assertEqual(context2.right, [])
def test_substitution_skip_base(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "marka" GLYPH "markb" END_ENUM\n'
"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"
).statements
self.assertEqual((lookup.name, lookup.process_base), ("SomeSub", False))
def test_substitution_process_base(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "marka" GLYPH "markb" END_ENUM\n'
"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"
).statements
self.assertEqual((lookup.name, lookup.process_base), ("SomeSub", True))
def test_substitution_process_marks(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "marka" GLYPH "markb" END_ENUM\n'
"END_GROUP\n"
'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS "SomeMarks"\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", "SomeMarks"))
def test_substitution_process_marks_all(self):
[lookup] = self.parse(
'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS ALL\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", True))
def test_substitution_process_marks_none(self):
[lookup] = self.parse_(
'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS "NONE"\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", False))
def test_substitution_process_marks_bad(self):
with self.assertRaisesRegex(
VoltLibError, "Expected ALL, NONE, MARK_GLYPH_SET or an ID"
):
self.parse(
'DEF_GROUP "SomeMarks" ENUM GLYPH "marka" GLYPH "markb" '
"END_ENUM END_GROUP\n"
'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS SomeMarks '
"AS_SUBSTITUTION\n"
'SUB GLYPH "A" WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
)
def test_substitution_skip_marks(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "marka" GLYPH "markb" END_ENUM\n'
"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"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", False))
def test_substitution_mark_attachment(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "acutecmb" GLYPH "gravecmb" END_ENUM\n'
"END_GROUP\n"
'DEF_LOOKUP "SomeSub" PROCESS_BASE '
'PROCESS_MARKS "SomeMarks" DIRECTION RTL\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", "SomeMarks"))
def test_substitution_mark_glyph_set(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "acutecmb" GLYPH "gravecmb" END_ENUM\n'
"END_GROUP\n"
'DEF_LOOKUP "SomeSub" PROCESS_BASE '
'PROCESS_MARKS MARK_GLYPH_SET "SomeMarks" DIRECTION RTL\n'
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.mark_glyph_set), ("SomeSub", "SomeMarks"))
def test_substitution_process_all_marks(self):
[group, lookup] = self.parse(
'DEF_GROUP "SomeMarks"\n'
' ENUM GLYPH "acutecmb" GLYPH "gravecmb" END_ENUM\n'
"END_GROUP\n"
'DEF_LOOKUP "SomeSub" PROCESS_BASE PROCESS_MARKS ALL '
"DIRECTION RTL\n"
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_SUBSTITUTION\n"
'SUB GLYPH "A"\n'
'WITH GLYPH "A.c2sc"\n'
"END_SUB\n"
"END_SUBSTITUTION"
).statements
self.assertEqual((lookup.name, lookup.process_marks), ("SomeSub", True))
def test_substitution_no_reversal(self):
# TODO: check right context with no reversal
[lookup] = 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"
).statements
self.assertEqual((lookup.name, lookup.reversal), ("Lookup", None))
def test_substitution_reversal(self):
lookup = 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"
).statements[-1]
self.assertEqual((lookup.name, lookup.reversal), ("RevLookup", True))
def test_substitution_single_to_multiple(self):
[lookup] = 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"
).statements
self.assertEqual(lookup.name, "ccmp")
self.assertSubEqual(
lookup.sub,
[["aacute"], ["agrave"]],
[["a", "acutecomb"], ["a", "gravecomb"]],
)
def test_substitution_multiple_to_single(self):
[lookup] = 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"
).statements
self.assertEqual(lookup.name, "liga")
self.assertSubEqual(lookup.sub, [["f", "i"], ["f", "t"]], [["f_i"], ["f_t"]])
def test_substitution_reverse_chaining_single(self):
[lookup] = 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"
).statements
mapping = lookup.sub.mapping
glyphs = [[(r.start, r.end) for r in v] for v in mapping.keys()]
replacement = [[(r.start, r.end) for r in v] for v in mapping.values()]
self.assertEqual(lookup.name, "numr")
self.assertEqual(glyphs, [[("zero", "nine")]])
self.assertEqual(replacement, [[("zero.numr", "nine.numr")]])
self.assertEqual(len(lookup.context[0].right), 1)
self.assertEqual(len(lookup.context[0].right[0]), 1)
enum = lookup.context[0].right[0][0]
self.assertEqual(len(enum.enum), 2)
self.assertEqual(enum.enum[0].glyph, "fraction")
self.assertEqual(
(enum.enum[1].start, enum.enum[1].end), ("zero.numr", "nine.numr")
)
# GPOS
# ATTACH_CURSIVE
# ATTACH
# ADJUST_PAIR
# ADJUST_SINGLE
def test_position_empty(self):
with self.assertRaisesRegex(
VoltLibError, "Expected ATTACH, ATTACH_CURSIVE, ADJUST_PAIR, ADJUST_SINGLE"
):
[lookup] = self.parse(
'DEF_LOOKUP "empty_position" PROCESS_BASE PROCESS_MARKS ALL '
"DIRECTION LTR\n"
"EXCEPT_CONTEXT\n"
' LEFT GLYPH "glyph"\n'
"END_CONTEXT\n"
"AS_POSITION\n"
"END_POSITION"
).statements
def test_position_attach(self):
[lookup, anchor1, anchor2, anchor3, anchor4] = 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"
).statements
pos = lookup.pos
coverage = [g.glyph for g in pos.coverage]
coverage_to = [[[g.glyph for g in e], a] for (e, a) in pos.coverage_to]
self.assertEqual(
(lookup.name, coverage, coverage_to),
(
"anchor_top",
["a", "e"],
[[["acutecomb"], "top"], [["gravecomb"], "top"]],
),
)
self.assertEqual(
(
anchor1.name,
anchor1.gid,
anchor1.glyph_name,
anchor1.component,
anchor1.locked,
anchor1.pos,
),
("MARK_top", 120, "acutecomb", 1, False, (None, 0, 450, {}, {}, {})),
)
self.assertEqual(
(
anchor2.name,
anchor2.gid,
anchor2.glyph_name,
anchor2.component,
anchor2.locked,
anchor2.pos,
),
("MARK_top", 121, "gravecomb", 1, False, (None, 0, 450, {}, {}, {})),
)
self.assertEqual(
(
anchor3.name,
anchor3.gid,
anchor3.glyph_name,
anchor3.component,
anchor3.locked,
anchor3.pos,
),
("top", 31, "a", 1, False, (None, 210, 450, {}, {}, {})),
)
self.assertEqual(
(
anchor4.name,
anchor4.gid,
anchor4.glyph_name,
anchor4.component,
anchor4.locked,
anchor4.pos,
),
("top", 35, "e", 1, False, (None, 215, 450, {}, {}, {})),
)
def test_position_attach_cursive(self):
[lookup] = self.parse(
'DEF_LOOKUP "SomeLookup" PROCESS_BASE PROCESS_MARKS ALL '
"DIRECTION RTL\n"
"IN_CONTEXT\n"
"END_CONTEXT\n"
"AS_POSITION\n"
'ATTACH_CURSIVE\nEXIT GLYPH "a" GLYPH "b"\nENTER GLYPH "c"\n'
"END_ATTACH\n"
"END_POSITION"
).statements
exit = [[g.glyph for g in v] for v in lookup.pos.coverages_exit]
enter = [[g.glyph for g in v] for v in lookup.pos.coverages_enter]
self.assertEqual(
(lookup.name, exit, enter), ("SomeLookup", [["a", "b"]], [["c"]])
)
def test_position_adjust_pair(self):
[lookup] = 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"\n'
' SECOND GLYPH "V"\n'
" 1 2 BY POS ADV -30 END_POS POS END_POS\n"
" 2 1 BY POS ADV -30 END_POS POS END_POS\n\n"
"END_ADJUST\n"
"END_POSITION"
).statements
coverages_1 = [[g.glyph for g in v] for v in lookup.pos.coverages_1]
coverages_2 = [[g.glyph for g in v] for v in lookup.pos.coverages_2]
self.assertEqual(
(lookup.name, coverages_1, coverages_2, lookup.pos.adjust_pair),
(
"kern1",
[["A"]],
[["V"]],
{
(1, 2): (
(-30, None, None, {}, {}, {}),
(None, None, None, {}, {}, {}),
),
(2, 1): (
(-30, None, None, {}, {}, {}),
(None, None, None, {}, {}, {}),
),
},
),
)
def test_position_adjust_single(self):
[lookup] = self.parse(
'DEF_LOOKUP "TestLookup" PROCESS_BASE PROCESS_MARKS ALL '
"DIRECTION LTR\n"
"IN_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'
' GLYPH "glyph2" BY POS ADV 0 DX 456 END_POS\n'
"END_ADJUST\n"
"END_POSITION"
).statements
pos = lookup.pos
adjust = [[[g.glyph for g in a], b] for (a, b) in pos.adjust_single]
self.assertEqual(
(lookup.name, adjust),
(
"TestLookup",
[
[["glyph1"], (0, 123, None, {}, {}, {})],
[["glyph2"], (0, 456, None, {}, {}, {})],
],
),
)
def test_def_anchor(self):
[anchor1, anchor2, anchor3] = self.parse(
'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\n"
'DEF_ANCHOR "bottom" ON 120 GLYPH a '
"COMPONENT 1 AT POS DX 250 DY 0 END_POS END_ANCHOR"
).statements
self.assertEqual(
(
anchor1.name,
anchor1.gid,
anchor1.glyph_name,
anchor1.component,
anchor1.locked,
anchor1.pos,
),
("top", 120, "a", 1, False, (None, 250, 450, {}, {}, {})),
)
self.assertEqual(
(
anchor2.name,
anchor2.gid,
anchor2.glyph_name,
anchor2.component,
anchor2.locked,
anchor2.pos,
),
("MARK_top", 120, "acutecomb", 1, False, (None, 0, 450, {}, {}, {})),
)
self.assertEqual(
(
anchor3.name,
anchor3.gid,
anchor3.glyph_name,
anchor3.component,
anchor3.locked,
anchor3.pos,
),
("bottom", 120, "a", 1, False, (None, 250, 0, {}, {}, {})),
)
def test_def_anchor_multi_component(self):
[anchor1, anchor2] = self.parse(
'DEF_ANCHOR "top" ON 120 GLYPH a '
"COMPONENT 1 AT POS DX 250 DY 450 END_POS END_ANCHOR\n"
'DEF_ANCHOR "top" ON 120 GLYPH a '
"COMPONENT 2 AT POS DX 250 DY 450 END_POS END_ANCHOR"
).statements
self.assertEqual(
(anchor1.name, anchor1.gid, anchor1.glyph_name, anchor1.component),
("top", 120, "a", 1),
)
self.assertEqual(
(anchor2.name, anchor2.gid, anchor2.glyph_name, anchor2.component),
("top", 120, "a", 2),
)
def test_def_anchor_duplicate(self):
self.assertRaisesRegex(
VoltLibError,
'Anchor "dupe" already defined, ' "anchor names are case insensitive",
self.parse,
'DEF_ANCHOR "dupe" ON 120 GLYPH a '
"COMPONENT 1 AT POS DX 250 DY 450 END_POS END_ANCHOR\n"
'DEF_ANCHOR "dupe" ON 120 GLYPH a '
"COMPONENT 1 AT POS DX 250 DY 450 END_POS END_ANCHOR",
)
def test_def_anchor_locked(self):
[anchor] = self.parse(
'DEF_ANCHOR "top" ON 120 GLYPH a '
"COMPONENT 1 LOCKED AT POS DX 250 DY 450 END_POS END_ANCHOR"
).statements
self.assertEqual(
(
anchor.name,
anchor.gid,
anchor.glyph_name,
anchor.component,
anchor.locked,
anchor.pos,
),
("top", 120, "a", 1, True, (None, 250, 450, {}, {}, {})),
)
def test_anchor_adjust_device(self):
[anchor] = 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"
).statements
self.assertEqual(
(anchor.name, anchor.pos),
("MARK_top", (None, 0, 456, {}, {}, {34: 12, 78: 56})),
)
def test_ppem(self):
[grid_ppem, pres_ppem, ppos_ppem] = self.parse(
"GRID_PPEM 20\n" "PRESENTATION_PPEM 72\n" "PPOSITIONING_PPEM 144"
).statements
self.assertEqual(
(
(grid_ppem.name, grid_ppem.value),
(pres_ppem.name, pres_ppem.value),
(ppos_ppem.name, ppos_ppem.value),
),
(("GRID_PPEM", 20), ("PRESENTATION_PPEM", 72), ("PPOSITIONING_PPEM", 144)),
)
def test_compiler_flags(self):
[setting1, setting2] = self.parse(
"COMPILER_USEEXTENSIONLOOKUPS\n" "COMPILER_USEPAIRPOSFORMAT2"
).statements
self.assertEqual(
((setting1.name, setting1.value), (setting2.name, setting2.value)),
(
("COMPILER_USEEXTENSIONLOOKUPS", True),
("COMPILER_USEPAIRPOSFORMAT2", True),
),
)
def test_cmap(self):
[cmap_format1, cmap_format2, cmap_format3] = self.parse(
"CMAP_FORMAT 0 3 4\n" "CMAP_FORMAT 1 0 6\n" "CMAP_FORMAT 3 1 4"
).statements
self.assertEqual(
(
(cmap_format1.name, cmap_format1.value),
(cmap_format2.name, cmap_format2.value),
(cmap_format3.name, cmap_format3.value),
),
(
("CMAP_FORMAT", (0, 3, 4)),
("CMAP_FORMAT", (1, 0, 6)),
("CMAP_FORMAT", (3, 1, 4)),
),
)
def test_do_not_touch_cmap(self):
[option1, option2, option3, option4] = self.parse(
"DO_NOT_TOUCH_CMAP\n"
"CMAP_FORMAT 0 3 4\n"
"CMAP_FORMAT 1 0 6\n"
"CMAP_FORMAT 3 1 4"
).statements
self.assertEqual(
(
(option1.name, option1.value),
(option2.name, option2.value),
(option3.name, option3.value),
(option4.name, option4.value),
),
(
("DO_NOT_TOUCH_CMAP", True),
("CMAP_FORMAT", (0, 3, 4)),
("CMAP_FORMAT", (1, 0, 6)),
("CMAP_FORMAT", (3, 1, 4)),
),
)
def test_stop_at_end(self):
doc = self.parse_('DEF_GLYPH ".notdef" ID 0 TYPE BASE END_GLYPH END\0\0\0\0')
[def_glyph] = doc.statements
self.assertEqual(
(
def_glyph.name,
def_glyph.id,
def_glyph.unicode,
def_glyph.type,
def_glyph.components,
),
(".notdef", 0, None, "BASE", None),
)
self.assertEqual(
str(doc), '\nDEF_GLYPH ".notdef" ID 0 TYPE BASE END_GLYPH END\n'
)
def parse_(self, text):
return Parser(StringIO(text)).parse()
def parse(self, text):
doc = self.parse_(text)
self.assertEqual("\n".join(str(s) for s in doc.statements), text)
return Parser(StringIO(text)).parse()
if __name__ == "__main__":
import sys
sys.exit(unittest.main())