97 lines
3.1 KiB
Python
97 lines
3.1 KiB
Python
|
__all__ = ["maxCtxFont"]
|
||
|
|
||
|
|
||
|
def maxCtxFont(font):
|
||
|
"""Calculate the usMaxContext value for an entire font."""
|
||
|
|
||
|
maxCtx = 0
|
||
|
for tag in ("GSUB", "GPOS"):
|
||
|
if tag not in font:
|
||
|
continue
|
||
|
table = font[tag].table
|
||
|
if not table.LookupList:
|
||
|
continue
|
||
|
for lookup in table.LookupList.Lookup:
|
||
|
for st in lookup.SubTable:
|
||
|
maxCtx = maxCtxSubtable(maxCtx, tag, lookup.LookupType, st)
|
||
|
return maxCtx
|
||
|
|
||
|
|
||
|
def maxCtxSubtable(maxCtx, tag, lookupType, st):
|
||
|
"""Calculate usMaxContext based on a single lookup table (and an existing
|
||
|
max value).
|
||
|
"""
|
||
|
|
||
|
# single positioning, single / multiple substitution
|
||
|
if (tag == "GPOS" and lookupType == 1) or (
|
||
|
tag == "GSUB" and lookupType in (1, 2, 3)
|
||
|
):
|
||
|
maxCtx = max(maxCtx, 1)
|
||
|
|
||
|
# pair positioning
|
||
|
elif tag == "GPOS" and lookupType == 2:
|
||
|
maxCtx = max(maxCtx, 2)
|
||
|
|
||
|
# ligatures
|
||
|
elif tag == "GSUB" and lookupType == 4:
|
||
|
for ligatures in st.ligatures.values():
|
||
|
for ligature in ligatures:
|
||
|
maxCtx = max(maxCtx, ligature.CompCount)
|
||
|
|
||
|
# context
|
||
|
elif (tag == "GPOS" and lookupType == 7) or (tag == "GSUB" and lookupType == 5):
|
||
|
maxCtx = maxCtxContextualSubtable(maxCtx, st, "Pos" if tag == "GPOS" else "Sub")
|
||
|
|
||
|
# chained context
|
||
|
elif (tag == "GPOS" and lookupType == 8) or (tag == "GSUB" and lookupType == 6):
|
||
|
maxCtx = maxCtxContextualSubtable(
|
||
|
maxCtx, st, "Pos" if tag == "GPOS" else "Sub", "Chain"
|
||
|
)
|
||
|
|
||
|
# extensions
|
||
|
elif (tag == "GPOS" and lookupType == 9) or (tag == "GSUB" and lookupType == 7):
|
||
|
maxCtx = maxCtxSubtable(maxCtx, tag, st.ExtensionLookupType, st.ExtSubTable)
|
||
|
|
||
|
# reverse-chained context
|
||
|
elif tag == "GSUB" and lookupType == 8:
|
||
|
maxCtx = maxCtxContextualRule(maxCtx, st, "Reverse")
|
||
|
|
||
|
return maxCtx
|
||
|
|
||
|
|
||
|
def maxCtxContextualSubtable(maxCtx, st, ruleType, chain=""):
|
||
|
"""Calculate usMaxContext based on a contextual feature subtable."""
|
||
|
|
||
|
if st.Format == 1:
|
||
|
for ruleset in getattr(st, "%s%sRuleSet" % (chain, ruleType)):
|
||
|
if ruleset is None:
|
||
|
continue
|
||
|
for rule in getattr(ruleset, "%s%sRule" % (chain, ruleType)):
|
||
|
if rule is None:
|
||
|
continue
|
||
|
maxCtx = maxCtxContextualRule(maxCtx, rule, chain)
|
||
|
|
||
|
elif st.Format == 2:
|
||
|
for ruleset in getattr(st, "%s%sClassSet" % (chain, ruleType)):
|
||
|
if ruleset is None:
|
||
|
continue
|
||
|
for rule in getattr(ruleset, "%s%sClassRule" % (chain, ruleType)):
|
||
|
if rule is None:
|
||
|
continue
|
||
|
maxCtx = maxCtxContextualRule(maxCtx, rule, chain)
|
||
|
|
||
|
elif st.Format == 3:
|
||
|
maxCtx = maxCtxContextualRule(maxCtx, st, chain)
|
||
|
|
||
|
return maxCtx
|
||
|
|
||
|
|
||
|
def maxCtxContextualRule(maxCtx, st, chain):
|
||
|
"""Calculate usMaxContext based on a contextual feature rule."""
|
||
|
|
||
|
if not chain:
|
||
|
return max(maxCtx, st.GlyphCount)
|
||
|
elif chain == "Reverse":
|
||
|
return max(maxCtx, st.GlyphCount + st.LookAheadGlyphCount)
|
||
|
return max(maxCtx, st.InputGlyphCount + st.LookAheadGlyphCount)
|