summaryrefslogtreecommitdiffstats
path: root/src/silfont/ufo.py
diff options
context:
space:
mode:
Diffstat (limited to 'src/silfont/ufo.py')
-rw-r--r--src/silfont/ufo.py1386
1 files changed, 1386 insertions, 0 deletions
diff --git a/src/silfont/ufo.py b/src/silfont/ufo.py
new file mode 100644
index 0000000..d115ec1
--- /dev/null
+++ b/src/silfont/ufo.py
@@ -0,0 +1,1386 @@
+#!/usr/bin/env python3
+'Classes and functions for use handling Ufont UFO font objects in pysilfont scripts'
+__url__ = 'https://github.com/silnrsi/pysilfont'
+__copyright__ = 'Copyright (c) 2015 SIL International (https://www.sil.org)'
+__license__ = 'Released under the MIT License (https://opensource.org/licenses/MIT)'
+__author__ = 'David Raymond'
+
+from xml.etree import ElementTree as ET
+import sys, os, shutil, filecmp, io, re
+import warnings
+import collections
+import datetime
+import silfont.core
+import silfont.util as UT
+import silfont.etutil as ETU
+
+_glifElemMulti = ('unicode', 'guideline', 'anchor') # glif elements that can occur multiple times
+_glifElemF1 = ('advance', 'unicode', 'outline', 'lib') # glif elements valid in format 1 glifs (ie UFO2 glfis)
+
+# Define illegal characters and reserved names for makeFileName
+_illegalChars = "\"*+/:><?[\]|" + chr(0x7F)
+for i in range(0, 32): _illegalChars += chr(i)
+_illegalChars = list(_illegalChars)
+_reservedNames = "CON PRN AUX CLOCK$ NUL COM1 COM2 COM3 COM4 PT1 LPT2 LPT3".lower().split(" ")
+
+obsoleteLibKeys = [ # Used by "check and fix" + some scripts
+ "com.schriftgestaltung.blueFuzz",
+ "com.schriftgestaltung.blueScale",
+ "com.schriftgestaltung.blueShift",
+ "com.schriftgestaltung.customValue",
+ "com.schriftgestaltung.Disable Last Change",
+ "com.schriftgestaltung.disablesAutomaticAlignment",
+ "com.schriftgestaltung.disablesLastChange",
+ "com.schriftgestaltung.DisplayStrings",
+ "com.schriftgestaltung.font.Disable Last Change",
+ "com.schriftgestaltung.font.glyphOrder",
+ "com.schriftgestaltung.font.license",
+ "com.schriftgestaltung.useNiceNames",
+ "org.sil.glyphsappversion",
+ "UFOFormat"]
+
+class _Ucontainer(object):
+ # Parent class for other objects (eg Ulayer)
+ def __init__(self):
+ self._contents = {}
+
+ # Define methods so it acts like an immutable container
+ # (changes should be made via object functions etc)
+ def __len__(self):
+ return len(self._contents)
+
+ def __getitem__(self, key):
+ return self._contents[key]
+
+ def __iter__(self):
+ return iter(self._contents)
+
+ def get(self, key, default=None):
+ return self._contents.get(key, default=default)
+
+ def keys(self):
+ return self._contents.keys()
+
+
+class _plist(object):
+ # Used for common plist methods inherited by Uplist and Ulib classes
+
+ def addval(self, key, valuetype, value): # For simple single-value elements - use addelem for dicts or arrays
+ if valuetype not in ("integer", "real", "string"):
+ self.font.logger.log("addval() can only be used with simple elements", "X")
+ if key in self._contents: self.font.logger.log("Attempt to add duplicate key " + key + " to plist", "X")
+ dict = self.etree[0]
+
+ keyelem = ET.Element("key")
+ keyelem.text = key
+ dict.append(keyelem)
+
+ valelem = ET.Element(valuetype)
+ valelem.text = str(value)
+ dict.append(valelem)
+
+ self._contents[key] = [keyelem, valelem]
+
+ def setval(self, key, valuetype, value): # For simple single-value elements - use setelem for dicts or arrays
+ if valuetype not in ("integer", "real", "string"):
+ self.font.logger.log("setval() can only be used with simple elements", "X")
+ if key in self._contents:
+ self._contents[key][1].text = str(value)
+ else:
+ self.addval(key, valuetype, value)
+
+ def getval(self, key, default=None): # Returns a value for integer, real, string, true, false, dict or array keys or None for other keys
+ elem = self._contents.get(key, [None, None])[1]
+ if elem is None:
+ return default
+ return self._valelem(elem)
+
+ def _valelem(self, elem): # Used by getval to recursively process dict and array elements
+ if elem.tag == "integer": return int(elem.text)
+ elif elem.tag == "real": return float(elem.text)
+ elif elem.tag == "string": return elem.text
+ elif elem.tag == "true": return True
+ elif elem.tag == "false": return False
+ elif elem.tag == "array":
+ array = []
+ for subelem in elem: array.append(self._valelem(subelem))
+ return array
+ elif elem.tag == "dict":
+ dict = {}
+ for i in range(0, len(elem), 2): dict[elem[i].text] = self._valelem(elem[i + 1])
+ return dict
+ else:
+ return None
+
+ def remove(self, key):
+ item = self._contents[key]
+ self.etree[0].remove(item[0])
+ self.etree[0].remove(item[1])
+ del self._contents[key]
+
+ def addelem(self, key, element): # For non-simple elements (eg arrays) the calling script needs to build the etree element
+ if key in self._contents: self.font.logger.log("Attempt to add duplicate key " + key + " to plist", "X")
+ dict = self.etree[0]
+
+ keyelem = ET.Element("key")
+ keyelem.text = key
+ dict.append(keyelem)
+ dict.append(element)
+
+ self._contents[key] = [keyelem, element]
+
+ def setelem(self, key, element):
+ if key in self._contents: self.remove(key)
+ self.addelem(key, element)
+
+
+class Uelement(_Ucontainer):
+ # Class for an etree element. Mainly used as a parent class
+ # For each tag in the element, returns list of sub-elements with that tag
+ def __init__(self, element):
+ self.element = element
+ self.reindex()
+
+ def reindex(self):
+ self._contents = collections.defaultdict(list)
+ for e in self.element:
+ self._contents[e.tag].append(e)
+
+ def remove(self, subelement):
+ self._contents[subelement.tag].remove(subelement)
+ self.element.remove(subelement)
+
+ def append(self, subelement):
+ self._contents[subelement.tag].append(subelement)
+ self.element.append(subelement)
+
+ def insert(self, index, subelement):
+ self._contents[subelement.tag].insert(index, subelement)
+ self.element.insert(index, subelement)
+
+ def replace(self, index, subelement):
+ oldsubelement = self.element[index]
+ cindex = self._contents[subelement.tag].index(oldsubelement)
+ self._contents[subelement.tag][cindex] = subelement
+ self.element[index] = subelement
+
+
+class UtextFile(object):
+ # Generic object for handling non-xml text files
+ def __init__(self, font, dirn, filen):
+ self.type = "textfile"
+ self.font = font
+ self.filen = filen
+ self.dirn = dirn
+ if dirn == font.ufodir:
+ dtree = font.dtree
+ else:
+ dtree = font.dtree.subtree(dirn)
+ if not dtree: font.logger.log("Missing directory " + dirn, "X")
+ if filen not in dtree:
+ dtree[filen] = UT.dirTreeItem(added=True)
+ dtree[filen].setinfo(read=True)
+ dtree[filen].fileObject = self
+ dtree[filen].fileType = "text"
+
+ def write(self, dtreeitem, dir, ofilen, exists):
+ # For now just copies source to destination if changed
+ inpath = os.path.join(self.dirn, self.filen)
+ changed = True
+ if exists: changed = not (filecmp.cmp(inpath, os.path.join(dir, self.filen)))
+ if changed:
+ try:
+ shutil.copy2(inpath, dir)
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ dtreeitem.written = True
+
+class Udirectory(object):
+ # Generic object for handling directories - used for data and images
+ def __init__(self, font, parentdir, dirn):
+ self.type = "directory"
+ self.font = font
+ self.parentdir = parentdir
+ self.dirn = dirn
+ if parentdir != font.ufodir:
+ self.font.logger.log("Currently Udir only supports top-level directories", "X")
+ dtree = font.dtree
+ if dirn not in dtree:
+ self.font.logger.log("Udir directory " + dirn + " does not exist", "X")
+ dtree[dirn].setinfo(read=True)
+ dtree[dirn].fileObject = self
+ dtree[dirn].fileType = "directory"
+
+ def write(self, dtreeitem, oparentdir):
+ # For now just copies source to destination
+ if self.parentdir == oparentdir: return # No action needed
+ inpath = os.path.join(self.parentdir, self.dirn)
+ outpath = os.path.join(oparentdir, self.dirn)
+ try:
+ if os.path.isdir(outpath):
+ shutil.rmtree(outpath)
+ shutil.copytree(inpath, outpath)
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ dtreeitem.written = True
+
+class Ufont(object):
+ """ Object to hold all the data from a UFO"""
+
+ def __init__(self, ufodir, logger=None, params=None):
+ if logger is not None and params is not None:
+ params.logger.log("Only supply a logger if params not set (since that has one)", "X")
+ if params is None:
+ params = silfont.core.parameters()
+ if logger is not None: params.logger = logger
+ self.params = params
+ self.logger = params.logger
+ logger = self.logger
+ self.ufodir = ufodir
+ logger.log('Reading UFO: ' + ufodir, 'P')
+ if not os.path.isdir(ufodir):
+ logger.log(ufodir + " is not a directory", "S")
+ # Read list of files and folders
+ self.dtree = UT.dirTree(ufodir)
+ # Read metainfo (which must exist)
+ self.metainfo = self._readPlist("metainfo.plist")
+ self.UFOversion = self.metainfo["formatVersion"][1].text
+ # Read lib.plist then process pysilfont parameters if present
+ libparams = {}
+ if "lib.plist" in self.dtree:
+ self.lib = self._readPlist("lib.plist")
+ if "org.sil.pysilfontparams" in self.lib:
+ elem = self.lib["org.sil.pysilfontparams"][1]
+ if elem.tag != "array":
+ logger.log("Invalid parameter XML lib.plist - org.sil.pysilfontparams must be an array", "S")
+ for param in elem:
+ parn = param.tag
+ if not (parn in params.paramclass) or params.paramclass[parn] not in ("outparams", "ufometadata"):
+ logger.log("lib.plist org.sil.pysilfontparams must only contain outparams or ufometadata values: " + parn + " invalid", "S")
+ libparams[parn] = param.text
+ # Create font-specific parameter set (with updates from lib.plist) Prepend names with ufodir to ensure uniqueness if multiple fonts open
+ params.addset(ufodir + "lib", "lib.plist in " + ufodir, inputdict=libparams)
+ if "command line" in params.sets:
+ params.sets[ufodir + "lib"].updatewith("command line", log=False) # Command line parameters override lib.plist ones
+ copyset = "main" if "main" in params.sets else "default"
+ params.addset(ufodir, copyset=copyset)
+ params.sets[ufodir].updatewith(ufodir + "lib", sourcedesc="lib.plist")
+ self.paramset = params.sets[ufodir]
+ # Validate specific parameters
+ if self.paramset["UFOversion"] not in ("", "2", "3"): logger.log("UFO version must be 2 or 3", "S")
+ if sorted(self.paramset["glifElemOrder"]) != sorted(self.params.sets["default"]["glifElemOrder"]):
+ logger.log("Invalid values for glifElemOrder", "S")
+
+ # Create outparams based on values in paramset, building attriborders from separate attriborders.<type> parameters.
+ self.outparams = {"attribOrders": {}}
+ for parn in params.classes["outparams"]:
+ value = self.paramset[parn]
+ if parn[0:12] == 'attribOrders':
+ elemname = parn.split(".")[1]
+ self.outparams["attribOrders"][elemname] = ETU.makeAttribOrder(value)
+ else:
+ self.outparams[parn] = value
+ if self.outparams["UFOversion"] == "": self.outparams["UFOversion"] = self.UFOversion
+
+ # Set flags for checking and fixing metadata
+ cf = self.paramset["checkfix"].lower()
+ if cf not in ("check", "fix", "none", ""): logger.log("Invalid value '" + cf + "' for checkfix parameter", "S")
+
+ self.metacheck = True if cf in ("check", "fix") else False
+ self.metafix = True if cf == "fix" else False
+ if "fontinfo.plist" not in self.dtree:
+ logger.log("fontinfo.plist missing so checkfix routines can't be run", "E")
+ self.metacheck = False
+ self.metafix = False
+
+ # Read other top-level plists
+ if "fontinfo.plist" in self.dtree: self.fontinfo = self._readPlist("fontinfo.plist")
+ if "groups.plist" in self.dtree: self.groups = self._readPlist("groups.plist")
+ if "kerning.plist" in self.dtree: self.kerning = self._readPlist("kerning.plist")
+ createlayercontents = False
+ if self.UFOversion == "2": # Create a dummy layer contents so 2 & 3 can be handled the same
+ createlayercontents = True
+ else:
+ if "layercontents.plist" in self.dtree:
+ self.layercontents = self._readPlist("layercontents.plist")
+ else:
+ logger.log("layercontents.plist missing - one will be created", "W")
+ createlayercontents = True
+ if createlayercontents:
+ if "glyphs" not in self.dtree: logger.log('No glyphs directory in font', "S")
+ self.layercontents = Uplist(font=self)
+ self.dtree['layercontents.plist'] = UT.dirTreeItem(read=True, added=True, fileObject=self.layercontents,
+ fileType="xml")
+ dummylc = "<plist>\n<array>\n<array>\n<string>public.default</string>\n<string>glyphs</string>\n</array>\n</array>\n</plist>"
+ self.layercontents.etree = ET.fromstring(dummylc)
+ self.layercontents.populate_dict()
+
+ # Process features.fea
+ if "features.fea" in self.dtree:
+ self.features = UfeatureFile(self, ufodir, "features.fea")
+ # Process the glyphs directories)
+ self.layers = []
+ self.deflayer = None
+ for i in sorted(self.layercontents.keys()):
+ layername = self.layercontents[i][0].text
+ layerdir = self.layercontents[i][1].text
+ logger.log("Processing Glyph Layer " + str(i) + ": " + layername + layerdir, "I")
+ layer = Ulayer(layername, layerdir, self)
+ if layer:
+ self.layers.append(layer)
+ if layername == "public.default": self.deflayer = layer
+ else:
+ logger.log("Glyph directory " + layerdir + " missing", "S")
+ if self.deflayer is None: logger.log("No public.default layer", "S")
+ # Process other directories
+ if "images" in self.dtree:
+ self.images = Udirectory(self,ufodir, "images")
+ if "data" in self.dtree:
+ self.data = Udirectory(self, ufodir, "data")
+
+ # Run best practices check and fix routines
+ if self.metacheck:
+ initwarnings = logger.warningcount
+ initerrors = logger.errorcount
+
+ fireq = ("ascender", "copyright", "descender", "familyName", "openTypeNameManufacturer",
+ "styleName", "unitsPerEm", "versionMajor", "versionMinor")
+ fiwarnifmiss = ("capHeight", "copyright", "openTypeNameDescription", "openTypeNameDesigner",
+ "openTypeNameDesignerURL", "openTypeNameLicense", "openTypeNameLicenseURL",
+ "openTypeNameManufacturerURL", "openTypeOS2CodePageRanges",
+ "openTypeOS2UnicodeRanges", "openTypeOS2VendorID",
+ "openTypeOS2WeightClass", "openTypeOS2WinAscent", "openTypeOS2WinDescent")
+ fiwarnifnot = {"unitsPerEm": (1000, 2048),
+ "styleMapStyleName": ("regular", "bold", "italic", "bold italic")},
+ fiwarnifpresent = ("note",)
+ fidel = ("macintoshFONDFamilyID", "macintoshFONDName", "openTypeNameCompatibleFullName",
+ "openTypeGaspRangeRecords", "openTypeHheaCaretOffset",
+ "openTypeOS2FamilyClass", "postscriptForceBold", "postscriptIsFixedPitch",
+ "postscriptBlueFuzz", "postscriptBlueScale", "postscriptBlueShift", "postscriptWeightName",
+ "year")
+ fidelifempty = ("guidelines", "postscriptBlueValues", "postscriptFamilyBlues", "postscriptFamilyOtherBlues",
+ "postscriptOtherBlues")
+ fiint = ("ascender", "capHeight", "descender", "postscriptUnderlinePosition",
+ "postscriptUnderlineThickness", "unitsPerEm", "xHeight")
+ ficapitalize = ("styleMapFamilyName", "styleName")
+ fisetifmissing = {}
+ fisettoother = {"openTypeHheaAscender": "ascender", "openTypeHheaDescender": "descender",
+ "openTypeNamePreferredFamilyName": "familyName",
+ "openTypeNamePreferredSubfamilyName": "styleName", "openTypeOS2TypoAscender": "ascender",
+ "openTypeOS2TypoDescender": "descender"}
+ fisetto = {"openTypeHheaLineGap": 0, "openTypeOS2TypoLineGap": 0, "openTypeOS2WidthClass": 5,
+ "openTypeOS2Selection": [7], "openTypeOS2Type": []} # Other values are added below
+
+ libdel = ("com.fontlab.v2.tth", "com.typemytype.robofont.italicSlantOffset")
+ libsetto = {"com.schriftgestaltung.customParameter.GSFont.disablesAutomaticAlignment": True,
+ "com.schriftgestaltung.customParameter.GSFont.disablesLastChange": True}
+ libwarnifnot = {"com.schriftgestaltung.customParameter.GSFont.useNiceNames": False}
+ libwarnifmissing = ("public.glyphOrder",)
+
+ # fontinfo.plist checks
+ logger.log("Checking fontinfo.plist metadata", "P")
+
+ # Check required fields, some of which are needed for remaining checks
+ missing = []
+ for key in fireq:
+ if key not in self.fontinfo or self.fontinfo.getval(key) is None: missing.append(key)
+ # Collect values for constructing other fields, setting dummy values when missing and in check-only mode
+ dummies = False
+ storedvals = {}
+ for key in ("ascender", "copyright", "descender", "familyName", "styleName", "openTypeNameManufacturer", "versionMajor", "versionMinor"):
+ if key in self.fontinfo and self.fontinfo.getval(key) is not None:
+ storedvals[key] = self.fontinfo.getval(key)
+ if key == "styleName":
+ sn = storedvals[key]
+ sn = re.sub(r"(\w)(Italic)", r"\1 \2", sn) # Add a space before Italic if missing
+ # Capitalise first letter of words
+ sep = b' ' if type(sn) is bytes else ' '
+ sn = sep.join(s[:1].upper() + s[1:] for s in sn.split(sep))
+ if sn != storedvals[key]:
+ if self.metafix:
+ self.fontinfo.setval(key, "string", sn)
+ logmess = " updated "
+ else:
+ logmess = " would be updated "
+ self.logchange(logmess, key, storedvals[key], sn)
+ storedvals[key] = sn
+ if key in ("ascender", "descender"):
+ storedvals[key] = int(storedvals[key])
+ else:
+ dummies = True
+ if key in ("ascender", "descender", "versionMajor", "versionMinor"):
+ storedvals[key] = 999
+ else:
+ storedvals[key] = "Dummy"
+ if missing:
+ logtype = "S" if self.metafix else "W"
+ logger.log("Required fields missing from fontinfo.plist: " + str(missing), logtype)
+ if dummies:
+ logger.log("Checking will continue with values of 'Dummy' or 999 for missing fields", "W")
+ # Construct values for certain fields
+ value = storedvals["openTypeNameManufacturer"] + ": " + storedvals["familyName"] + " "
+ value = value + storedvals["styleName"] + ": " + datetime.datetime.now().strftime("%Y")
+ fisetto["openTypeNameUniqueID"] = value
+# fisetto["openTypeOS2WinDescent"] = -storedvals["descender"]
+ if "openTypeNameVersion" not in self.fontinfo:
+ fisetto["openTypeNameVersion"] = "Version " + str(storedvals["versionMajor"]) + "."\
+ + str(storedvals["versionMinor"])
+ if "openTypeOS2WeightClass" not in self.fontinfo:
+ sn = storedvals["styleName"]
+ sn2wc = {"Regular": 400, "Italic": 400, "Bold": 700, "BoldItalic": 700}
+ if sn in sn2wc: fisetto["openTypeOS2WeightClass"] = sn2wc[sn]
+ if "xHeight" not in self.fontinfo:
+ fisetto["xHeight"] = int(storedvals["ascender"] * 0.6)
+ if "openTypeOS2Selection" in self.fontinfo: # If already present, need to ensure bit 7 is set
+ fisetto["openTypeOS2Selection"] = sorted(list(set(self.fontinfo.getval("openTypeOS2Selection") + [7])))
+
+ for key in fisetifmissing:
+ if key not in self.fontinfo:
+ fisetto[key] = fisetifmissing[key]
+
+ changes = 0
+ # Warn about missing fields
+ for key in fiwarnifmiss:
+ if key not in self.fontinfo:
+ logmess = key + " is missing from fontinfo.plist"
+ logger.log(logmess, "W")
+ # Warn about bad values
+ for key in fiwarnifnot:
+ if key in self.fontinfo:
+ value = self.fontinfo.getval(key)
+ if value not in fiwarnifnot[key]:
+ logger.log(key + " should be one of " + str(fiwarnifnot[key]), "W")
+ # Warn about keys where use of discouraged
+ for key in fiwarnifpresent:
+ if key in self.fontinfo:
+ logger.log(key + " is present - it's use is discouraged")
+
+ # Now do all remaining checks - which will lead to values being changed
+ for key in fidel + fidelifempty:
+ if key in self.fontinfo:
+ old = self.fontinfo.getval(key)
+ if not(key in fidelifempty and old != []): # Delete except for non-empty fidelifempty
+ if self.metafix:
+ self.fontinfo.remove(key)
+ logmess = " removed from fontinfo. "
+ else:
+ logmess = " would be removed from fontinfo "
+ self.logchange(logmess, key, old, None)
+ changes += 1
+
+ # Set to integer values
+ for key in fiint:
+ if key in self.fontinfo:
+ old = self.fontinfo.getval(key)
+ if old != int(old):
+ new = int(old)
+ if self.metafix:
+ self.fontinfo.setval(key, "integer", new)
+ logmess = " updated "
+ else:
+ logmess = " would be updated "
+ self.logchange(logmess, key, old, new)
+ changes += 1
+ # Capitalize words
+ for key in ficapitalize:
+ if key in self.fontinfo:
+ old = self.fontinfo.getval(key)
+ sep = b' ' if type(old) is bytes else ' '
+ new = sep.join(s[:1].upper() + s[1:] for s in old.split(sep)) # Capitalise words
+ if new != old:
+ if self.metafix:
+ self.fontinfo.setval(key, "string", new)
+ logmess = " uppdated "
+ else:
+ logmess = " would be uppdated "
+ self.logchange(logmess, key, old, new)
+ changes += 1
+ # Set to specific values
+ for key in list(fisetto.keys()) + list(fisettoother.keys()):
+ if key in self.fontinfo:
+ old = self.fontinfo.getval(key)
+ logmess = " updated "
+ else:
+ old = None
+ logmess = " added "
+ if key in fisetto:
+ new = fisetto[key]
+ else:
+ new = storedvals[fisettoother[key]]
+ if new != old:
+ if self.metafix:
+ if isinstance(new, list): # Currently only integer arrays
+ array = ET.Element("array")
+ for val in new: # Only covers integer at present for openTypeOS2Selection
+ ET.SubElement(array, "integer").text = val
+ self.fontinfo.setelem(key, array)
+ else: # Does not cover real at present
+ valtype = "integer" if isinstance(new, int) else "string"
+ self.fontinfo.setval(key, valtype, new)
+ else:
+ logmess = " would be" + logmess
+ self.logchange(logmess, key, old, new)
+ changes += 1
+ # Specific checks
+ if "italicAngle" in self.fontinfo:
+ old = self.fontinfo.getval("italicAngle")
+ if old == 0: # Should be deleted if 0
+ logmess = " removed since it is 0 "
+ if self.metafix:
+ self.fontinfo.remove("italicAngle")
+ else:
+ logmess = " would be" + logmess
+ self.logchange(logmess, "italicAngle", old, None)
+ changes += 1
+ if "versionMajor" in self.fontinfo: # If missing, an error will already have been reported...
+ vm = self.fontinfo.getval("versionMajor")
+ if vm == 0: logger.log("versionMajor is 0", "W")
+
+ # lib.plist checks
+ if "lib" not in self.__dict__:
+ logger.log("lib.plist missing so not checked by check & fix routines", "E")
+ else:
+ logger.log("Checking lib.plist metadata", "P")
+
+ for key in libdel:
+ if key in self.lib:
+ old = self.lib.getval(key)
+ if self.metafix:
+ self.lib.remove(key)
+ logmess = " removed from lib.plist. "
+ else:
+ logmess = " would be removed from lib.plist "
+ self.logchange(logmess, key, old, None)
+ changes += 1
+
+ for key in libsetto:
+ if key in self.lib:
+ old = self.lib.getval(key)
+ logmess = " updated "
+ else:
+ old = None
+ logmess = " added "
+ new = libsetto[key]
+ if new != old:
+ if self.metafix:
+ # Currently just supports True. See fisetto for adding other types
+ if new == True:
+ self.lib.setelem(key, ET.fromstring("<true/>"))
+ else: # Does not cover real at present
+ logger.log("Invalid value type for libsetto", "X")
+ else:
+ logmess = " would be" + logmess
+ self.logchange(logmess, key, old, new)
+ changes += 1
+ for key in libwarnifnot:
+ value = self.lib.getval(key) if key in self.lib else None
+ if value != libwarnifnot[key]:
+ addmess = "; currently missing" if value is None else "; currently set to " + str(value)
+ logger.log(key + " should normally be " + str(libwarnifnot[key]) + addmess, "W")
+
+ for key in libwarnifmissing:
+ if key not in self.lib:
+ logger.log(key + " is missing from lib.plist", "W")
+
+ logmess = " deleted - obsolete key" if self.metafix else " would be deleted - obsolete key"
+ for key in obsoleteLibKeys: # For obsolete keys that have been added historically by some tools
+ if key in self.lib:
+ old = self.lib.getval(key)
+ if self.metafix: self.lib.remove(key)
+ self.logchange(logmess,key,old,None)
+ changes += 1
+
+ # Show check&fix summary
+ warnings = logger.warningcount - initwarnings - changes
+ errors = logger.errorcount - initerrors
+ if errors or warnings or changes:
+ changemess = ", Changes made: " if self.metafix else ", Changes to make: "
+ logger.log("Check & fix results:- Errors: " + str(errors) + changemess + str(changes) +
+ ", Other warnings: " + str(warnings), "P")
+ if logger.scrlevel not in "WIV": logger.log("See log file for details", "P")
+ if missing and not self.metafix:
+ logger.log("**** Since some required fields were missing, checkfix=fix would fail", "P")
+ else:
+ logger.log("Check & Fix ran cleanly", "P")
+
+ def _readPlist(self, filen):
+ if filen in self.dtree:
+ plist = Uplist(font=self, filen=filen)
+ self.dtree[filen].setinfo(read=True, fileObject=plist, fileType="xml")
+ return plist
+ else:
+ self.logger.log(filen + " does not exist", "S")
+
+ def write(self, outdir):
+ # Write UFO out to disk, based on values set in self.outparams
+ self.logger.log("Processing font for output", "P")
+ if not os.path.exists(outdir):
+ try:
+ os.mkdir(outdir)
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ if not os.path.isdir(outdir):
+ self.logger.log(outdir + " not a directory", "S")
+
+ # If output UFO already exists, need to open so only changed files are updated and redundant files deleted
+ if outdir == self.ufodir: # In special case of output and input being the same, simply copy the input font
+ odtree = UT.dirTree(outdir)
+ else:
+ if not os.path.exists(outdir): # If outdir does not exist, create it
+ try:
+ os.mkdir(outdir)
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ odtree = {}
+ else:
+ if not os.path.isdir(outdir): self.logger.log(outdir + " not a directory", "S")
+ dirlist = os.listdir(outdir)
+ if dirlist == []: # Outdir is empty
+ odtree = {}
+ else:
+ self.logger.log("Output UFO already exists - reading for comparison", "P")
+ odtree = UT.dirTree(outdir)
+ # Update version info etc
+ UFOversion = self.outparams["UFOversion"]
+ self.metainfo["formatVersion"][1].text = str(UFOversion)
+ self.metainfo["creator"][1].text = "org.sil.scripts.pysilfont"
+
+ # Set standard UFO files for output
+ dtree = self.dtree
+ setFileForOutput(dtree, "metainfo.plist", self.metainfo, "xml")
+ if "fontinfo" in self.__dict__: setFileForOutput(dtree, "fontinfo.plist", self.fontinfo, "xml")
+ if "groups" in self.__dict__: # With groups, sort by glyph name
+ for gname in list(self.groups):
+ group = self.groups.getval(gname)
+ elem = ET.Element("array")
+ for glyph in sorted(group):
+ ET.SubElement(elem, "string").text = glyph
+ self.groups.setelem(gname, elem)
+ setFileForOutput(dtree, "groups.plist", self.groups, "xml")
+ if "kerning" in self.__dict__: setFileForOutput(dtree, "kerning.plist", self.kerning, "xml")
+ if "lib" in self.__dict__: setFileForOutput(dtree, "lib.plist", self.lib, "xml")
+ if UFOversion == "3":
+ # Sort layer contents by layer name
+ lc = self.layercontents
+ lcindex = {lc[x][0].text: lc[x] for x in lc} # index on layer name
+ for (x, name) in enumerate(sorted(lcindex)):
+ lc.etree[0][x] = lcindex[name] # Replace array elements in new order
+ setFileForOutput(dtree, "layercontents.plist", self.layercontents, "xml")
+ if "features" in self.__dict__: setFileForOutput(dtree, "features.fea", self.features, "text")
+ # Set glyph layers for output
+ for layer in self.layers: layer.setForOutput()
+
+ # Write files to disk
+
+ self.logger.log("Writing font to " + outdir, "P")
+
+ changes = writeToDisk(dtree, outdir, self, odtree)
+ if changes: # Need to update openTypeHeadCreated if there have been any changes to the font
+ if "fontinfo" in self.__dict__:
+ self.fontinfo.setval("openTypeHeadCreated", "string",
+ datetime.datetime.now().strftime("%Y/%m/%d %H:%M:%S"))
+ self.fontinfo.outxmlstr="" # Need to reset since writeXMLobject has already run once
+ writeXMLobject(self.fontinfo, self.outparams, outdir, "fontinfo.plist", True, fobject=True)
+
+ def addfile(self, filetype): # Add empty plist file for optional files
+ if filetype not in ("fontinfo", "groups", "kerning", "lib"): self.logger.log("Invalid file type to add", "X")
+ if filetype in self.__dict__: self.logger.log("File already in font", "X")
+ obj = Uplist(font=self)
+ setattr(self, filetype, obj)
+ self.dtree[filetype + '.plist'] = UT.dirTreeItem(read=True, added=True, fileObject=obj, fileType="xml")
+ obj.etree = ET.fromstring("<plist>\n<dict/>\n</plist>")
+
+ def logchange(self, logmess, key, old, new):
+ oldstr = str(old) if len(str(old)) < 22 else str(old)[0:20] + "..."
+ newstr = str(new) if len(str(new)) < 22 else str(new)[0:20] + "..."
+ logmess = key + logmess
+ if old is None:
+ logmess = logmess + " New value: " + newstr
+ else:
+ if new is None:
+ logmess = logmess + " Old value: " + oldstr
+ else:
+ logmess = logmess + " Old value: " + oldstr + ", new value: " + newstr
+ self.logger.log(logmess, "W")
+ # Extra verbose logging
+ if len(str(old)) > 21:
+ self.logger.log("Full old value: " + str(old), "I")
+ if len(str(new)) > 21:
+ self.logger.log("Full new value: " + str(new), "I")
+ otype = "string" if isinstance(old, (bytes, str)) else type(old).__name__ # To produce consistent reporting
+ ntype = "string" if isinstance(new, (bytes, str)) else type(new).__name__ # with Python 2 & 3
+ self.logger.log("Types: Old - " + otype + ", New - " + ntype, "I")
+
+class Ulayer(_Ucontainer):
+ def __init__(self, layername, layerdir, font):
+ self._contents = collections.OrderedDict()
+ self.dtree = font.dtree.subTree(layerdir)
+ font.dtree[layerdir].read = True
+ self.layername = layername
+ self.layerdir = layerdir
+ self.font = font
+ fulldir = os.path.join(font.ufodir, layerdir)
+ self.contents = Uplist(font=font, dirn=fulldir, filen="contents.plist")
+ self.dtree["contents.plist"].setinfo(read=True, fileObject=self.contents, fileType="xml")
+
+ if font.UFOversion == "3":
+ if 'layerinfo.plist' in self.dtree:
+ self.layerinfo = Uplist(font=font, dirn=fulldir, filen="layerinfo.plist")
+ self.dtree["layerinfo.plist"].setinfo(read=True, fileObject=self.layerinfo, fileType="xml")
+
+ for glyphn in sorted(self.contents.keys()):
+ glifn = self.contents[glyphn][1].text
+ if glifn in self.dtree:
+ glyph = Uglif(layer=self, filen=glifn)
+ self._contents[glyphn] = glyph
+ self.dtree[glifn].setinfo(read=True, fileObject=glyph, fileType="xml")
+ if glyph.name != glyphn:
+ super(Uglif, glyph).__setattr__("name", glyphn) # Need to use super to bypass normal glyph renaming logic
+ self.font.logger.log("Glyph names in glif and contents.plist did not match for " + glyphn + "; corrected", "W")
+ else:
+ self.font.logger.log("Missing glif " + glifn + " in " + fulldir, "S")
+
+ def setForOutput(self):
+
+ UFOversion = self.font.outparams["UFOversion"]
+ convertg2f1 = True if UFOversion == "2" or self.font.outparams["format1Glifs"] else False
+ dtree = self.font.dtree.subTree(self.layerdir)
+ if self.font.outparams["renameGlifs"]: self.renameGlifs()
+
+ setFileForOutput(dtree, "contents.plist", self.contents, "xml")
+ if "layerinfo" in self.__dict__ and UFOversion == "3":
+ setFileForOutput(dtree, "layerinfo.plist", self.layerinfo, "xml")
+
+ for glyphn in self:
+ glyph = self._contents[glyphn]
+ if convertg2f1: glyph.convertToFormat1()
+ if glyph["advance"] is not None:
+ if glyph["advance"].width is None and glyph["advance"].height is None: glyph.remove("advance")
+ # Normalize so that, when both exist, components come before contour
+ outline = glyph["outline"]
+ if len(outline.components) > 0 and list(outline)[0] == "contour":
+ # Need to move components to the front...
+ contours = outline.contours
+ components = outline.components
+ oldcontours = list(contours) # Easiest way to 'move' components is to delete contours then append back at the end
+ for contour in oldcontours: outline.removeobject(contour, "contour")
+ for contour in oldcontours: outline.appendobject(contour, "contour")
+
+ setFileForOutput(dtree, glyph.filen, glyph, "xml")
+
+ def renameGlifs(self):
+ namelist = []
+ for glyphn in sorted(self.keys()):
+ glyph = self._contents[glyphn]
+ filename = makeFileName(glyphn, namelist)
+ namelist.append(filename.lower())
+ filename += ".glif"
+ if filename != glyph.filen:
+ self.renameGlif(glyphn, glyph, filename)
+
+ def renameGlif(self, glyphn, glyph, newname):
+ self.font.logger.log("Renaming glif for " + glyphn + " from " + glyph.filen + " to " + newname, "I")
+ self.dtree.removedfiles[glyph.filen] = newname # Track so original glif does not get reported as invalid
+ glyph.filen = newname
+ self.contents[glyphn][1].text = newname
+
+ def addGlyph(self, glyph):
+ glyphn = glyph.name
+ if glyphn in self._contents: self.font.logger.log(glyphn + " already in font", "X")
+ self._contents[glyphn] = glyph
+ # Set glif name
+ glifn = makeFileName(glyphn)
+ names = []
+ while glifn in self.contents: # need to check for duplicate glif names
+ names.append(glifn)
+ glifn = makeFileName(glyphn, names)
+ glifn += ".glif"
+ glyph.filen = glifn
+ # Add to contents.plist and dtree
+ self.contents.addval(glyphn, "string", glifn)
+ self.dtree[glifn] = UT.dirTreeItem(read=False, added=True, fileObject=glyph, fileType="xml")
+
+ def delGlyph(self, glyphn):
+ self.dtree.removedfiles[self[glyphn].filen] = "deleted" # Track so original glif does not get reported as invalid
+ del self._contents[glyphn]
+ self.contents.remove(glyphn)
+
+
+class Uplist(ETU.xmlitem, _plist):
+ def __init__(self, font=None, dirn=None, filen=None, parse=True):
+ if dirn is None and font: dirn = font.ufodir
+ logger = font.logger if font else silfont.core.loggerobj()
+ ETU.xmlitem.__init__(self, dirn, filen, parse, logger)
+ self.type = "plist"
+ self.font = font
+ self.outparams = None
+ if filen and dirn: self.populate_dict()
+
+ def populate_dict(self):
+ self._contents.clear() # Clear existing contents, if any
+ pl = self.etree[0]
+ if pl.tag == "dict":
+ for i in range(0, len(pl), 2):
+ key = pl[i].text
+ self._contents[key] = [pl[i], pl[i + 1]] # The two elements for the item
+ else: # Assume array of 2 element arrays (eg layercontents.plist)
+ for i in range(len(pl)):
+ self._contents[i] = pl[i]
+
+
+class Uglif(ETU.xmlitem):
+ # Unlike plists, glifs can have multiples of some sub-elements (eg anchors) so create lists for those
+
+ def __init__(self, layer, filen=None, parse=True, name=None, format=None):
+ dirn = os.path.join(layer.font.ufodir, layer.layerdir)
+ ETU.xmlitem.__init__(self, dirn, filen, parse, layer.font.logger) # Will read item from file if dirn and filen both present
+ self.type = "glif"
+ self.layer = layer
+ self.format = format if format else '2'
+ self.name = name
+ self.outparams = None
+ self.glifElemOrder = self.layer.font.outparams["glifElemOrder"]
+ # Set initial values for sub-objects
+ for elem in self.glifElemOrder:
+ if elem in _glifElemMulti:
+ self._contents[elem] = []
+ else:
+ self._contents[elem] = None
+ if self.etree is not None: self.process_etree()
+
+ def __setattr__(self, name, value):
+ if name == "name" and getattr(self, "name", None): # Existing glyph name is being changed
+ oname = self.name
+ if value in self.layer._contents: self.layer.font.logger.log(name + " already in font", "X")
+ # Update the _contents dictionary
+ del self.layer._contents[oname]
+ self.layer._contents[value] = self
+ # Set glif name
+ glifn = makeFileName(value)
+ names = []
+ while glifn in self.layer.contents: # need to check for duplicate glif names
+ names.append(glifn)
+ glifn = makeFileName(value, names)
+ glifn += ".glif"
+
+ # Update to contents.plist, filen and dtree
+ self.layer.contents.remove(oname)
+ self.layer.contents.addval(value, "string", glifn)
+ self.layer.dtree.removedfiles[self.filen] = glifn # Track so original glif does not get reported as invalid
+ self.filen = glifn
+ self.layer.dtree[glifn] = UT.dirTreeItem(read=False, added=True, fileObject=self, fileType="xml")
+ super(Uglif, self).__setattr__(name, value)
+
+ def process_etree(self):
+ et = self.etree
+ self.name = getattrib(et, "name")
+ self.format = getattrib(et, "format")
+ if self.format is None:
+ if self.layer.font.UFOversion == "3":
+ self.format = '2'
+ else:
+ self.format = '1'
+ for i in range(len(et)):
+ element = et[i]
+ tag = element.tag
+ if not tag in self.glifElemOrder: self.layer.font.logger.log(
+ "Invalid element " + tag + " in glif " + self.name, "E")
+ if tag in _glifElemF1 or self.format == '2':
+ if tag in _glifElemMulti:
+ self._contents[tag].append(self.makeObject(tag, element))
+ else:
+ self._contents[tag] = self.makeObject(tag, element)
+
+ # Convert UFO2 style anchors to UFO3 anchors
+ if self._contents['outline'] is not None and self.format == "1":
+ for contour in self._contents['outline'].contours[:]:
+ if contour.UFO2anchor:
+ del contour.UFO2anchor["type"] # remove type="move"
+ self.add('anchor', contour.UFO2anchor)
+ self._contents['outline'].removeobject(contour, "contour")
+ if self._contents['outline'] is None: self.add('outline')
+
+ self.format = "2"
+
+ def rebuildET(self):
+ self.etree = ET.Element("glyph")
+ et = self.etree
+ et.attrib["name"] = self.name
+ et.attrib["format"] = self.format
+ # Insert sub-elements
+ for elem in self.glifElemOrder:
+ if elem in _glifElemF1 or self.format == "2": # Check element is valid for glif format
+ item = self._contents[elem]
+ if item is not None:
+ if elem in _glifElemMulti:
+ for object in item:
+ et.append(object.element)
+ else:
+ et.append(item.element)
+
+ def add(self, ename, attrib=None):
+ # Add an element and corresponding object to a glif
+ element = ET.Element(ename)
+ if attrib: element.attrib = attrib
+ if ename == "lib": ET.SubElement(element, "dict")
+ multi = True if ename in _glifElemMulti else False
+
+ if multi and ename not in self._contents:
+ self._contents[ename] = []
+
+ # Check element does not already exist for single elements
+ if ename in self._contents and not multi:
+ if self._contents[ename] is not None: self.layer.font.logger.log("Already an " + ename + " in glif", "X")
+
+ # Add new object
+ if multi:
+ self._contents[ename].append(self.makeObject(ename, element))
+ else:
+ self._contents[ename] = self.makeObject(ename, element)
+
+ def remove(self, ename, index=None, object=None):
+ # Remove object from a glif
+ # For multi objects, an index or object must be supplied to identify which
+ # to delete
+ if ename in _glifElemMulti:
+ item = self._contents[ename]
+ if index is None: index = item.index(object)
+ del item[index]
+ else:
+ self._contents[ename] = None
+
+ def convertToFormat1(self):
+ # Convert to a glif format of 1 (for UFO2) prior to writing out
+ self.format = "1"
+ # Change anchors to UFO2 style anchors. Sort anchors by anchor name first
+ anchororder = sorted(self._contents['anchor'], key=lambda x: x.element.attrib['name'])
+ for anchor in anchororder:
+ element = anchor.element
+ for attrn in ('colour', 'indentifier'): # Remove format 2 attributes
+ if attrn in element.attrib: del element.attrib[attrn]
+ element.attrib['type'] = 'move'
+ contelement = ET.Element("contour")
+ contelement.append(ET.Element("point", element.attrib))
+ self._contents['outline'].appendobject(Ucontour(self._contents['outline'], contelement), "contour")
+ self.remove('anchor', object=anchor)
+
+ def makeObject(self, type, element):
+ if type == 'advance': return Uadvance(self, element)
+ if type == 'unicode': return Uunicode(self, element)
+ if type == 'outline': return Uoutline(self, element)
+ if type == 'lib': return Ulib(self, element)
+ if type == 'note': return Unote(self, element)
+ if type == 'image': return Uimage(self, element)
+ if type == 'guideline': return Uguideline(self, element)
+ if type == 'anchor': return Uanchor(self, element)
+
+
+class Uadvance(Uelement):
+ def __init__(self, glif, element):
+ super(Uadvance, self).__init__(element)
+ self.glif = glif
+ if 'width' in element.attrib:
+ self.width = element.attrib[str('width')]
+ else:
+ self.width = None
+ if 'height' in element.attrib:
+ self.height = element.attrib[str('height')]
+ else:
+ self.height = None
+
+ def __setattr__(self, name, value):
+ if name in ('width', 'height'):
+ if value == "0" : value = None
+ if value is None:
+ if name in self.element.attrib: del self.element.attrib[name]
+ else:
+ value = str(value)
+ self.element.attrib[name] = value
+ super(Uadvance, self).__setattr__(name, value)
+
+class Uunicode(Uelement):
+ def __init__(self, glif, element):
+ super(Uunicode, self).__init__(element)
+ self.glif = glif
+ if 'hex' in element.attrib:
+ self.hex = element.attrib['hex']
+ else:
+ self.hex = ""
+ self.glif.logger.log("No unicode hex attribute for " + glif.name, "E")
+
+ def __setattr__(self, name, value):
+ if name == "hex": self.element.attrib['hex'] = value
+ super(Uunicode, self).__setattr__(name, value)
+
+
+class Unote(Uelement):
+ def __init__(self, glif, element):
+ self.glif = glif
+ super(Unote, self).__init__(element)
+
+
+class Uimage(Uelement):
+ def __init__(self, glif, element):
+ self.glif = glif
+ super(Uimage, self).__init__(element)
+
+
+class Uguideline(Uelement):
+ def __init__(self, glif, element):
+ self.glif = glif
+ super(Uguideline, self).__init__(element)
+
+
+class Uanchor(Uelement):
+ def __init__(self, glif, element):
+ self.glif = glif
+ super(Uanchor, self).__init__(element)
+
+
+class Uoutline(Uelement):
+ def __init__(self, glif, element):
+ super(Uoutline, self).__init__(element)
+ self.glif = glif
+ self.components = []
+ self.contours = []
+ for tag in self._contents:
+ if tag == "component":
+ for component in self._contents[tag]:
+ self.components.append(Ucomponent(self, component))
+ if tag == "contour":
+ for contour in self._contents[tag]:
+ self.contours.append(Ucontour(self, contour))
+
+ def removeobject(self, obj, typ):
+ super(Uoutline, self).remove(obj.element)
+ if typ == "component": self.components.remove(obj)
+ if typ == "contour": self.contours.remove(obj)
+
+ def replaceobject(self, oldobj, newobj, typ):
+ eindex = list(self.element).index(oldobj.element)
+ super(Uoutline, self).replace(eindex, newobj.element)
+ if typ == "component":
+ cindex = self.components.index(oldobj)
+ self.components[cindex]= newobj
+ if typ == "contour":
+ cindex = self.contours.index(oldobj)
+ self.contours[cindex]= newobj
+
+ def appendobject(self, item, typ): # Item can be an contour/component object, element or attribute list
+ if isinstance(item, (Ucontour, Ucomponent)):
+ obj = item
+ else:
+ if isinstance(item, dict):
+ elem = ET.Element(typ)
+ elem.attrib = item
+ elif isinstance(item, ET.Element):
+ elem = item
+ else:
+ self.glif.logger.log("item should be dict, element, Ucontour or Ucomponent", "S")
+ if typ == 'component':
+ obj = Ucomponent(self,elem)
+ else:
+ obj = Ucontour(self,elem)
+ super(Uoutline, self).append(obj.element)
+ if typ == "component": self.components.append(obj)
+ if typ == "contour": self.contours.append(obj)
+
+ def insertobject(self, index, item, typ): # Needs updating to match appendobject
+ self.glif.logger.log("insertobject currently buggy so don't use!", "X")
+ # Bug is that index for super... should be different than components/contours.
+ # need to think through logic to sort this out...
+ # May need to take some logic from appendobject and some from replaceobj
+
+ #super(Uoutline, self).insert(index, obj.element)
+ #if typ == "component": self.components.insert(index, obj)
+ #if typ == "contour": self.contours.insert(index, obj)
+
+
+class Ucomponent(Uelement):
+ def __init__(self, outline, element):
+ super(Ucomponent, self).__init__(element)
+ self.outline = outline
+
+
+class Ucontour(Uelement):
+ def __init__(self, outline, element):
+ super(Ucontour, self).__init__(element)
+ self.outline = outline
+ self.UFO2anchor = None
+ points = self._contents['point']
+ # Identify UFO2-style anchor points
+ if len(points) == 1 and "type" in points[0].attrib:
+ if points[0].attrib["type"] == "move":
+ if "name" in points[0].attrib:
+ self.UFO2anchor = points[0].attrib
+ else:
+ self.outline.glif.layer.font.logger.log(
+ "Glyph " + self.outline.glif.name + " contains a single-point contour with no anchor name", "E")
+
+
+class Ulib(_Ucontainer, _plist):
+ # For glif lib elements; top-level lib files use Uplist
+ def __init__(self, glif, element):
+ self.glif = glif
+ self.element = element # needs both element and etree for compatibility
+ self.etree = element # with other glif components and _plist methods
+ self._contents = {}
+ self.reindex()
+
+ def reindex(self):
+ self._contents.clear() # Clear existing contents, if any
+ pl = self.element[0]
+ if pl.tag == "dict":
+ for i in range(0, len(pl), 2):
+ key = pl[i].text
+ self._contents[key] = [pl[i], pl[i + 1]] # The two elements for the item
+
+
+class UfeatureFile(UtextFile):
+ def __init__(self, font, dirn, filen):
+ super(UfeatureFile, self).__init__(font, dirn, filen)
+
+
+def writeXMLobject(dtreeitem, params, dirn, filen, exists, fobject=False):
+ object = dtreeitem if fobject else dtreeitem.fileObject # Set fobject to True if a file object is passed ratehr than dtreeitem
+ if object.outparams: params = object.outparams # override default params with object-specific ones
+ indentFirst = params["indentFirst"]
+ attribOrder = {}
+ if object.type in params['attribOrders']: attribOrder = params['attribOrders'][object.type]
+ if object.type == "plist":
+ indentFirst = params["plistIndentFirst"]
+ object.etree.attrib[".doctype"] = 'plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"'
+
+ # Format ET data if any data parameters are set
+ if params["sortDicts"] or params["precision"] is not None: normETdata(object.etree, params, type=object.type)
+
+ etw = ETU.ETWriter(object.etree, attributeOrder=attribOrder, indentIncr=params["indentIncr"],
+ indentFirst=indentFirst, indentML=params["indentML"], precision=params["precision"],
+ floatAttribs=params["floatAttribs"], intAttribs=params["intAttribs"])
+ object.outxmlstr=etw.serialize_xml()
+ # Now we have the output xml, need to compare with existing item's xml, if present
+ changed = True
+
+ if exists: # File already on disk
+ if exists == "same": # Output and input locations the same
+ oxmlstr = object.inxmlstr
+ else: # Read existing XML from disk
+ oxmlstr = ""
+ try:
+ oxml = io.open(os.path.join(dirn, filen), "r", encoding="utf-8")
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ for line in oxml.readlines():
+ oxmlstr += line
+ oxml.close()
+ with warnings.catch_warnings():
+ warnings.simplefilter("ignore", UnicodeWarning)
+ if oxmlstr == object.outxmlstr: changed = False
+
+ if changed: object.write_to_file(dirn, filen)
+ if not fobject: dtreeitem.written = True # Mark as True, even if not changed - the file should still be there!
+ return changed # Boolean to indicate file updated on disk
+
+
+def setFileForOutput(dtree, filen, fileObject, fileType): # Put details in dtree, creating item if needed
+ if filen not in dtree:
+ dtree[filen] = UT.dirTreeItem()
+ dtree[filen].added = True
+ dtree[filen].setinfo(fileObject=fileObject, fileType=fileType, towrite=True)
+
+
+def writeToDisk(dtree, outdir, font, odtree=None, logindent="", changes = False):
+ if odtree is None: odtree = {}
+ # Make lists of items in dtree and odtree with type prepended for sorting and comparison purposes
+ dtreelist = []
+ for filen in dtree: dtreelist.append(dtree[filen].type + filen)
+ dtreelist.sort()
+ odtreelist = []
+ if odtree == {}:
+ locationtype = "Empty"
+ else:
+ if outdir == font.ufodir:
+ locationtype = "Same"
+ else:
+ locationtype = "Different"
+ for filen in odtree: odtreelist.append(odtree[filen].type + filen)
+ odtreelist.sort()
+
+ okey = odtreelist.pop(0) if odtreelist != [] else None
+
+ for key in dtreelist:
+ type = key[0:1]
+ filen = key[1:]
+ dtreeitem = dtree[filen]
+
+ while okey and okey < key: # Item in output UFO no longer needed
+ ofilen = okey[1:]
+ if okey[0:1] == "f":
+ logmess = 'Deleting ' + ofilen + ' from existing output UFO'
+ os.remove(os.path.join(outdir, ofilen))
+ else:
+ logmess = 'Deleting directory ' + ofilen + ' from existing output UFO'
+ shutil.rmtree(os.path.join(outdir, ofilen))
+ if ofilen not in dtree.removedfiles: font.logger.log(logmess, "W") # No need to log for renamed files
+ okey = odtreelist.pop(0) if odtreelist != [] else None
+
+ if key == okey:
+ exists = locationtype
+ okey = odtreelist.pop(0) if odtreelist != [] else None # Ready for next loop
+ else:
+ exists = False
+
+ if dtreeitem.type == "f":
+ if dtreeitem.towrite:
+ font.logger.log(logindent + filen, "V")
+ if dtreeitem.fileType == "xml":
+ if dtreeitem.fileObject: # Only write if object has items
+ if dtreeitem.fileObject.type == "glif":
+ glif = dtreeitem.fileObject
+ if glif["lib"] is not None: # Delete lib if no items in it
+ if glif["lib"].__len__() == 0:
+ glif.remove("lib")
+ # Sort UFO3 anchors by name (UFO2 anchors will have been sorted on conversion)
+ glif["anchor"].sort(key=lambda anchor: anchor.element.get("name"))
+ glif.rebuildET()
+ result = writeXMLobject(dtreeitem, font.outparams, outdir, filen, exists)
+ if result: changes = True
+ else: # Delete existing item if the current object is empty
+ if exists:
+ font.logger.log('Deleting empty item ' + filen + ' from existing output UFO', "I")
+ os.remove(os.path.join(outdir, filen))
+ changes = True
+ elif dtreeitem.fileType == "text":
+ dtreeitem.fileObject.write(dtreeitem, outdir, filen, exists)
+ ## Need to add code for other file types
+ else:
+ if filen in dtree.removedfiles:
+ if exists:
+ os.remove(os.path.join(outdir, filen)) # Silently remove old file for renamed files
+ changes = True
+ exists = False
+ else: # File should not have been in original UFO
+ if exists == "same":
+ font.logger.log('Deleting ' + filen + ' from existing UFO', "W")
+ os.remove(os.path.join(outdir, filen))
+ changes = True
+ exists = False
+ else:
+ if not dtreeitem.added:
+ font.logger.log('Skipping invalid file ' + filen + ' from input UFO', "W")
+ if exists:
+ font.logger.log('Deleting ' + filen + ' from existing output UFO', "W")
+ os.remove(os.path.join(outdir, filen))
+ changes = True
+
+ else: # Must be directory
+ if not dtreeitem.read:
+ font.logger.log(logindent + "Skipping invalid input directory " + filen, "W")
+ if exists:
+ font.logger.log('Deleting directory ' + filen + ' from existing output UFO', "W")
+ shutil.rmtree(os.path.join(outdir, filen))
+ changes = True
+ continue
+ font.logger.log(logindent + "Processing " + filen + " directory", "I")
+ subdir = os.path.join(outdir, filen)
+ if isinstance(dtreeitem.fileObject, Udirectory):
+ dtreeitem.fileObject.write(dtreeitem, outdir)
+ else:
+ if not os.path.exists(subdir): # If outdir does not exist, create it
+ try:
+ os.mkdir(subdir)
+ except Exception as e:
+ print(e)
+ sys.exit(1)
+ changes = True
+
+ if exists:
+ subodtree = odtree[filen].dirtree
+ else:
+ subodtree = {}
+ subindent = logindent + " "
+ changes = writeToDisk(dtreeitem.dirtree, subdir, font, subodtree, subindent, changes)
+ if os.listdir(subdir) == []:
+ os.rmdir(subdir) # Delete directory if empty
+ changes = True
+
+ while okey: # Any remaining items in odree list are no longer needed
+ ofilen = okey[1:]
+ if okey[0:1] == "f":
+ logmess = 'Deleting ' + ofilen + ' from existing output UFO'
+ os.remove(os.path.join(outdir, ofilen))
+ changes = True
+ else:
+ logmess = 'Deleting directory ' + ofilen + ' from existing output UFO', "W"
+ shutil.rmtree(os.path.join(outdir, ofilen))
+ changes = True
+ if ofilen not in dtree.removedfiles: font.logger.log(logmess, "W") # No need to log warning for removed files
+ okey = odtreelist.pop(0) if odtreelist != [] else None
+ return changes
+
+def normETdata(element, params, type):
+ # Recursively normalise the data an an ElementTree element
+ for subelem in element:
+ normETdata(subelem, params, type)
+
+ precision = params["precision"]
+ if precision is not None:
+ if element.tag in ("integer", "real"):
+ num = round(float(element.text), precision)
+ if num == int(num):
+ element.tag = "integer"
+ element.text = "{}".format(int(num))
+ else:
+ element.tag = "real"
+ element.text = "{}".format(num)
+
+ if params["sortDicts"] and element.tag == "dict":
+ edict = {}
+ elist = []
+ for i in range(0, len(element), 2):
+ edict[element[i].text] = [element[i], element[i + 1]]
+ elist.append(element[i].text)
+ keylist = sorted(edict.keys())
+ if elist != keylist:
+ i = 0
+ for key in keylist:
+ element[i] = edict[key][0]
+ element[i + 1] = edict[key][1]
+ i = i + 2
+
+
+def getattrib(element, attrib): return element.attrib[attrib] if attrib in element.attrib else None
+
+
+def makeFileName(name, namelist=None):
+ if namelist is None: namelist = []
+ # Replace illegal characters and add _ after UC letters
+ newname = ""
+ for x in name:
+ if x in _illegalChars:
+ x = "_"
+ else:
+ if x != x.lower(): x += "_"
+ newname += x
+ # Replace initial . if present
+ if newname[0] == ".": newname = "_" + newname[1:]
+ parts = []
+ for part in newname.split("."):
+ if part in _reservedNames:
+ part = "_" + part
+ parts.append(part)
+ name = ".".join(parts)
+ if name.lower() in namelist: # case-insensitive name already used, so add a suffix
+ newname = None
+ i = 1
+ while newname is None:
+ test = name + '{0:015d}'.format(i)
+ if not (test.lower() in namelist): newname = test
+ i += 1
+ name = newname
+ return name