Home | History | Annotate | Download | only in tables
      1 from __future__ import print_function, division, absolute_import
      2 from fontTools.misc.py23 import *
      3 from fontTools import ttLib
      4 from fontTools.misc import sstruct
      5 from fontTools.misc.fixedTools import fixedToFloat, floatToFixed
      6 from fontTools.misc.textTools import safeEval
      7 from fontTools.ttLib import TTLibError
      8 from . import DefaultTable
      9 import array
     10 import struct
     11 import logging
     12 
     13 
     14 log = logging.getLogger(__name__)
     15 
     16 # Apple's documentation of 'avar':
     17 # https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6avar.html
     18 
     19 AVAR_HEADER_FORMAT = """
     20     > # big endian
     21     majorVersion:  H
     22     minorVersion:  H
     23     reserved:      H
     24     axisCount:     H
     25 """
     26 assert sstruct.calcsize(AVAR_HEADER_FORMAT) == 8, sstruct.calcsize(AVAR_HEADER_FORMAT)
     27 
     28 
     29 class table__a_v_a_r(DefaultTable.DefaultTable):
     30 
     31     dependencies = ["fvar"]
     32 
     33     def __init__(self, tag=None):
     34         DefaultTable.DefaultTable.__init__(self, tag)
     35         self.segments = {}
     36 
     37     def compile(self, ttFont):
     38         axisTags = [axis.axisTag for axis in ttFont["fvar"].axes]
     39         header = {
     40             "majorVersion": 1,
     41             "minorVersion": 0,
     42             "reserved": 0,
     43             "axisCount": len(axisTags)
     44         }
     45         result = [sstruct.pack(AVAR_HEADER_FORMAT, header)]
     46         for axis in axisTags:
     47             mappings = sorted(self.segments[axis].items())
     48             result.append(struct.pack(">H", len(mappings)))
     49             for key, value in mappings:
     50                 fixedKey = floatToFixed(key, 14)
     51                 fixedValue = floatToFixed(value, 14)
     52                 result.append(struct.pack(">hh", fixedKey, fixedValue))
     53         return bytesjoin(result)
     54 
     55     def decompile(self, data, ttFont):
     56         axisTags = [axis.axisTag for axis in ttFont["fvar"].axes]
     57         header = {}
     58         headerSize = sstruct.calcsize(AVAR_HEADER_FORMAT)
     59         header = sstruct.unpack(AVAR_HEADER_FORMAT, data[0:headerSize])
     60         majorVersion = header["majorVersion"]
     61         if majorVersion != 1:
     62             raise TTLibError("unsupported 'avar' version %d" % majorVersion)
     63         pos = headerSize
     64         for axis in axisTags:
     65             segments = self.segments[axis] = {}
     66             numPairs = struct.unpack(">H", data[pos:pos+2])[0]
     67             pos = pos + 2
     68             for _ in range(numPairs):
     69                 fromValue, toValue = struct.unpack(">hh", data[pos:pos+4])
     70                 segments[fixedToFloat(fromValue, 14)] = fixedToFloat(toValue, 14)
     71                 pos = pos + 4
     72 
     73     def toXML(self, writer, ttFont):
     74         axisTags = [axis.axisTag for axis in ttFont["fvar"].axes]
     75         for axis in axisTags:
     76             writer.begintag("segment", axis=axis)
     77             writer.newline()
     78             for key, value in sorted(self.segments[axis].items()):
     79                 # roundtrip float -> fixed -> float to normalize TTX output
     80                 # as dumped after decompiling or straight from varLib
     81                 key = fixedToFloat(floatToFixed(key, 14), 14)
     82                 value = fixedToFloat(floatToFixed(value, 14), 14)
     83                 writer.simpletag("mapping", **{"from": key, "to": value})
     84                 writer.newline()
     85             writer.endtag("segment")
     86             writer.newline()
     87 
     88     def fromXML(self, name, attrs, content, ttFont):
     89         if name == "segment":
     90             axis = attrs["axis"]
     91             segment = self.segments[axis] = {}
     92             for element in content:
     93                 if isinstance(element, tuple):
     94                     elementName, elementAttrs, _ = element
     95                     if elementName == "mapping":
     96                         fromValue = safeEval(elementAttrs["from"])
     97                         toValue = safeEval(elementAttrs["to"])
     98                         if fromValue in segment:
     99                             log.warning("duplicate entry for %s in axis '%s'",
    100                                         fromValue, axis)
    101                         segment[fromValue] = toValue
    102