| #!/usr/bin/env python |
| |
| # Copyright (C) 2014 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the 'License'); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an 'AS IS' BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| """ |
| Rename the PS name of the input font. |
| |
| OpenType fonts (*.otf) and TrueType Collections (*.ttc) are not currently supported. They are copied to the destination |
| without renaming. XML files are also copied in case they are passed there by mistake. |
| |
| Usage: build_font_single.py /path/to/input_font.ttf /path/to/output_font.ttf |
| |
| """ |
| |
| import glob |
| import os |
| import re |
| import shutil |
| import sys |
| import xml.etree.ElementTree as etree |
| |
| # Prevent .pyc files from being created. |
| sys.dont_write_bytecode = True |
| |
| # fontTools is available at platform/external/fonttools |
| from fontTools import ttx |
| |
| |
| class FontInfo(object): |
| family = None |
| style = None |
| version = None |
| ends_in_regular = False |
| fullname = None |
| |
| |
| class InvalidFontException(Exception): |
| pass |
| |
| |
| # A constant to copy the font without modifying. This is useful when running |
| # locally and speed up the time to build the SDK. |
| COPY_ONLY = False |
| |
| # These constants represent the value of nameID parameter in the namerecord for |
| # different information. |
| # see http://scripts.sil.org/cms/scripts/page.php?item_id=IWS-Chapter08#3054f18b |
| NAMEID_FAMILY = 1 |
| NAMEID_STYLE = 2 |
| NAMEID_FULLNAME = 4 |
| NAMEID_VERSION = 5 |
| |
| NAMEID_LIST = [NAMEID_FAMILY, NAMEID_STYLE, NAMEID_FULLNAME, NAMEID_VERSION] |
| NAMEID_LIST_MIN = min(NAMEID_LIST) |
| NAMEID_LIST_MAX = max(NAMEID_LIST) |
| |
| # A list of extensions to process. |
| EXTENSIONS = ['.ttf', '.ttc', '.otf', '.xml'] |
| |
| def main(argv): |
| if len(argv) < 2: |
| print 'Incorrect usage: ' + str(argv) |
| sys.exit('Usage: build_font_single.py /path/to/input/font.ttf /path/to/out/font.ttf') |
| dest_path = argv[-1] |
| input_path = argv[0] |
| extension = os.path.splitext(input_path)[1].lower() |
| if extension in EXTENSIONS: |
| if not COPY_ONLY and extension == '.ttf': |
| convert_font(input_path, dest_path) |
| return |
| shutil.copy(input_path, dest_path) |
| |
| |
| def convert_font(input_path, dest_path): |
| filename = os.path.basename(input_path) |
| print 'Converting font: ' + filename |
| # the path to the output file. The file name is the fontfilename.ttx |
| ttx_path = dest_path[:-1] + 'x' |
| try: |
| # run ttx to generate an xml file in the output folder which represents all |
| # its info |
| ttx_args = ['--no-recalc-timestamp', '-q', '-o', ttx_path, input_path] |
| ttx.main(ttx_args) |
| # now parse the xml file to change its PS name. |
| tree = etree.parse(ttx_path) |
| root = tree.getroot() |
| for name in root.iter('name'): |
| update_tag(name, get_font_info(name)) |
| tree.write(ttx_path, xml_declaration=True, encoding='utf-8') |
| # generate the udpated font now. |
| ttx_args = ['-q', '-o', dest_path, ttx_path] |
| ttx.main(ttx_args) |
| except InvalidFontException: |
| # assume, like for .ttc and .otf that font might be valid, but warn. |
| print 'Family and/or Style nameIDs not found in '+ filename |
| shutil.copy(input_path, dest_path) |
| except Exception as e: |
| print 'Error converting font: ' + filename |
| print e |
| # Some fonts are too big to be handled by the ttx library. |
| # Just copy paste them. |
| shutil.copy(input_path, dest_path) |
| try: |
| # delete the temp ttx file is it exists. |
| os.remove(ttx_path) |
| except OSError: |
| pass |
| |
| |
| def get_font_info(tag): |
| """ Returns a list of FontInfo representing the various sets of namerecords |
| found in the name table of the font. """ |
| fonts = [] |
| font = None |
| last_name_id = sys.maxint |
| for namerecord in tag.iter('namerecord'): |
| if 'nameID' in namerecord.attrib: |
| name_id = int(namerecord.attrib['nameID']) |
| # skip irrelevant records |
| if name_id < NAMEID_LIST_MIN or name_id > NAMEID_LIST_MAX: |
| continue |
| # A new font should be created for each platform, encoding and language |
| # id. But, since the nameIDs are sorted, we use the easy approach of |
| # creating a new one when the nameIDs reset. |
| if name_id <= last_name_id and font is not None: |
| fonts.append(font) |
| font = None |
| last_name_id = name_id |
| if font is None: |
| font = FontInfo() |
| if name_id == NAMEID_FAMILY: |
| font.family = namerecord.text.strip() |
| if name_id == NAMEID_STYLE: |
| font.style = namerecord.text.strip() |
| if name_id == NAMEID_FULLNAME: |
| font.ends_in_regular = ends_in_regular(namerecord.text) |
| font.fullname = namerecord.text.strip() |
| if name_id == NAMEID_VERSION: |
| font.version = get_version(namerecord.text) |
| if font is not None: |
| fonts.append(font) |
| return fonts |
| |
| |
| def update_tag(tag, fonts): |
| last_name_id = sys.maxint |
| fonts_iterator = fonts.__iter__() |
| font = None |
| for namerecord in tag.iter('namerecord'): |
| if 'nameID' in namerecord.attrib: |
| name_id = int(namerecord.attrib['nameID']) |
| # skip irrelevant records |
| if name_id < NAMEID_LIST_MIN or name_id > NAMEID_LIST_MAX: |
| continue |
| if name_id <= last_name_id: |
| font = fonts_iterator.next() |
| font = update_font_name(font) |
| last_name_id = name_id |
| if name_id == NAMEID_FAMILY: |
| namerecord.text = font.family |
| if name_id == NAMEID_FULLNAME: |
| namerecord.text = font.fullname |
| |
| |
| def update_font_name(font): |
| """ Compute the new font family name and font fullname. If the font has a |
| valid version, it's sanitized and appended to the font family name. The |
| font fullname is then created by joining the new family name and the |
| style. If the style is 'Regular', it is appended only if the original font |
| had it. """ |
| if font.family is None or font.style is None: |
| raise InvalidFontException('Font doesn\'t have proper family name or style') |
| if font.version is not None: |
| new_family = font.family + font.version |
| else: |
| new_family = font.family |
| if font.style is 'Regular' and not font.ends_in_regular: |
| font.fullname = new_family |
| else: |
| font.fullname = new_family + ' ' + font.style |
| font.family = new_family |
| return font |
| |
| |
| def ends_in_regular(string): |
| """ According to the specification, the font fullname should not end in |
| 'Regular' for plain fonts. However, some fonts don't obey this rule. We |
| keep the style info, to minimize the diff. """ |
| string = string.strip().split()[-1] |
| return string is 'Regular' |
| |
| |
| def get_version(string): |
| string = string.strip() |
| # The spec says that the version string should start with "Version ". But not |
| # all fonts do. So, we return the complete string if it doesn't start with |
| # the prefix, else we return the rest of the string after sanitizing it. |
| prefix = 'Version ' |
| if string.startswith(prefix): |
| string = string[len(prefix):] |
| return sanitize(string) |
| |
| |
| def sanitize(string): |
| """ Remove non-standard chars. """ |
| return re.sub(r'[^\w-]+', '', string) |
| |
| if __name__ == '__main__': |
| main(sys.argv[1:]) |