hvar.py 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. from fontTools.misc.roundTools import noRound
  2. from fontTools.ttLib import TTFont, newTable
  3. from fontTools.ttLib.tables import otTables as ot
  4. from fontTools.ttLib.tables.otBase import OTTableWriter
  5. from fontTools.varLib import HVAR_FIELDS, VVAR_FIELDS, _add_VHVAR
  6. from fontTools.varLib import builder, models, varStore
  7. from fontTools.misc.fixedTools import fixedToFloat as fi2fl
  8. from fontTools.misc.cliTools import makeOutputFileName
  9. from functools import partial
  10. import logging
  11. log = logging.getLogger("fontTools.varLib.avar")
  12. def _get_advance_metrics(font, axisTags, tableFields):
  13. # There's two ways we can go from here:
  14. # 1. For each glyph, at each master peak, compute the value of the
  15. # advance width at that peak. Then pass these all to a VariationModel
  16. # builder to compute back the deltas.
  17. # 2. For each master peak, pull out the deltas of the advance width directly,
  18. # and feed these to the VarStoreBuilder, forgoing the remodeling step.
  19. # We'll go with the second option, as it's simpler, faster, and more direct.
  20. gvar = font["gvar"]
  21. vhAdvanceDeltasAndSupports = {}
  22. glyphOrder = font.getGlyphOrder()
  23. phantomIndex = tableFields.phantomIndex
  24. for glyphName in glyphOrder:
  25. supports = []
  26. deltas = []
  27. variations = gvar.variations.get(glyphName, [])
  28. for tv in variations:
  29. supports.append(tv.axes)
  30. phantoms = tv.coordinates[-4:]
  31. phantoms = phantoms[phantomIndex * 2 : phantomIndex * 2 + 2]
  32. assert len(phantoms) == 2
  33. phantoms[0] = phantoms[0][phantomIndex] if phantoms[0] is not None else 0
  34. phantoms[1] = phantoms[1][phantomIndex] if phantoms[1] is not None else 0
  35. deltas.append(phantoms[1] - phantoms[0])
  36. vhAdvanceDeltasAndSupports[glyphName] = (deltas, supports)
  37. vOrigDeltasAndSupports = None # TODO
  38. return vhAdvanceDeltasAndSupports, vOrigDeltasAndSupports
  39. def add_HVAR(font):
  40. if "HVAR" in font:
  41. del font["HVAR"]
  42. axisTags = [axis.axisTag for axis in font["fvar"].axes]
  43. getAdvanceMetrics = partial(_get_advance_metrics, font, axisTags, HVAR_FIELDS)
  44. _add_VHVAR(font, axisTags, HVAR_FIELDS, getAdvanceMetrics)
  45. def add_VVAR(font):
  46. if "VVAR" in font:
  47. del font["VVAR"]
  48. getAdvanceMetrics = partial(_get_advance_metrics, font, axisTags, VVAR_FIELDS)
  49. axisTags = [axis.axisTag for axis in font["fvar"].axes]
  50. _add_VHVAR(font, axisTags, VVAR_FIELDS, getAdvanceMetrics)
  51. def main(args=None):
  52. """Add `HVAR` table to variable font."""
  53. if args is None:
  54. import sys
  55. args = sys.argv[1:]
  56. from fontTools import configLogger
  57. from fontTools.designspaceLib import DesignSpaceDocument
  58. import argparse
  59. parser = argparse.ArgumentParser(
  60. "fonttools varLib.hvar",
  61. description="Add `HVAR` table from to variable font.",
  62. )
  63. parser.add_argument("font", metavar="varfont.ttf", help="Variable-font file.")
  64. parser.add_argument(
  65. "-o",
  66. "--output-file",
  67. type=str,
  68. help="Output font file name.",
  69. )
  70. options = parser.parse_args(args)
  71. configLogger(level="WARNING")
  72. font = TTFont(options.font)
  73. if not "fvar" in font:
  74. log.error("Not a variable font.")
  75. return 1
  76. add_HVAR(font)
  77. if "vmtx" in font:
  78. add_VVAR(font)
  79. if options.output_file is None:
  80. outfile = makeOutputFileName(options.font, overWrite=True, suffix=".hvar")
  81. else:
  82. outfile = options.output_file
  83. if outfile:
  84. log.info("Saving %s", outfile)
  85. font.save(outfile)
  86. if __name__ == "__main__":
  87. import sys
  88. sys.exit(main())