#!/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 = "\"*+/:> 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 = "\n\n\npublic.default\nglyphs\n\n\n" 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("")) 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("\n\n") 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