108 lines
3.2 KiB
Python
Raw Normal View History

from __future__ import print_function, division, absolute_import
from fontTools.misc.py23 import *
from fontTools import ttLib
from fontTools.misc.textTools import safeEval
2013-11-27 02:34:11 -05:00
from . import DefaultTable
import sys
import array
import logging
log = logging.getLogger(__name__)
class table__h_m_t_x(DefaultTable.DefaultTable):
2015-04-26 02:01:01 -04:00
headerTag = 'hhea'
advanceName = 'width'
sideBearingName = 'lsb'
numberOfMetricsName = 'numberOfHMetrics'
2015-04-26 02:01:01 -04:00
def decompile(self, data, ttFont):
numGlyphs = ttFont['maxp'].numGlyphs
numberOfMetrics = int(getattr(ttFont[self.headerTag], self.numberOfMetricsName))
if numberOfMetrics > numGlyphs:
numberOfMetrics = numGlyphs # We warn later.
if len(data) < 4 * numberOfMetrics:
raise ttLib.TTLibError("not enough '%s' table data" % self.tableTag)
# Note: advanceWidth is unsigned, but we read/write as signed.
metrics = array.array("h", data[:4 * numberOfMetrics])
2013-11-27 02:40:30 -05:00
if sys.byteorder != "big":
metrics.byteswap()
data = data[4 * numberOfMetrics:]
numberOfSideBearings = numGlyphs - numberOfMetrics
sideBearings = array.array("h", data[:2 * numberOfSideBearings])
data = data[2 * numberOfSideBearings:]
2013-11-27 02:40:30 -05:00
if sys.byteorder != "big":
sideBearings.byteswap()
if data:
log.warning("too much '%s' table data" % self.tableTag)
self.metrics = {}
2014-05-14 00:30:02 -04:00
glyphOrder = ttFont.getGlyphOrder()
for i in range(numberOfMetrics):
2014-05-14 00:30:02 -04:00
glyphName = glyphOrder[i]
self.metrics[glyphName] = list(metrics[i*2:i*2+2])
lastAdvance = metrics[-2]
for i in range(numberOfSideBearings):
2014-05-14 00:30:02 -04:00
glyphName = glyphOrder[i + numberOfMetrics]
self.metrics[glyphName] = [lastAdvance, sideBearings[i]]
2015-04-26 02:01:01 -04:00
def compile(self, ttFont):
metrics = []
for glyphName in ttFont.getGlyphOrder():
metrics.append(self.metrics[glyphName])
lastAdvance = metrics[-1][0]
lastIndex = len(metrics)
while metrics[lastIndex-2][0] == lastAdvance:
lastIndex -= 1
if lastIndex <= 1:
# all advances are equal
lastIndex = 1
break
additionalMetrics = metrics[lastIndex:]
additionalMetrics = [sb for advance, sb in additionalMetrics]
metrics = metrics[:lastIndex]
setattr(ttFont[self.headerTag], self.numberOfMetricsName, len(metrics))
2015-04-26 02:01:01 -04:00
allMetrics = []
for item in metrics:
allMetrics.extend(item)
allMetrics = array.array("h", allMetrics)
2013-11-27 02:40:30 -05:00
if sys.byteorder != "big":
allMetrics.byteswap()
data = allMetrics.tostring()
2015-04-26 02:01:01 -04:00
additionalMetrics = array.array("h", additionalMetrics)
2013-11-27 02:40:30 -05:00
if sys.byteorder != "big":
additionalMetrics.byteswap()
data = data + additionalMetrics.tostring()
return data
2015-04-26 02:01:01 -04:00
def toXML(self, writer, ttFont):
2013-11-27 04:15:34 -05:00
names = sorted(self.metrics.keys())
for glyphName in names:
advance, sb = self.metrics[glyphName]
writer.simpletag("mtx", [
2015-04-26 02:01:01 -04:00
("name", glyphName),
(self.advanceName, advance),
(self.sideBearingName, sb),
])
writer.newline()
2015-04-26 02:01:01 -04:00
def fromXML(self, name, attrs, content, ttFont):
if not hasattr(self, "metrics"):
self.metrics = {}
if name == "mtx":
2015-04-26 02:01:01 -04:00
self.metrics[attrs["name"]] = [safeEval(attrs[self.advanceName]),
safeEval(attrs[self.sideBearingName])]
2014-05-14 13:51:10 -06:00
def __delitem__(self, glyphName):
del self.metrics[glyphName]
2015-04-26 02:01:01 -04:00
def __getitem__(self, glyphName):
return self.metrics[glyphName]
2015-04-26 02:01:01 -04:00
def __setitem__(self, glyphName, advance_sb_pair):
self.metrics[glyphName] = tuple(advance_sb_pair)