From 176ac3c9b24c90ed86c022c4c8d65c78f5994ca4 Mon Sep 17 00:00:00 2001 From: Simon Cozens Date: Wed, 11 Sep 2024 16:28:47 +0100 Subject: [PATCH 1/4] Allow a localMetadata field in the google fonts schema --- Lib/gftools/builder/schema.py | 1 + 1 file changed, 1 insertion(+) diff --git a/Lib/gftools/builder/schema.py b/Lib/gftools/builder/schema.py index 64f2b0e11..1487f1019 100644 --- a/Lib/gftools/builder/schema.py +++ b/Lib/gftools/builder/schema.py @@ -109,5 +109,6 @@ Optional("extraStaticFontmakeArgs"): Str(), Optional("buildSmallCap"): Bool(), Optional("splitItalic"): Bool(), + Optional("localMetadata"): Any(), } ) From 59b829a7e54c72b2825027bae455a6a6e5106f22 Mon Sep 17 00:00:00 2001 From: Simon Cozens Date: Wed, 11 Sep 2024 16:29:38 +0100 Subject: [PATCH 2/4] Add to docs --- docs/gftools-builder/README.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/docs/gftools-builder/README.md b/docs/gftools-builder/README.md index ec22228d8..0d9243169 100644 --- a/docs/gftools-builder/README.md +++ b/docs/gftools-builder/README.md @@ -204,6 +204,9 @@ The build can be customized by adding the following keys to the YAML file: subspace them into separate roman and italic VF files to comply with Google Fonts' specification. Defaults to true. +- `localMetadata`: A field that's ignored so you can put whatever you like in it. + + ## *Really* customizing the build process If the options above aren't enough for you - let's say you want to run your From 802b8c69d23b802fabdd8a2cbe5f583ea84bf22b Mon Sep 17 00:00:00 2001 From: Simon Cozens Date: Mon, 14 Oct 2024 17:46:04 +0100 Subject: [PATCH 3/4] Some general tidying up which seems good to do anyway --- Lib/gftools/builder/operations/__init__.py | 32 +++++++++++++++++-- Lib/gftools/builder/operations/buildOTF.py | 20 ++---------- Lib/gftools/builder/operations/buildTTF.py | 20 ++---------- .../builder/operations/buildVariable.py | 3 +- 4 files changed, 36 insertions(+), 39 deletions(-) diff --git a/Lib/gftools/builder/operations/__init__.py b/Lib/gftools/builder/operations/__init__.py index 0b0330735..2107686d0 100644 --- a/Lib/gftools/builder/operations/__init__.py +++ b/Lib/gftools/builder/operations/__init__.py @@ -131,9 +131,16 @@ def validate(self): class FontmakeOperationBase(OperationBase): + COMPILER_ENV_KEY = "GFTOOLS_COMPILER" + static = False + format = "ttf" + rule = "fontmake --output-path $out -o $format $fontmake_type $in $args" + @property def variables(self): vars = defaultdict(str) + vars["compiler"] = self.compiler + vars["format"] = self.format # buildTTF -> ttf, etc. for k, v in self.original.items(): if k != "needs": vars[k] = v @@ -144,11 +151,32 @@ def variables(self): vars["fontmake_type"] = "-m" elif self.first_source.is_ufo: vars["fontmake_type"] = "-u" - if "--verbose" not in vars["fontmake_args"]: - vars["fontmake_args"] += " --verbose WARNING " + if "--verbose" not in vars["args"]: + vars["args"] += " --verbose WARNING " return vars + def validate(self): + if self.static: + if not self.first_source.exists(): + # We can't check this file (assume it was generated as part of the + # build), so user is on their own. + return + if ( + self.first_source.is_glyphs + and len(self.first_source.gsfont.masters) > 1 + ): + raise ValueError( + f"Cannot build a static font from {self.first_source.path}" + ) + if ( + self.first_source.designspace + and len(self.first_source.designspace.sources) > 1 + ): + raise ValueError( + f"Cannot build a static font from {self.first_source.path}" + ) + known_operations = {} diff --git a/Lib/gftools/builder/operations/buildOTF.py b/Lib/gftools/builder/operations/buildOTF.py index 84cb2fc57..0831b312c 100644 --- a/Lib/gftools/builder/operations/buildOTF.py +++ b/Lib/gftools/builder/operations/buildOTF.py @@ -3,21 +3,5 @@ class BuildOTF(FontmakeOperationBase): description = "Build a OTF from a source file" - rule = "fontmake --output-path $out -o otf $fontmake_type $in $args" - - def validate(self): - if not self.first_source.exists(): - # We can't check this file (assume it was generated as part of the - # build), so user is on their own. - return - if self.first_source.is_glyphs and len(self.first_source.gsfont.masters) > 1: - raise ValueError( - f"Cannot build a static font from {self.first_source.path}" - ) - if ( - self.first_source.designspace - and len(self.first_source.designspace.sources) > 1 - ): - raise ValueError( - f"Cannot build a static font from {self.first_source.path}" - ) + format = "otf" + static = True diff --git a/Lib/gftools/builder/operations/buildTTF.py b/Lib/gftools/builder/operations/buildTTF.py index 8dc3d727e..f86223b9d 100644 --- a/Lib/gftools/builder/operations/buildTTF.py +++ b/Lib/gftools/builder/operations/buildTTF.py @@ -3,21 +3,5 @@ class BuildTTF(FontmakeOperationBase): description = "Build a TTF from a source file" - rule = "fontmake --output-path $out -o ttf $fontmake_type $in $args" - - def validate(self): - if not self.first_source.exists(): - # We can't check this file (assume it was generated as part of the - # build), so user is on their own. - return - if self.first_source.is_glyphs and len(self.first_source.gsfont.masters) > 1: - raise ValueError( - f"Cannot build a static font from {self.first_source.path}" - ) - if ( - self.first_source.designspace - and len(self.first_source.designspace.sources) > 1 - ): - raise ValueError( - f"Cannot build a static font from {self.first_source.path}" - ) + static = True + format = "ttf" diff --git a/Lib/gftools/builder/operations/buildVariable.py b/Lib/gftools/builder/operations/buildVariable.py index 63e1be7b4..1629a99e1 100644 --- a/Lib/gftools/builder/operations/buildVariable.py +++ b/Lib/gftools/builder/operations/buildVariable.py @@ -3,4 +3,5 @@ class BuildVariable(FontmakeOperationBase): description = "Build a variable font from a source file" - rule = "fontmake --output-path $out -o variable $fontmake_type $in $args" + format = "variable" + static = False From 18a8f6c33f5c462c672d5d371dc61f9424cb21bc Mon Sep 17 00:00:00 2001 From: Simon Cozens Date: Mon, 14 Oct 2024 17:54:59 +0100 Subject: [PATCH 4/4] Allow switching compilers --- Lib/gftools/builder/operations/__init__.py | 32 +++++++++++++++++++++- 1 file changed, 31 insertions(+), 1 deletion(-) diff --git a/Lib/gftools/builder/operations/__init__.py b/Lib/gftools/builder/operations/__init__.py index 2107686d0..8cb06db65 100644 --- a/Lib/gftools/builder/operations/__init__.py +++ b/Lib/gftools/builder/operations/__init__.py @@ -5,6 +5,7 @@ import pkgutil import importlib import inspect +import logging import sys from os.path import dirname from tempfile import NamedTemporaryFile @@ -12,6 +13,17 @@ from gftools.builder.file import File from gftools.utils import shell_quote +log = logging.getLogger(__name__) + + +# Grossness to allow class properties in Python +class classproperty: + def __init__(self, func): + self.fget = func + + def __get__(self, instance, owner): + return self.fget(owner) + @dataclass class OperationBase: @@ -134,7 +146,15 @@ class FontmakeOperationBase(OperationBase): COMPILER_ENV_KEY = "GFTOOLS_COMPILER" static = False format = "ttf" - rule = "fontmake --output-path $out -o $format $fontmake_type $in $args" + + @classproperty + def rule(self): + if self.compiler == "fontmake": + return "fontmake --output-path $out -o $format $fontmake_type $in $args" + else: + # Construct a ninja rule which correctly builds the requested font type + # (Good luck with fontmake args...) + return "fontc -o $out $in $args" @property def variables(self): @@ -154,8 +174,18 @@ def variables(self): if "--verbose" not in vars["args"]: vars["args"] += " --verbose WARNING " + if "--filter" in vars["args"] and self.compiler == "fontc": + log.warning("Cannot use --filter with fontc, dropping arguments...") + del vars["args"] + return vars + @property + def compiler(self): + return self.original.get( + "compiler", os.environ.get(self.COMPILER_ENV_KEY, "fontmake") + ) + def validate(self): if self.static: if not self.first_source.exists():