120 lines
4.5 KiB
Python
Raw Normal View History

from __future__ import print_function, division, absolute_import
2013-12-06 19:41:49 -05:00
from fontTools.misc.py23 import *
from fontTools.misc import sstruct
2014-09-24 12:43:43 +02:00
from fontTools.misc.textTools import readHex, safeEval
2013-12-06 19:41:49 -05:00
import struct
2013-12-06 19:35:09 -05:00
sbixGlyphHeaderFormat = """
>
originOffsetX: h # The x-value of the point in the glyph relative to its
# lower-left corner which corresponds to the origin of
# the glyph on the screen, that is the point on the
# baseline at the left edge of the glyph.
originOffsetY: h # The y-value of the point in the glyph relative to its
# lower-left corner which corresponds to the origin of
# the glyph on the screen, that is the point on the
# baseline at the left edge of the glyph.
2014-09-24 11:54:32 +02:00
graphicType: 4s # e.g. "png "
2013-12-06 19:35:09 -05:00
"""
sbixGlyphHeaderFormatSize = sstruct.calcsize(sbixGlyphHeaderFormat)
2013-12-06 19:35:09 -05:00
2014-09-24 12:07:35 +02:00
class Glyph(object):
2014-09-24 11:54:32 +02:00
def __init__(self, glyphName=None, referenceGlyphName=None, originOffsetX=0, originOffsetY=0, graphicType=None, imageData=None, rawdata=None, gid=0):
2013-12-06 19:35:09 -05:00
self.gid = gid
self.glyphName = glyphName
self.referenceGlyphName = referenceGlyphName
self.originOffsetX = originOffsetX
self.originOffsetY = originOffsetY
2013-12-06 19:35:09 -05:00
self.rawdata = rawdata
2014-09-24 11:54:32 +02:00
self.graphicType = graphicType
2013-12-06 19:35:09 -05:00
self.imageData = imageData
# fix self.graphicType if it is null terminated or too short
if self.graphicType is not None:
if self.graphicType[-1] == "\0":
self.graphicType = self.graphicType[:-1]
if len(self.graphicType) > 4:
from fontTools import ttLib
raise ttLib.TTLibError("Glyph.graphicType must not be longer than 4 characters.")
elif len(self.graphicType) < 4:
# pad with spaces
self.graphicType += " "[:(4 - len(self.graphicType))]
2013-12-06 19:40:00 -05:00
2013-12-06 19:35:09 -05:00
def decompile(self, ttFont):
self.glyphName = ttFont.getGlyphName(self.gid)
if self.rawdata is None:
from fontTools import ttLib
raise ttLib.TTLibError("No table data to decompile")
2013-12-06 19:35:09 -05:00
if len(self.rawdata) > 0:
if len(self.rawdata) < sbixGlyphHeaderFormatSize:
2013-12-06 19:35:09 -05:00
from fontTools import ttLib
2014-09-24 12:07:35 +02:00
#print "Glyph %i header too short: Expected %x, got %x." % (self.gid, sbixGlyphHeaderFormatSize, len(self.rawdata))
raise ttLib.TTLibError("Glyph header too short.")
2013-12-06 19:40:00 -05:00
sstruct.unpack(sbixGlyphHeaderFormat, self.rawdata[:sbixGlyphHeaderFormatSize], self)
2013-12-06 19:40:00 -05:00
2014-09-24 11:54:32 +02:00
if self.graphicType == "dupe":
2014-09-24 12:07:35 +02:00
# this glyph is a reference to another glyph's image data
gid, = struct.unpack(">H", self.rawdata[sbixGlyphHeaderFormatSize:])
2013-12-06 19:35:09 -05:00
self.referenceGlyphName = ttFont.getGlyphName(gid)
else:
self.imageData = self.rawdata[sbixGlyphHeaderFormatSize:]
2013-12-06 19:35:09 -05:00
self.referenceGlyphName = None
# clean up
del self.rawdata
del self.gid
2013-12-06 19:40:00 -05:00
2013-12-06 19:35:09 -05:00
def compile(self, ttFont):
if self.glyphName is None:
from fontTools import ttLib
2014-09-24 12:07:35 +02:00
raise ttLib.TTLibError("Can't compile Glyph without glyph name")
2013-12-06 19:35:09 -05:00
# TODO: if ttFont has no maxp, cmap etc., ignore glyph names and compile by index?
# (needed if you just want to compile the sbix table on its own)
self.gid = struct.pack(">H", ttFont.getGlyphID(self.glyphName))
2014-09-24 11:54:32 +02:00
if self.graphicType is None:
2013-12-06 19:35:09 -05:00
self.rawdata = ""
else:
self.rawdata = sstruct.pack(sbixGlyphHeaderFormat, self) + self.imageData
2013-12-06 19:40:00 -05:00
2013-12-06 19:35:09 -05:00
def toXML(self, xmlWriter, ttFont):
2014-09-24 11:54:32 +02:00
if self.graphicType == None:
2014-09-23 18:29:42 +02:00
# TODO: ignore empty glyphs?
# a glyph data entry is required for each glyph,
2013-12-06 19:35:09 -05:00
# but empty ones can be calculated at compile time
2014-09-23 18:47:13 +02:00
xmlWriter.simpletag("glyph", name=self.glyphName)
2013-12-06 19:35:09 -05:00
xmlWriter.newline()
return
2014-09-23 18:29:42 +02:00
xmlWriter.begintag("glyph",
2014-09-24 11:54:32 +02:00
graphicType=self.graphicType,
2014-09-23 18:47:13 +02:00
name=self.glyphName,
originOffsetX=self.originOffsetX,
originOffsetY=self.originOffsetY,
)
2013-12-06 19:35:09 -05:00
xmlWriter.newline()
2014-09-24 11:54:32 +02:00
if self.graphicType == "dupe":
# graphicType == "dupe" is a reference to another glyph id.
2013-12-06 19:35:09 -05:00
xmlWriter.simpletag("ref", glyphname=self.referenceGlyphName)
else:
xmlWriter.begintag("hexdata")
xmlWriter.newline()
xmlWriter.dumphex(self.imageData)
xmlWriter.endtag("hexdata")
xmlWriter.newline()
2014-09-23 18:29:42 +02:00
xmlWriter.endtag("glyph")
2013-12-06 19:35:09 -05:00
xmlWriter.newline()
def fromXML(self, name, attrs, content, ttFont):
2013-12-06 19:35:09 -05:00
if name == "ref":
2014-09-23 18:29:42 +02:00
# glyph is a "dupe", i.e. a reference to another glyph's image data.
2013-12-06 19:35:09 -05:00
# in this case imageData contains the glyph id of the reference glyph
# get glyph id from glyphname
2014-09-24 12:43:43 +02:00
self.imageData = struct.pack(">H", ttFont.getGlyphID(safeEval("'''" + attrs["glyphname"] + "'''")))
2013-12-06 19:35:09 -05:00
elif name == "hexdata":
self.imageData = readHex(content)
else:
from fontTools import ttLib
raise ttLib.TTLibError("can't handle '%s' element" % name)