#!/usr/bin/env python3 import argparse from concurrent.futures import ProcessPoolExecutor, as_completed import importlib.util import json import re import shutil import time from functools import partial from os import environ, getcwd, listdir, makedirs, path, remove, getenv from typing import Callable from fontTools.ttLib import TTFont, newTable from fontTools.feaLib.builder import addOpenTypeFeatures from fontTools.merge import Merger from source.py.utils import ( check_font_patcher, verify_glyph_width, compress_folder, download_cn_base_font, get_font_forge_bin, get_font_name, is_ci, match_unicode_names, run, set_font_name, joinPaths, ) from source.py.feature import freeze_feature, get_freeze_config_str FONT_VERSION = "v7.0-beta36" # ========================================================================================= def check_ftcli(): package_name = "foundryToolsCLI" package_installed = importlib.util.find_spec(package_name) is not None if not package_installed: print( f"❗ {package_name} is not found. Please run `pip install foundrytools-cli`" ) exit(1) # ========================================================================================= def parse_args(): parser = argparse.ArgumentParser( description="✨ Builder and optimizer for Maple Mono", ) parser.add_argument( "-v", "--version", action="version", version=f"Maple Mono Builder v{FONT_VERSION}", ) parser.add_argument( "-d", "--dry", dest="dry", action="store_true", help="Output config and exit", ) parser.add_argument( "--debug", action="store_true", help="Add `Debug` suffix to family name, skip optimization", ) feature_group = parser.add_argument_group("Feature Options") feature_group.add_argument( "-n", "--normal", dest="normal", action="store_true", help="Use normal preset, just like `JetBrains Mono` with slashed zero", ) feature_group.add_argument( "--feat", type=lambda x: x.strip().split(","), help="Freeze font features, splited by `,` (e.g. `--feat zero,cv01,ss07,ss08`). No effect on variable format", ) feature_group.add_argument( "--apply-fea-file", default=None, action="store_true", help="Load feature file from `source/features/{regular,italic}.fea` to variable font", ) hint_group = feature_group.add_mutually_exclusive_group() hint_group.add_argument( "--hinted", dest="hinted", default=None, action="store_true", help="Use hinted font as base font in NF / CN / NF-CN (default)", ) hint_group.add_argument( "--no-hinted", dest="hinted", default=None, action="store_false", help="Use unhinted font as base font in NF / CN / NF-CN", ) liga_group = feature_group.add_mutually_exclusive_group() liga_group.add_argument( "--liga", dest="liga", default=None, action="store_true", help="Preserve all the ligatures (default)", ) liga_group.add_argument( "--no-liga", dest="liga", default=None, action="store_false", help="Remove all the ligatures", ) feature_group.add_argument( "--cn-narrow", action="store_true", help="Make CN characters narrow (experimental)", ) build_group = parser.add_argument_group("Build Options") nf_group = build_group.add_mutually_exclusive_group() nf_group.add_argument( "--nerd-font", dest="nerd_font", default=None, action="store_true", help="Build Nerd-Font version (default)", ) nf_group.add_argument( "--no-nerd-font", dest="nerd_font", default=None, action="store_false", help="Do not build Nerd-Font version", ) cn_group = build_group.add_mutually_exclusive_group() cn_group.add_argument( "--cn", dest="cn", default=None, action="store_true", help="Build Chinese version", ) cn_group.add_argument( "--no-cn", dest="cn", default=None, action="store_false", help="Do not build Chinese version (default)", ) build_group.add_argument( "--cn-both", action="store_true", help="Build both `Maple Mono CN` and `Maple Mono NF CN`. Nerd-Font version must be enabled", ) build_group.add_argument( "--ttf-only", action="store_true", help="Only build TTF format", ) build_group.add_argument( "--cache", action="store_true", help="Reuse font cache of TTF, OTF and Woff2 formats", ) build_group.add_argument( "--cn-rebuild", action="store_true", help="Reinstantiate CN base font", ) build_group.add_argument( "--archive", action="store_true", help="Build font archives with config and license. If has `--cache` flag, only archive Nerd-Font and CN formats", ) return parser.parse_args() # ========================================================================================= class FontConfig: def __init__(self, args): self.archive = None self.use_cn_both = None self.ttf_only = None self.debug = None self.apply_fea_file = None # the number of parallel tasks # when run in codespace, this will be 1 self.pool_size = 1 if not getenv("CODESPACE_NAME") else 4 # font family name self.family_name = "Maple Mono" self.family_name_compact = "MapleMono" # whether to use hinted ttf as base font self.use_hinted = True # whether to enable ligature self.enable_liga = True self.feature_freeze = { "cv01": "ignore", "cv02": "ignore", "cv03": "ignore", "cv04": "ignore", "cv31": "ignore", "cv32": "ignore", "cv33": "ignore", "cv34": "ignore", "cv35": "ignore", "cv36": "ignore", "cv96": "ignore", "cv97": "ignore", "cv98": "ignore", "cv99": "ignore", "ss01": "ignore", "ss02": "ignore", "ss03": "ignore", "ss04": "ignore", "ss05": "ignore", "ss06": "ignore", "ss07": "ignore", "ss08": "ignore", "zero": "ignore", } # Nerd-Font settings self.nerd_font = { # whether to enable Nerd-Font "enable": True, # target version of Nerd-Font if font-patcher not exists "version": "3.2.1", # whether to make icon width fixed "mono": False, # prefer to use Font Patcher instead of using prebuild NerdFont base font # if you want to custom build Nerd-Font using font-patcher, you need to set this to True "use_font_patcher": False, # symbol Fonts settings. # default args: ["--complete"] # if not, will use font-patcher to generate fonts # full args: https://github.com/ryanoasis/nerd-fonts?tab=readme-ov-file#font-patcher "glyphs": ["--complete"], # extra args for font-patcher # default args: ["-l", "--careful", "--outputdir", output_nf] # if "mono" is set to True, "--mono" will be added # full args: https://github.com/ryanoasis/nerd-fonts?tab=readme-ov-file#font-patcher "extra_args": [], } # chinese font settings self.cn = { # whether to build Chinese fonts # skip if Chinese base fonts are not founded "enable": False, # whether to patch Nerd-Font "with_nerd_font": True, # fix design language and supported languages "fix_meta_table": True, # whether to clean instantiated base CN fonts "clean_cache": False, # whether to narrow CN glyphs "narrow": False, # whether to hint CN font (will increase about 33% size) "use_hinted": False, # whether to use pre-instantiated static CN font as base font "use_static_base_font": True, } self.glyph_width = 600 self.glyph_width_cn_narrow = 1000 self.__load_config(args.normal) self.__load_args(args) ver = FONT_VERSION self.beta = None if "-" in FONT_VERSION: ver, beta = FONT_VERSION.split("-") self.beta = beta major, minor = ver.split(".") if major.startswith("v"): major = major[1:] self.version_str = f"Version {major}.{minor:03}" def __load_config(self, use_normal): try: config_file_path = ( "./source/preset-normal.json" if use_normal else "config.json" ) with open(config_file_path, "r") as f: data = json.load(f) for prop in [ "family_name", "use_hinted", "enable_liga", "pool_size", "github_mirror", "feature_freeze", "nerd_font", "cn", ]: if prop in data: val = data[prop] setattr( self, prop, val if type(val) is not dict else {**getattr(self, prop), **val}, ) except FileNotFoundError: print(f"🚨 Config file not found: {config_file_path}, use default config") pass except json.JSONDecodeError: print(f"❗ Error: Invalid JSON in config file: {config_file_path}") exit(1) except Exception as e: # Catch any other unexpected error print(f"❗ An unexpected error occurred: {e}") exit(1) def __load_args(self, args): self.archive = args.archive self.use_cn_both = args.cn_both self.debug = args.debug if "font_forge_bin" not in self.nerd_font: self.nerd_font["font_forge_bin"] = get_font_forge_bin() if args.feat is not None: for f in args.feat: if f in self.feature_freeze: self.feature_freeze[f] = "enable" if args.hinted is not None: self.use_hinted = args.hinted if args.liga is not None: self.enable_liga = args.liga if args.nerd_font is not None: self.nerd_font["enable"] = args.nerd_font if args.cn is not None: self.cn["enable"] = args.cn if args.cn_narrow: self.cn["narrow"] = True if args.ttf_only: self.ttf_only = True if args.apply_fea_file: self.apply_fea_file = True if args.cn_rebuild: self.cn["clean_cache"] = True self.cn["use_static_base_font"] = False name_arr = [word.capitalize() for word in self.family_name.split(" ")] if not self.enable_liga: name_arr.append("NL") if self.debug: name_arr.append("Debug") self.family_name = " ".join(name_arr) self.family_name_compact = "".join(name_arr) self.freeze_config_str = get_freeze_config_str( self.feature_freeze, self.enable_liga ) def should_build_nf_cn(self) -> bool: return self.cn["with_nerd_font"] and self.nerd_font["enable"] def toggle_nf_cn_config(self) -> bool: if not self.nerd_font["enable"]: print("❗Nerd-Font version is disabled, skip toggle.") return False self.cn["with_nerd_font"] = not self.cn["with_nerd_font"] return True def get_valid_glyph_width_list(self, cn=False): if cn: return [ 0, self.glyph_width, self.glyph_width_cn_narrow if self.cn["narrow"] else 2 * self.glyph_width, ] else: return [0, self.glyph_width] class BuildOption: def __init__(self, config: FontConfig): # paths self.src_dir = "source" self.output_dir = "fonts" self.output_otf = joinPaths(self.output_dir, "OTF") self.output_ttf = joinPaths(self.output_dir, "TTF") self.output_ttf_hinted = joinPaths(self.output_dir, "TTF-AutoHint") self.output_variable = joinPaths(self.output_dir, "Variable") self.output_woff2 = joinPaths(self.output_dir, "Woff2") self.output_nf = joinPaths(self.output_dir, "NF") self.ttf_base_dir = joinPaths( self.output_dir, "TTF-AutoHint" if config.use_hinted else "TTF" ) self.cn_variable_dir = f"{self.src_dir}/cn" self.cn_static_dir = f"{self.cn_variable_dir}/static" self.cn_base_font_dir = None self.cn_suffix = None self.cn_suffix_compact = None self.output_cn = None # In these subfamilies: # - NameID1 should be the family name # - NameID2 should be the subfamily name # - NameID16 and NameID17 should be removed # Other subfamilies: # - NameID1 should be the family name, append with subfamily name without "Italic" # - NameID2 should be the "Regular" or "Italic" # - NameID16 should be the family name # - NameID17 should be the subfamily name # https://github.com/subframe7536/maple-font/issues/182 # https://github.com/subframe7536/maple-font/issues/183 # # same as `ftcli assistant commit . --ls 400 700` # https://github.com/ftCLI/FoundryTools-CLI/issues/166#issuecomment-2095756721 self.skip_subfamily_list = ["Regular", "Bold", "Italic", "BoldItalic"] self.is_nf_built = False self.is_cn_built = False self.has_cache = ( self.__check_cache_dir(self.output_variable, count=2) and self.__check_cache_dir(self.output_otf) and self.__check_cache_dir(self.output_ttf) and self.__check_cache_dir(self.output_ttf_hinted) and self.__check_cache_dir(self.output_woff2) ) self.github_mirror = environ.get("GITHUB", "github.com") def load_cn_dir_and_suffix(self, with_nerd_font: bool) -> None: if with_nerd_font: self.cn_base_font_dir = self.output_nf self.cn_suffix = "NF CN" self.cn_suffix_compact = "NF-CN" else: self.cn_base_font_dir = joinPaths(self.output_dir, "TTF") self.cn_suffix = self.cn_suffix_compact = "CN" self.output_cn = joinPaths(self.output_dir, self.cn_suffix_compact) def should_use_font_patcher(self, config: FontConfig) -> bool: if not ( len(config.nerd_font["extra_args"]) > 0 or config.nerd_font["use_font_patcher"] or config.nerd_font["glyphs"] != ["--complete"] ): return False if check_font_patcher( version=config.nerd_font["version"], github_mirror=self.github_mirror, ) and not path.exists(config.nerd_font["font_forge_bin"]): print( f"FontForge bin({config.nerd_font['font_forge_bin']}) not found. Use prebuild Nerd-Font base font instead." ) return False return True def should_build_cn(self, config: FontConfig) -> bool: if not config.cn["enable"] and not config.use_cn_both: print( '\nNo `"cn.enable": true` in config.json or no `--cn` / `--cn-both` in argv. Skip CN build.' ) return False return self.__ensure_cn_static_fonts( clean_cache=config.cn["clean_cache"], use_static=config.cn["use_static_base_font"], pool_size=config.pool_size, ) def __ensure_cn_static_fonts( self, clean_cache: bool, use_static: bool, pool_size: int ) -> bool: if clean_cache: print("Clean CN static fonts") shutil.rmtree(self.cn_static_dir, ignore_errors=True) if path.exists(self.cn_static_dir) and len(listdir(self.cn_static_dir)) == 16: return True tag = "cn-base" if is_ci() or use_static: if download_cn_base_font( tag=tag, zip_path="cn-base-static.zip", target_dir=self.cn_static_dir, github_mirror=self.github_mirror, ): return True # Try using variable fonts if static fonts aren't available if ( path.exists(self.cn_variable_dir) and len(listdir(self.cn_variable_dir)) == 2 ): print( "No static CN fonts but detect variable version, start instantiating..." ) instantiate_cn_base( cn_variable_dir=self.cn_variable_dir, cn_static_dir=self.cn_static_dir, pool_size=pool_size, ) return True # Download variable fonts and instantiate if necessary if download_cn_base_font( tag=tag, zip_path="cn-base-variable.zip", target_dir=self.cn_variable_dir, github_mirror=self.github_mirror, ): instantiate_cn_base( cn_variable_dir=self.cn_variable_dir, cn_static_dir=self.cn_static_dir, pool_size=pool_size, ) return True print("\nCN base fonts don't exist. Skip CN build.") return False def __check_cache_dir(self, cache_dir: str, count: int = 16) -> bool: if not path.isdir(cache_dir): return False return len(listdir(cache_dir)) == count def handle_ligatures( font: TTFont, enable_ligature: bool, freeze_config: dict[str, str] ): """ whether to enable ligatures and freeze font features """ freeze_feature( font=font, calt=enable_ligature, moving_rules=["ss03", "ss07", "ss08"], config=freeze_config, ) def instantiate_cn_var(f: str, base_dir: str, output_dir: str): run( f"ftcli converter vf2i -out {output_dir} {joinPaths(base_dir, f)}", log=True, ) def optimize_cn_base(f: str, base_dir: str): font_path = joinPaths(base_dir, f) print(f"✨ Optimize {font_path}") run(f"ftcli ttf fix-contours {font_path}") run(f"ftcli ttf remove-overlaps {font_path}") run( f"ftcli utils del-table -t kern -t GPOS {font_path}", ) def instantiate_cn_base(cn_variable_dir: str, cn_static_dir: str, pool_size: int): print("=========================================") print("Instantiating CN Base font, be patient...") print("=========================================") run_build( pool_size=pool_size, fn=partial( instantiate_cn_var, base_dir=cn_variable_dir, output_dir=cn_static_dir ), dir=cn_variable_dir, ) run_build( pool_size=pool_size, fn=partial(optimize_cn_base, base_dir=cn_static_dir), dir=cn_static_dir, ) def parse_style_name(style_name_compact: str, skip_subfamily_list: list[str]): is_italic = style_name_compact.endswith("Italic") _style_name = style_name_compact if is_italic and style_name_compact[0] != "I": _style_name = style_name_compact[:-6] + " Italic" if style_name_compact in skip_subfamily_list: return "", _style_name, _style_name, True else: return ( " " + style_name_compact.replace("Italic", ""), "Italic" if is_italic else "Regular", _style_name, False, ) def fix_cn_cv(font: TTFont): gsub_table = font["GSUB"].table config = { "cv96": ["quoteleft", "quoteright", "quotedblleft", "quotedblright"], "cv97": ["ellipsis"], "cv98": ["emdash"], } for feature_record in gsub_table.FeatureList.FeatureRecord: if feature_record.FeatureTag in config: sub_table = gsub_table.LookupList.Lookup[ feature_record.Feature.LookupListIndex[0] ].SubTable[0] sub_table.mapping = { value: f"{value}.full" for value in config[feature_record.FeatureTag] } def remove_locl(font: TTFont): gsub = font["GSUB"] features_to_remove = [] for feature in gsub.table.FeatureList.FeatureRecord: feature_tag = feature.FeatureTag if feature_tag == "locl": features_to_remove.append(feature) for feature in features_to_remove: gsub.table.FeatureList.FeatureRecord.remove(feature) def drop_mac_names(dir: str): run(f"ftcli name del-mac-names -r {dir}") def get_new_name_from_map(old_name: str, map: dict[str, str]): new_name = map.get(old_name) if not new_name: arr = re.split(r"[\._]", old_name, maxsplit=2) if map.get(arr[0]): new_name = map.get(arr[0]) + old_name[len(arr[0]) :] return new_name def rename_glyph_name( font: TTFont, map: dict[str, str], post_extra_names: bool = True, ): print("Rename glyph names") glyph_names = font.getGlyphOrder() extra_names = font["post"].extraNames modified = False merged_map = { **map, **{ "uni2047.liga": "question_question.liga", "dotlessi": "idotless", "f_f": "f_f.liga", }, } for i, _ in enumerate(glyph_names): old_name = str(glyph_names[i]) new_name = get_new_name_from_map(old_name, merged_map) if not new_name or new_name == old_name: continue # print(f"[Rename] {old_name} -> {new_name}") glyph_names[i] = new_name modified = True if post_extra_names and old_name in extra_names: extra_names[extra_names.index(old_name)] = new_name if modified: font.setGlyphOrder(glyph_names) def get_unique_identifier( font_config: FontConfig, postscript_name: str, narrow: bool = False, ignore_suffix: bool = False, ) -> str: if ignore_suffix: suffix = "" else: suffix = font_config.freeze_config_str if "CN" in postscript_name and narrow: suffix += "Narrow;" if "NF" in postscript_name: nf_ver = font_config.nerd_font["version"] suffix = f"NF{nf_ver};{suffix}" beta_str = f"-{font_config.beta}" if font_config.beta else "" return f"{font_config.version_str}{beta_str};SUBF;{postscript_name};2024;FL830;{suffix}" def change_glyph_width(font: TTFont, match_width: int, target_width: int): font["hhea"].advanceWidthMax = target_width for name in font.getGlyphOrder(): glyph = font["glyf"][name] width, lsb = font["hmtx"][name] if width != match_width: continue if glyph.numberOfContours == 0: font["hmtx"][name] = (target_width, lsb) continue delta = round((target_width - width) / 2) glyph.coordinates.translate((delta, 0)) glyph.xMin, glyph.yMin, glyph.xMax, glyph.yMax = ( glyph.coordinates.calcIntBounds() ) font["hmtx"][name] = (target_width, lsb + delta) def update_font_names( font: TTFont, family_name: str, # NameID 1 style_name: str, # NameID 2 unique_identifier: str, # NameID 3 full_name: str, # NameID 4 version_str: str, # NameID 5 postscript_name: str, # NameID 6 is_skip_subfamily: bool, preferred_family_name: str = None, # NameID 16 preferred_style_name: str = None, # NameID 17 ): set_font_name(font, family_name, 1) set_font_name(font, style_name, 2) set_font_name(font, unique_identifier, 3) set_font_name(font, full_name, 4) set_font_name(font, version_str, 5) set_font_name(font, postscript_name, 6) if not is_skip_subfamily and preferred_family_name and preferred_style_name: set_font_name(font, preferred_family_name, 16) set_font_name(font, preferred_style_name, 17) def add_gasp(font: TTFont): print("Fix GASP table") gasp = newTable("gasp") gasp.gaspRange = {65535: 15} font["gasp"] = gasp def build_mono(f: str, font_config: FontConfig, build_option: BuildOption): print(f"πŸ‘‰ Minimal version for {f}") source_path = joinPaths(build_option.output_ttf, f) run(f"ftcli fix italic-angle {source_path}") run(f"ftcli fix monospace {source_path}") run(f"ftcli fix strip-names {source_path}") if font_config.debug: run(f"ftcli ttf dehint {source_path}") else: # dehint, remove overlap and fix contours run(f"ftcli ttf fix-contours --silent {source_path}") font = TTFont(source_path) style_compact = f.split("-")[-1].split(".")[0] style_with_prefix_space, style_in_2, style_in_17, is_skip_subfamily = ( parse_style_name( style_name_compact=style_compact, skip_subfamily_list=build_option.skip_subfamily_list, ) ) postscript_name = f"{font_config.family_name_compact}-{style_compact}" update_font_names( font=font, family_name=font_config.family_name + style_with_prefix_space, style_name=style_in_2, full_name=f"{font_config.family_name} {style_in_17}", version_str=font_config.version_str, postscript_name=postscript_name, unique_identifier=get_unique_identifier( font_config=font_config, postscript_name=postscript_name, ), is_skip_subfamily=is_skip_subfamily, preferred_family_name=font_config.family_name, preferred_style_name=style_in_17, ) # https://github.com/ftCLI/FoundryTools-CLI/issues/166#issuecomment-2095433585 if style_with_prefix_space == " Thin": font["OS/2"].usWeightClass = 250 elif style_with_prefix_space == " ExtraLight": font["OS/2"].usWeightClass = 275 handle_ligatures( font=font, enable_ligature=font_config.enable_liga, freeze_config=font_config.feature_freeze, ) verify_glyph_width( font=font, expect_widths=font_config.get_valid_glyph_width_list(), file_name=postscript_name, ) remove(source_path) target_path = joinPaths(build_option.output_ttf, f"{postscript_name}.ttf") font.save(target_path) font.close() # Autohint version print(f"Auto hint {postscript_name}.ttf") run(f"ftcli ttf autohint {target_path} -out {build_option.output_ttf_hinted}") if font_config.ttf_only: return # Woff2 version print(f"Convert {postscript_name}.ttf to WOFF2") run( f"ftcli converter ft2wf {target_path} -out {build_option.output_woff2} -f woff2" ) # OTF version _otf_path = joinPaths( build_option.output_otf, path.basename(target_path).replace(".ttf", ".otf") ) print(f"Convert {postscript_name}.ttf to OTF") run( f"ftcli converter ttf2otf --silent {target_path} -out {build_option.output_otf}" ) if not font_config.debug: print(f"Optimize {postscript_name}.otf") run(f"ftcli otf fix-contours --silent {_otf_path}") run(f"ftcli otf fix-version {_otf_path}") def build_nf_by_prebuild_nerd_font( font_basename: str, font_config: FontConfig, build_option: BuildOption ) -> TTFont: merger = Merger() return merger.merge( [ joinPaths(build_option.ttf_base_dir, font_basename), f"{build_option.src_dir}/MapleMono-NF-Base{'-Mono' if font_config.nerd_font['mono'] else ''}.ttf", ] ) def build_nf_by_font_patcher( font_basename: str, font_config: FontConfig, build_option: BuildOption ) -> TTFont: """ full args: https://github.com/ryanoasis/nerd-fonts?tab=readme-ov-file#font-patcher """ _nf_args = [ font_config.nerd_font["font_forge_bin"], "FontPatcher/font-patcher", "-l", "--careful", "--outputdir", build_option.output_nf, ] + font_config.nerd_font["glyphs"] if font_config.nerd_font["mono"]: _nf_args += ["--mono"] _nf_args += font_config.nerd_font["extra_args"] run(_nf_args + [joinPaths(build_option.ttf_base_dir, font_basename)], log=True) nf_file_name = "NerdFont" if font_config.nerd_font["mono"]: nf_file_name += "Mono" _path = joinPaths( build_option.output_nf, font_basename.replace("-", f"{nf_file_name}-") ) font = TTFont(_path) remove(_path) return font def build_nf( f: str, get_ttfont: Callable[[str, FontConfig, BuildOption], TTFont], font_config: FontConfig, build_option: BuildOption, ): print(f"πŸ‘‰ NerdFont version for {f}") makedirs(build_option.output_nf, exist_ok=True) nf_font = get_ttfont(f, font_config, build_option) # format font name style_compact_nf = f.split("-")[-1].split(".")[0] style_nf_with_prefix_space, style_in_2, style_in_17, is_skip_sufamily = ( parse_style_name( style_name_compact=style_compact_nf, skip_subfamily_list=build_option.skip_subfamily_list, ) ) postscript_name = f"{font_config.family_name_compact}-NF-{style_compact_nf}" update_font_names( font=nf_font, family_name=f"{font_config.family_name} NF{style_nf_with_prefix_space}", style_name=style_in_2, full_name=f"{font_config.family_name} NF {style_in_17}", version_str=font_config.version_str, postscript_name=postscript_name, unique_identifier=get_unique_identifier( font_config=font_config, postscript_name=postscript_name, ), is_skip_subfamily=is_skip_sufamily, preferred_family_name=f"{font_config.family_name} NF", preferred_style_name=style_in_17, ) verify_glyph_width( font=nf_font, expect_widths=font_config.get_valid_glyph_width_list(), file_name=postscript_name, ) target_path = joinPaths( build_option.output_nf, f"{postscript_name}.ttf", ) nf_font.save(target_path) nf_font.close() def build_cn(f: str, font_config: FontConfig, build_option: BuildOption): style_compact_cn = f.split("-")[-1].split(".")[0] print(f"πŸ‘‰ {build_option.cn_suffix_compact} version for {f}") merger = Merger() cn_font = merger.merge( [ joinPaths(build_option.cn_base_font_dir, f), joinPaths( build_option.cn_static_dir, f"MapleMonoCN-{style_compact_cn}.ttf" ), ] ) style_cn_with_prefix_space, style_in_2, style_in_17, is_skip_subfamily = ( parse_style_name( style_name_compact=style_compact_cn, skip_subfamily_list=build_option.skip_subfamily_list, ) ) postscript_name = f"{font_config.family_name_compact}-{build_option.cn_suffix_compact}-{style_compact_cn}" update_font_names( font=cn_font, family_name=f"{font_config.family_name} {build_option.cn_suffix}{style_cn_with_prefix_space}", style_name=style_in_2, full_name=f"{font_config.family_name} {build_option.cn_suffix} {style_in_17}", version_str=font_config.version_str, postscript_name=postscript_name, unique_identifier=get_unique_identifier( font_config=font_config, postscript_name=postscript_name, narrow=font_config.cn["narrow"], ), is_skip_subfamily=is_skip_subfamily, preferred_family_name=f"{font_config.family_name} {build_option.cn_suffix}", preferred_style_name=style_in_17, ) cn_font["OS/2"].xAvgCharWidth = 600 # https://github.com/subframe7536/maple-font/issues/188 # https://github.com/subframe7536/maple-font/issues/313 fix_cn_cv(cn_font) handle_ligatures( font=cn_font, enable_ligature=font_config.enable_liga, freeze_config=font_config.feature_freeze, ) if font_config.cn["narrow"]: change_glyph_width( font=cn_font, match_width=2 * font_config.glyph_width, target_width=font_config.glyph_width_cn_narrow, ) # https://github.com/subframe7536/maple-font/issues/239 # already removed at merge time # remove_locl(font) if font_config.cn["fix_meta_table"]: # add code page, Latin / Japanese / Simplify Chinese / Traditional Chinese cn_font["OS/2"].ulCodePageRange1 = 1 << 0 | 1 << 17 | 1 << 18 | 1 << 20 # fix meta table, https://learn.microsoft.com/en-us/typography/opentype/spec/meta meta = newTable("meta") meta.data = { "dlng": "Latn, Hans, Hant, Jpan", "slng": "Latn, Hans, Hant, Jpan", } cn_font["meta"] = meta verify_glyph_width( font=cn_font, expect_widths=font_config.get_valid_glyph_width_list(True), file_name=postscript_name, ) target_path = joinPaths( build_option.output_cn, f"{postscript_name}.ttf", ) cn_font.save(target_path) cn_font.close() def wrapped_fn(shutdown_event, fn, filename): if shutdown_event.is_set(): return try: return fn(filename) except Exception as e: shutdown_event.set() raise e def run_build(pool_size: int, fn: Callable, dir: str): files = listdir(dir) if pool_size <= 1: for f in files: fn(f) return with ProcessPoolExecutor(max_workers=pool_size) as executor: futures = { executor.submit(fn, f): f for f in files } for future in as_completed(futures): future.result() def main(): check_ftcli() parsed_args = parse_args() font_config = FontConfig(args=parsed_args) build_option = BuildOption(font_config) build_option.load_cn_dir_and_suffix(font_config.should_build_nf_cn()) if parsed_args.dry: print("font_config:", json.dumps(font_config.__dict__, indent=4)) if not is_ci(): print("build_option:", json.dumps(build_option.__dict__, indent=4)) print("parsed_args:", json.dumps(parsed_args.__dict__, indent=4)) return should_use_cache = parsed_args.cache if not should_use_cache: print("🧹 Clean cache...\n") shutil.rmtree(build_option.output_dir, ignore_errors=True) shutil.rmtree(build_option.output_woff2, ignore_errors=True) makedirs(build_option.output_dir, exist_ok=True) makedirs(build_option.output_variable, exist_ok=True) start_time = time.time() print("🚩 Start building ...\n") # ========================================================================================= # =================================== Build basic ===================================== # ========================================================================================= if not should_use_cache or not build_option.has_cache: input_files = [ joinPaths(build_option.src_dir, "MapleMono-Italic[wght]-VF.ttf"), joinPaths(build_option.src_dir, "MapleMono[wght]-VF.ttf"), ] for input_file in input_files: font = TTFont(input_file) basename = path.basename(input_file) print(f"πŸ‘‰ Variable version for {basename}") # fix auto rename by FontLab rename_glyph_name( font=font, map=match_unicode_names( input_file.replace(".ttf", ".glyphs").replace("-VF", "") ), ) if font_config.apply_fea_file: fea_path = joinPaths( build_option.src_dir, "features/italic.fea" if "Italic" in input_file else "features/regular.fea", ) print(f"Apply feature file [{fea_path}] into [{basename}]") addOpenTypeFeatures( font, fea_path, ) set_font_name( font, get_unique_identifier( font_config=font_config, postscript_name=get_font_name(font, 6), ignore_suffix=True, ), 3, ) verify_glyph_width( font=font, expect_widths=font_config.get_valid_glyph_width_list(), file_name=basename, ) add_gasp(font) font.save( input_file.replace( build_option.src_dir, build_option.output_variable ).replace("-VF", "") ) print("\n✨ Instatiate and optimize fonts...\n") print("Check and optimize variable fonts") if not font_config.debug: run(f"ftcli fix decompose-transformed {build_option.output_variable}") run(f"ftcli fix italic-angle {build_option.output_variable}") run(f"ftcli fix monospace {build_option.output_variable}") print("Instantiate TTF") run( f"ftcli converter vf2i -out {build_option.output_ttf} {build_option.output_variable}" ) _build_mono = partial( build_mono, font_config=font_config, build_option=build_option ) run_build(font_config.pool_size, _build_mono, build_option.output_ttf) drop_mac_names(build_option.output_variable) drop_mac_names(build_option.output_ttf) if not font_config.ttf_only: drop_mac_names(build_option.output_ttf_hinted) drop_mac_names(build_option.output_otf) drop_mac_names(build_option.output_woff2) # ========================================================================================= # ==================================== Build NF ======================================= # ========================================================================================= if font_config.nerd_font["enable"]: use_font_patcher = build_option.should_use_font_patcher(font_config) get_ttfont = ( build_nf_by_font_patcher if use_font_patcher else build_nf_by_prebuild_nerd_font ) _build_fn = partial( build_nf, get_ttfont=get_ttfont, font_config=font_config, build_option=build_option, ) _version = font_config.nerd_font["version"] print( f"\nπŸ”§ Patch Nerd-Font v{_version} using {'Font Patcher' if use_font_patcher else 'prebuild base font'}...\n" ) run_build(font_config.pool_size, _build_fn, build_option.output_ttf) drop_mac_names(build_option.output_ttf) build_option.is_nf_built = True # ========================================================================================= # ==================================== Build CN ======================================= # ========================================================================================= if build_option.should_build_cn(font_config): def _build_cn(): print( f"\nπŸ”Ž Build CN fonts {'with Nerd-Font' if font_config.should_build_nf_cn() else ''}...\n" ) makedirs(build_option.output_cn, exist_ok=True) fn = partial(build_cn, font_config=font_config, build_option=build_option) run_build(font_config.pool_size, fn, build_option.cn_base_font_dir) if font_config.cn["use_hinted"]: print("Auto hinting all glyphs") run(f"ftcli ttf autohint {build_option.output_cn}") drop_mac_names(build_option.cn_base_font_dir) _build_cn() if font_config.use_cn_both and font_config.toggle_nf_cn_config(): build_option.load_cn_dir_and_suffix(font_config.should_build_nf_cn()) _build_cn() build_option.is_cn_built = True # ========================================================================================= # ================================== Write Config ===================================== # ========================================================================================= with open( joinPaths(build_option.output_dir, "build-config.json"), "w", encoding="utf-8" ) as config_file: result = { "version": FONT_VERSION, "family_name": font_config.family_name, "use_hinted": font_config.use_hinted, "ligature": font_config.enable_liga, "feature_freeze": font_config.feature_freeze, "nerd_font": font_config.nerd_font, "cn": font_config.cn, } del result["nerd_font"]["font_forge_bin"] result["nerd_font"]["enable"] = build_option.is_nf_built result["cn"]["enable"] = build_option.is_cn_built config_file.write( json.dumps( result, indent=4, ) ) # ========================================================================================= # ==================================== archive ======================================== # ========================================================================================= if font_config.archive: print("\nπŸš€ archive files...\n") # archive fonts archive_dir_name = "archive" archive_dir = joinPaths(build_option.output_dir, archive_dir_name) makedirs(archive_dir, exist_ok=True) # archive fonts for f in listdir(build_option.output_dir): if f == archive_dir_name or f.endswith(".json"): continue if should_use_cache and f not in ["CN", "NF", "NF-CN"]: continue sha256, zip_file_name_without_ext = compress_folder( family_name_compact=font_config.family_name_compact, suffix="-unhinted" if not font_config.use_hinted else "", source_file_or_dir_path=joinPaths(build_option.output_dir, f), build_config_path=joinPaths( build_option.output_dir, "build-config.json" ), target_parent_dir_path=archive_dir, ) with open( joinPaths(archive_dir, f"{zip_file_name_without_ext}.sha256"), "w", encoding="utf-8", ) as hash_file: hash_file.write(sha256) print(f"πŸ‘‰ archive: {f}") # ========================================================================================= # ===================================== Finish ======================================== # ========================================================================================= if is_ci(): return freeze_str = ( font_config.freeze_config_str if font_config.freeze_config_str != "" else "default config" ) end_time = time.time() date_time_fmt = time.strftime("%H:%M:%S", time.localtime(end_time)) time_diff = end_time - start_time output = joinPaths(getcwd().replace("\\", "/"), build_option.output_dir) print( f"\n🏁 Build finished at {date_time_fmt}, cost {time_diff:.2f} s, family name is {font_config.family_name}, {freeze_str}\n See your fonts in {output}" ) if __name__ == "__main__": main()