Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

STAT table errors in variable fonts #2391

Open
Lorp opened this issue Mar 13, 2020 · 55 comments · Fixed by googlefonts/comfortaa#22
Open

STAT table errors in variable fonts #2391

Lorp opened this issue Mar 13, 2020 · 55 comments · Fixed by googlefonts/comfortaa#22

Comments

@Lorp
Copy link

Lorp commented Mar 13, 2020

Since adding STAT table handling to Samsa, I’ve been finding apparent errors in several variable fonts including IBM Plex Sans and Work Sans. (These are the only two Google fonts I have tested, so I can only assume the problem is widespread.)

The issue is that the STAT in the Italic VF repeats the string “Italic” in all the weight entries. Here is a screengrab from Work Sans Italic (WorkSans-Italic-VariableFont_wght.ttf) in Samsa.

Screenshot 2020-03-13 at 20 59 10

This leads to incorrect “composed names” such as Light Italic Italic.

Screenshot 2020-03-13 at 21 00 42

In fact, each of these strings is supposed to refer to a single axis, so the list of strings in Work Sans Italic’s Weight STAT should be: Thin, ExtraLight, Light, Regular, Medium, SemiBold, Bold, ExtraBold, Black. Only then can apps build menus where users select meaningful instance names from one axis at a time.

Work Sans Roman (WorkSans-VariableFont_wght.ttf) has a less important issue where the string “Roman” in the Italic axis should be marked elidable (flag = 2). When fixed, we would see composed names of the form “SemiBold” rather than “SemiBold Roman”. Finally, “Regular” should also be marked elidable in both Roman and Italic.

Screenshot 2020-03-13 at 21 09 02

Try in Samsa

@Lorp Lorp changed the title STAT table errors STAT table errors in variable fonts Mar 14, 2020
@thlinard
Copy link
Contributor

thlinard commented Mar 16, 2020

Hi,

I tested all the VF of this repo, here are the results:

  • Changa: no AxisValue.
  • Comfortaa: Regular lacking Flags value="2" (non elidable).
  • Crimson Pro Italic: "Italic" in Weight axis; namerecord nameID 259 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Crimson Pro: Regular non elidable, Roman non elidable.
  • Dancing Script: Regular non elidable.
  • Dosis: no AxisValue.
  • EB Garamond Italic: "Italic" in Weight axis; namerecord nameID 257 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • EB Garamond: Regular non elidable, Roman non elidable.
  • El Messiri: no AxisValue.
  • Exo2 Italic: no AxisValue.
  • Exo2: no AxisValue.
  • Faustina Italic: "Italic" in Weight axis; namerecord nameID 257 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Faustina: Regular non elidable, Roman non elidable.
  • Fira Code: no AxisValue.
  • Hepta Slab: Regular non elidable.
  • Inter: no AxisValue.
  • Josefin Sans Italic: "Italic" in Weight axis; namerecord nameID 259 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Josefin Sans: Regular non elidable, Roman non elidable.
  • Jura: no AxisValue.
  • Karla Italic: "Italic" in Weight axis; namerecord nameID 259 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Kreon: Regular non elidable.
  • Lemonada: no AxisValue.
  • Lora Italic: "Italic" in Weight axis; namerecord nameID 257 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Lora: Regular non elidable, Roman non elidable.
  • Manuale Italic: "Italic" in Weight axis; namerecord nameID 257 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Manuale: Regular non elidable, Roman non elidable.
  • Markazi Text: Regular non elidable.
  • Maven Pro: Regular non elidable.
  • Mohave Italic: "Italic" in Weight axis; namerecord nameID 258 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Mohave: Regular non elidable, Roman non elidable.
  • Mulish Italic: "Italic" in Weight axis; namerecord nameID 259 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Mulish: Regular non elidable, Roman non elidable.
  • Orbitron: no AxisValue.
  • Oswald: Regular non elidable.
  • Playfair Display Italic: "Italic" in Weight axis; namerecord nameID 257 is "Italic", should be "Regular", with Flags value="2" in the STAT table. namerecord nameID 260, 261, 262 are duplicate.
  • Playfair Display: Regular non elidable, Roman non elidable. namerecord nameID 260, 261, 262 are duplicate.
  • Podkova: Regular non elidable.
  • Public Sans Italic: no AxisValue.
  • Public Sans: no AxisValue.
  • Roboto Slab: Regular non elidable.
  • Rokkit: Regular non elidable.
  • Rosario Italic: "Italic" in Weight axis; namerecord nameID 258 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Rosario: Regular non elidable, Roman non elidable.
  • Ruda: no AxisValue.
  • Spartan: Regular non elidable.
  • Work Sans Italic: "Italic" in Weight axis; namerecord nameID 260 is "Italic", should be "Regular", with Flags value="2" in the STAT table.
  • Work Sans: Regular non elidable, Roman non elidable.
  • Yanone Kaffeesatz: Regular non elidable.

Edit:
Corrected:

Changa (PR #3667)
Comfortaa (PR #3631)
Crimson Pro (PR #3632)
Dancing Script (PR #3658)
Dosis (PR #3668)
EB Garamond (PR #3633)
El Messiri (PR #3669)
Exo (PR #3635)
Exo 2 (PR #3636)
Faustina (PR #3806)
Fira Code (PR #3786)
Hepta Slab (PR #3785)
Karla and Karla Italic (PR #2770)
Josefin Sans (PR #3664)
Jura (PR #3639)
Kreon (PR #3657)
Lemonada (PR #3640)
Lora (PR #3641)
Literata and Literata Italic (PR #2580, #3805)
Manuale (PR #3643)
Markazi Text (PR #3644)
Maven Pro (PR #3660)
Mohave (PR #3645)
Mulish (PR #3762)
Orbitron (PR #3670)
Oswald (PR #3661)
Playfair Display (PR #3649)
Podkova (PR #3662)
Public Sans (PR #3650)
Quicksand (PR #3410)
Roboto Slab (PR #3655)
Rokkit (PR #3663)
Rosario (PR #3651, #3764)
Ruda (PR #3652)
Spartan (PR #3601)
Vollkorn and Vollkorn Italic (PR #2960)
Work Sans (PR #3586 and #3654)
Yanone Kaffeesatz (PR #3665)

The issues can be summarized as follows:

  1. Flags value="0" instead of value="2". @thundernixon had already proposed an improvement to FontBakery over a year ago, that @m4rc1e should look into on "Monday" but it seems there was never a follow-up.
  2. Italic in Weight axis: it's so systematic that it seems a process to be revised.
  3. No AxisValue or no Axis: may be something to check in FontBakery?
  4. Duplicate entries: same.

@Lorp
Copy link
Author

Lorp commented Mar 16, 2020

What a great report, @thlinard!

@Lorp
Copy link
Author

Lorp commented Mar 16, 2020

To optimize fixes for the above issues, it may be helpful to look at the STAT tables from the Source variable fonts: Source Serif, Source Sans, Source Code by @adobe-fonts.

These define all the named Weights using Format 2, which includes not only a nominal value but also a range. Using ranges, STAT axisValue records can cover an entire axis with meaningful names. Such ranges could be automatically generated for the Google VFs, and will perhaps help the fonts perform better in a wider range of apps.

The following table shows the AxisValues for the Weight designAxis in SourceSerifVariable-Roman.ttf.

valueName format value min max linked
ExtraLight 2 200 200 250
Light 2 300 250 350
(Regular) 2 400 350 500
Semibold 2 600 500 650
Bold 2 700 650 800
Black 2 900 800 900
(Regular) 3 400 700

Also worth considering and, as the table shows, implemented in the Source* fonts, is the use of Format 3 axisValues for an additional elidable Regular (400) record in Weight that declares Bold (700) as stylistically related.

Finally, it would be good to represent the ital axis location for VF families that have an italic. Adobe implements this by:

  • an additional designAxis, tag ital, name “Italic” in both upright and italic VFs
  • an elidable axisValue record for ital in the upright VF named “Roman”, value 0.0, and linked using Format 3 to value 1.0
  • a non-elidable axisValue record for ital in the italic VF named “Italic”, value 1.0, Format 1

@m4rc1e
Copy link
Collaborator

m4rc1e commented Mar 17, 2020

Our STAT setups work in MS Word and All OS X Pages, Slide etc seem ok. Our setup was made before Indesign supported VFs.

@thlinard Thanks for this but I'm questioning your analysis. If I just pick Rosario Italic nameID 258 shouldn't be "Regular". NameID 258 clearly points to the Italic instance as referenced by the fvar table.

Your point about Flags value=2 is correct. We'll do a pass over all VFs this year.

We found the STAT to be a bit of a nightmare. Microsoft applications currently don't support STAT v2 which means Office apps cannot support multiple axes vfs yet. I also question whether Indesign supports vfs correctly? I don't have Indesign on my current machine so if someone could test Source Sans VF I'd be very happy.

Great work from both of you and thank you. Our approach has been to push variable fonts out before applications have supported them. We'll definitely revisit this issue since it seems applications support is better

@thlinard
Copy link
Contributor

thlinard commented Mar 17, 2020

Hi @m4rc1e

You're right to question my analysis. 😀

Let's take your example, with Rosario Italic.

nameID 258 is used by an AxisValue:

AxisIndex value="0"
ValueNameID value="258"
Value value="400.0"

AxisIndex value="0" refers to Axis index="0":
AxisTag value="wght"

and certainly not to Axis index="1":
AxisTag value="ital"

An AxisValue is defined for the Italic axis:

AxisIndex value="1"
ValueNameID value="262"

nameID 262 is "Italic" (same as nameID 258)

So, in this configuration, you put the name "Italic" in the Weight axis, while "Italic" is already in the Italic axis. When the name is composed, you get:
Rosario Italic Italic
As you can see in Samsa.

This is for theory. In practice, my Word 2019 for macOS is unable to manage a VF, so I tested with the Apple and Adobe applications (InDesign 2020). In particular I tried Playfair Display, so obviously wrong: no duplicate appeared. So two hypotheses:

  • Either all these applications show remarkable resilience to malformed fonts.
  • Or these applications don't use the STAT table at all, but the fvar table.

Tests with ttx and fake namerecords showed me that the second hypothesis was the right one.

So, I would rephrase the issue 2 as follows:

  • Italic in Weight axis: keep the existing namerecord for the fvar table, but create (following the Adobe Source fonts example) new namerecord entries, without "Italic" in the name, for the STAT table.

@m4rc1e
Copy link
Collaborator

m4rc1e commented Mar 17, 2020

Thanks for your testing and response.

I'm currently wrapping my head around the following issue. I'll try and work on both of these at the same time since they're related.

We'll try our best to make our fonts comply to the MS spec but we cannot guarantee the fonts will work correctly in all apps. As you've witnessed it is currently a free for all, hence why we supply static fonts for app use.

@thlinard
Copy link
Contributor

Yes, very interesting discussion, indeed. But it'll be hard to imagine what an UI using STAT would be like without some fonts with a correct STAT implementation.

@Lorp
Copy link
Author

Lorp commented Mar 18, 2020

For reference, here is the formatted STAT data for Source Serif Regular and Italic, taken from TTX dumps of the STAT tables. The only difference is the entry that asserts each font’s location on the ital/Italic designAxis. Elidable names are in parentheses, following FontLab notation.

SourceSerifVariable-Roman.ttf

  • Two designAxes: wght (Weight) and ital (Italic).
  • ElidedFallbackNameID points to Regular.
axis valueName format value min max linked
wght(0) ExtraLight 2 200 200 250
wght(0) Light 2 300 250 350
wght(0) (Regular) 2 400 350 500
wght(0) Semibold 2 600 500 650
wght(0) Bold 2 700 650 800
wght(0) Black 2 900 800 900
wght(0) (Regular) 3 400 700
ital(1) (Roman) 3 0.0 1.0

SourceSerifVariable-Italic.ttf

  • Two designAxes: wght (Weight) and ital (Italic).
  • ElidedFallbackNameID points to Italic.
axis valueName format value min max linked
wght(0) ExtraLight 2 200 200 250
wght(0) Light 2 300 250 350
wght(0) (Regular) 2 400 350 500
wght(0) Semibold 2 600 500 650
wght(0) Bold 2 700 650 800
wght(0) Black 2 900 800 900
wght(0) (Regular) 3 400 700
ital(1) Italic 1 1.0

@thlinard
Copy link
Contributor

thlinard commented Apr 23, 2020

Update on the latest VF released:

  • Exo and Exo Italic: no AxisValue.
  • Jost: no AxisValue.
  • Manrope: no AxisValue.
  • Inconsolata: the first VF Google Fonts with a correct STAT table!! There is different entries in the name table for fvar and for STAT, and Flags value="2" for the Regular. No range provided, which is a shame for the user experience, but it's a big improvement nonetheless.

According to https://github.com/googlefonts/gf-docs/tree/master/Spec#stat-table this good work is due to Marc. 😀

Edit:
Corrected:

Jost (PR #3638)
Manrope (PR #3642)

@Lorp
Copy link
Author

Lorp commented Apr 23, 2020

Great to hear that Inconsolata is fixed – the build of April 1 looks perfect, except for the addition of Format 2 ranges which might be nice as @thlinard mentions. Nice work, @m4rc1e.

Since I last posted on this thread, Samsa has been updated to handle Format 4 axisValueTables, so the old and new versions of Inconsolata can be usefully compared.

@tiroj
Copy link

tiroj commented Apr 23, 2020

On the subject of range definition in the STAT table, this format was initially targeted at the opsz axis, to enable the STAT table to define the range of intermediate sizes at which an opsz named instance should be used, since this isn't derivable from the names of the instances and is design-specific.

I noted when examining the STAT table in Adobe's Source Serif Pro that it defines ranges for the wght axis, and I'm trying to figure out whether there‘s any point to this. Unlike opsz use ranges, which are arbitrary and design-specific, we have weight ranges implied in the CSS weight class, and it seems to me that the range across which to use a nominal weight instance can be algorithmically determined by simply using whichever instance is closest to the nominal weight value. Am I missing something?

@Lorp
Copy link
Author

Lorp commented Apr 24, 2020

BTW @m4rc1e is there any reason that the variable versions of Source Serif, Source Sans and Source Code are not on Google Fonts?

@m4rc1e
Copy link
Collaborator

m4rc1e commented Apr 24, 2020

@Lorp lack of time mainly. It takes a while to diff a VF against statics to ensure there are no regressions.

@thlinard
Copy link
Contributor

Hi @tiroj

Well, I presume the same result could be achieved by algorithmic methods. The goal here is to provide a more meaningful composed name.

For example, for wght=600 and for wght=539, "Source Serif Variable Semibold" and "Source Serif Variable Semibold[529]" are seen as more desirable outcomes (for a UI based on the STAT table) than "Source Serif Variable Semibold" and "Source Serif Variable Regular" ("Regular" because, for wght=529, if no range is defined, the "composed name" returns a default value).

Thus, the ranges provide better consistency, better information for the end user. Furthermore, I know many designers, who are neither font developers nor CSS developers, who don't know what a weight expressed on a scale of 1 to 1000 corresponds to. Adding the familiar name of a weight next to the precise value is a good landmark for them.

@tiroj
Copy link

tiroj commented Apr 24, 2020

But do you need ranges to obtain those results? I'm questioning whether a format 2 axis value table is necessary for the purposes you describe, or if a format 1 table would always be adequate for the wght axis (and probably the width axis), since the axis scales are so clearly defined and not design-specific in the same way that opsz is. If the wght scale were not mapped to the CSS weight class, then it would make sense to have ranges in the STAT table, but we have avar mappings to fit design space weight progression to the linear weight class, so ranges in the STAT table seem redundant.

@tiroj
Copy link

tiroj commented Apr 24, 2020

For example, for wght=600 and for wght=539, "Source Serif Variable Semibold" and "Source Serif Variable Semibold[529]" are seen as more desirable outcomes (for a UI based on the STAT table) than "Source Serif Variable Semibold" and "Source Serif Variable Regular" ("Regular" because, for wght=529, if no range is defined, the "composed name" returns a default value).

But in that case, I'd argue that the default value should actually be ‘Source Serif Variable Medium 529’, because Medium is the default for the nearest standard weight class value (500). The STAT table axis value record is providing a record of relationships between named instances, but composed names for arbitrary instances on standardised scales could derive from the scales rather than the, as in this case, incomplete set of named instances.

Thus, the ranges provide better consistency, better information for the end user. Furthermore, I know many designers, who are neither font developers nor CSS developers, who don't know what a weight expressed on a scale of 1 to 1000 corresponds to. Adding the familiar name of a weight next to the precise value is a good landmark for them.

Right, but assigning names to the precise weight class values doesn't need ranges to be specified. A format 1 axis record table does the same thing.

Let me put the question another way:

Is there ever a case in which it would make sense for the range of covered by the nominal Regular weight instance to be something other than from halfway between it and the next lightest instance to halfway between it and the next heaviest instance?

If the answer is yes, doesn't that imply that the avar mappings for wght are incorrect?

@Lorp
Copy link
Author

Lorp commented Apr 24, 2020

Is there ever a case in which it would make sense for the range of covered by the nominal Regular weight instance to be something other than from halfway between it and the next lightest instance to halfway between it and the next heaviest instance?

In principle, absolutely (but it doesn’t imply anything is wrong with avar). When a non-linear formula is expressed in piecewise linear steps, the “half way” points between the mappings must also be mapped non-linearly to be mathematically correct. Aesthetic or pragmatic judgement might take precedence over the linear or non-linear mid-points in any case.

Regarding “nearest value mapping”, it would be a useful heuristic for composing names in the absence of Format 2 ranges.

@thlinard
Copy link
Contributor

This debate reminds me a bit of the debate "nameID 16-17 vs. nameID 21-22". Well, Adobe chooses for Source Serif Variable:

  • Regular: RangeMinValue value = "350.0", RangeMaxValue value = "500.0"
  • Semibold: RangeMinValue value = "500.0", RangeMaxValue value = "650.0"

We would have wished that Adobe (or Frank Grießhammer) used their freedom a little less and conform to a predefined set of weights (the CSS class). But as Laurence pointed out, aesthetic or pragmatic reasons can explain it.

But I agree, more standardization would allow the automation of ranges, without having to specify them.

@tiroj
Copy link

tiroj commented Apr 25, 2020

We would have wished that Adobe (or Frank Grießhammer) used their freedom a little less and conform to a predefined set of weights (the CSS class).

See, my reading of the Source Serif Pro variable STAT table is that Adobe did conform to the predefined set of weights; they just opted not to define a named instance for every weight in the set. Based on the format 2 wght ranges specified, the implied but unnamed Medium weight is at 500. So what the STAT table is saying is that non-variable environments should use the Regular for everything less than where Medium would be, and Semibold for everything greater than where Medium would be. There's nothing in that which couldn't be derived from using format 1 data and selecting the nearest named instance to the target weight.

I understand Laurence's point about the halfway points between non-linear mappings needing to be mapped non-linearly, but that's not what the STAT table ranges do. The STAT format 2 axis record provides information about which named instance to use for a given range when not using interpolation. It's targeting non-variable environments in which named instances behave like static fonts, and the format 2 axis record was defined to provide size range information for optical size (directly akin to the abandoned OS/2 version 5 size fields).


BTW, it strikes me that not only shouldn't it be necessary to define ranges for wght, it probably shouldn't be necessary to define most named instances on the weight axis, because any of the standard weight class instances could be synthesised and named as a heuristic. As it is, named instances give font makers a choice about how many virtual static fonts to present to users in non-variable environments. Some, as in this case, might decide not to include all intermediate weights e.g. Medium between the named Regular and Semibold, but it seems to me that this could be entirely client-side, and perhaps up to the user which nominal weight variants to expose in font menus.


Speaking of non-linear weight progression, linear mapping, and named instances, I can imagine that not all font makers approach this in the same way, and perhaps best practice recommendations should be documented. There are various tools available to calculate steps between weight design masters according to different equations as may be favoured by font makers. My approach when using such tools is to include all the steps implied by the CSS weight class, regardless of how many of those steps I intend to make named instances. So, if I have a Regular design master and a Black design master, I am going to calculate weight progression for four intermediate weights (Medium, Semibold, Bold, Extrabold) even if, for example, I only plan to define the Semibold and Bold as named instances. Does everyone do this, or do some people calculate only the number of intermediate steps for which they plan to have named instances?

@Lorp
Copy link
Author

Lorp commented Apr 27, 2020

@tiroj: Your description of STAT Format 2 is interesting, and helps me understand how older apps can make best use of variable fonts, and also how static fonts such as Sitka can make best use of STAT. One clarification would be useful: are you saying that the process in Windows non-variable environments is to use STAT to compose the subfamily name (based on the point size, weight and style), then to look up that name in the fvar Named Instances? Or are you using the term “Named Instance” more loosely to include instances “named” by STAT?

[From the spec] A format 2 table can be used if a given name is associated with a particular axis value, but is also associated with a range of values. For example, in a family that supports optical size variations, “Subhead” may be used in relation to a range of sizes. The rangeMinValue and rangeMaxValue fields are used to define that range. In a variable font, a named instance has specific coordinates for each axis. The nominalValue field allows some specific, nominal value to be associated with a name, to align with the named instances defined in the font variations table, while the rangeMinValue and rangeMaxValue fields allow the same name also to be associated with a range of axis values.

My reading of the STAT table Format 2 spec leads to these two conclusions:

  • a) the nominal value is a kind of default value rather than an avar-like remapping of a range;
  • b) ranges may prove useful in fully variable environments.

@tiroj
Copy link

tiroj commented Apr 28, 2020

I mean Named Instance in the fvar sense; I'll try to remember to capitalise it like this to avoid ambiguity. A Named Instance has a specific axis location in the design space, but by specifying a range using a format 2 STAT axis record, one can associate a Named Instance with a larger area of the design space. But that seems to me primarily relevant to non-variable environments, because in a variable environment one generally doesn't want e.g. the Subhead Named Instance to be used for every point size between e.g. 16pt and 22pt, but rather to use interpolated locations of the design space specific to the actual size.

It is possible, of course, that someone might want to use a variable font in a variable environment in a way that is compatible with use in a non-variable environment, or compatible with static fonts, and having format 2 range axis records for opsz would be relevant to that. But, again, that's because opsz Named Instances are not mapped to a standard scale, while wght Named Instances are. So I'm still not convinced there is ever a need to define range axis records for wght. I'm not saying it is wrong to do so, only that I wouldn't want font makers to look at something like Source Serif Pro and conclude from it that this is something they should be doing.

@tiroj
Copy link

tiroj commented Apr 28, 2020

cc. @PeterConstable who might be able to confirm or contradict my understanding, and provide insights into the design and intention of the format 2 axis record.

@davelab6
Copy link
Member

davelab6 commented Apr 28, 2020 via email

@thlinard
Copy link
Contributor

thlinard commented Apr 28, 2020

I tested all the VF of this repo, here are the results:

Where is the script to reproduce the test? :)

Hi @davelab6

You're welcome! 😆 It was a very manual test (ttx the fonts, and I simply looked at the result with my very eyes), so, sorry, no script.

But I can give the specifications of such a script:

@PeterConstable
Copy link

PeterConstable commented Apr 29, 2020

Also worth considering and, as the table shows, implemented in the Source* fonts, is the use of Format 3 axisValues for an additional elidable Regular (400) record in Weight that declares Bold (700) as stylistically related.

IIRC, Adobe brought up the idea of an additional AxisValue format that defines both a range and a style-linked value. I forget now if it hasn't been pursued because they just didn't pursue it, or because there was an issue that never got resolved. (Possible issue: What would it mean to link a range to a point value?)

Or would it make sense for the OT spec to explicitly sanction what Adobe has done, including both format 2 and format 3 entries for wght=400? (Explicit sanction could clue in app developers to anticipate this.)

@PeterConstable
Copy link

PeterConstable commented Apr 29, 2020

On the subject of range definition in the STAT table, this format was initially targeted at the opsz axis, to enable the STAT table to define the range of intermediate sizes at which an opsz named instance should be used, since this isn't derivable from the names of the instances and is design-specific.

IIRC, format 3 was originally designed having in mind opsz in static-instance fonts, not in variable fonts. In a variable font with an opsz axis, opsz ranges shouldn't matter since there is an instance available for every text point size. (I suppose there could be an app that might support auto-selection of optical size variants but without support for arbitrary instances, so the opsz ranges in the variable font would allow it to identify the opsz name to use in filtering named instances. But hopefully apps wouldn't get designed that way.)

Keep in mind that STAT was intended for use in variable or non-variable fonts; indeed, it was first conceived (at Adobe) before work on OT variations began.

@PeterConstable
Copy link

OK, read through the thread.

I think it's fair to say that the original design intent for format 2 ranges was for non-variable scenarios. That first of all meant non-variable fonts, to supersede the OS/2 format 5 usLowerOpticalPointSize and usUpperOpticalPointSize fields and the 'size' feature. (I'm sure neither of those would have even been considered if a STAT table had existed.) But clearly the ranges can also be useful for a variable font in a non-variable app.

Ranges may also be useful for variable fonts even in apps that support variations. For instance, consider a font fallback scenario: the content specifies weight=500 but the specified font family isn't available and fallback is being used: what instance from the fallback font family is selected? You could infer that algorithmically using a nearest numeric match. But designer-provided range values are another way that could be used.

Generalizing that, consider what an API might need to do: an API might allow a client to request matches for some vector of axis values: that API could use nearest numeric matching, or it could use font-provided ranges. Or it could use font-provided ranges if available and use numeric matching if not—IIRC, that's what Dwyane Robinson did in Windows DirectWrite.

So, is format 2 needed for wght? No, probably not: an app or API that pays attention to STAT axis value tables probably also implemented some behaviour when STAT isn't present. Is it useful? Maybe, if designer-provided values can give better results than a numeric algorithm.

Btw, 'avar' was mentioned here. I would not expect any app or API that uses a numeric algorithm to evaluate what is happening in an 'avar' table to tune that algorithm: there's no easy way to know what the designer's intent was with the 'avar' table, and there's nothing to suggest it would be worth the investment to even try.

@aaronbell
Copy link
Collaborator

aaronbell commented Jul 15, 2021

I've been progressing through the many variable fonts in the library (both live and not) and wanted to give an update on the efforts.

Categories:
Upstream PR Submitted - I've submitted a PR from my repro to the upstream repro with a fix (either updating the existing build process or replacing with an UFR-based one)
Previously fixed - fonts wherein the STAT table issue was resolved prior to my review, but the new version needs to be PR'd to GF
Needs Work - These fonts still need work, or are not ready for PR
Reviewed - Fonts that don't appear to have STAT-related problems
Blocked - League Gothic is not live on Google Fonts, likely due to issues around the necessity of the width axis, so I've skipped it for now.

PR Submitted Already fixed Needs more work Needs new build script Reviewed Blocked
Changa Faustina Josefin Sans Inconsolata Sansita Swashed League Gothic
Comfortaa Fira Code   Reem Kufi Trispace  
Crimson Pro Inter   Cairo Truculenta  
Dosis     Spartan Alegreya  
EB Garamond       Andada  
El Messiri       Antonio  
Epilogue       Archivo  
Exo       Arimo  
Exo 2       Asap  
Grenze Gotisch       Bitter  
Jost       Brygada 1918  
Jura       Cabin  
Lemonada       Caveat  
Lora       Cinzel  
Manrope       Cuprum  
Manuale       Dancing Script  
Markazi Text       Domine  
Mohave       Encode Sans  
Museo Moderno       Georama  
Orbitron       Grandstander  
Playfair Display       Hahmlet  
Public Sans       Handjet  
Rosario       Heebo  
Ruda       Hepta Slab  
Signika       Ibarra Real Nova  
Work Sans       JetBrains Mono  
Besley       Kapakana  
Roboto       Karla  
Saira       Kufam  
Kreon       Kumbh Sans  
        Libre Caslon  
        Libre Franklin  
        Maven Pro  
        Merriweather Sans  
        Newsreader  
        Open Sans  
        Oswald  
        Petrona  
        Podkova  
        Raleway  
        Rasa  
        Recursive  
        Red Hat Mono  
        Red Rose  
        Roboto Mono  
        Roboto Slab  
        Rokkitt  
        Sofia Sans  
        Sora  
        Space Grotesk  
        STIX Two Text  
        Syne  
        Texturina  
        Tourney  
        Varta  
        Vollkorn  
        Yanone Kaffeesatz  
        Yrsa  

@thlinard
Copy link
Contributor

Fantastic job! There is not much left after that.

I see you don't consider a non elidable default value to be an error in the STAT table, is that intended?

@aaronbell
Copy link
Collaborator

@thlinard For the "reviewed" fonts, I primarily looked at them in Mac Word to check if all of the STAT values are showing properly. Many didn't default to the "Regular" location, but I actually expected that to be the case.

The problem in Mac Word is that it appears that if a font only has a master at either end of the designspace (and lacks one at the "Regular" location) that it will never be able to find the "Regular" location, regardless of the 'elidable' state.

That said, you're right that I should make double-sure that they have 'elidable' for the wonderful day in the future when Mac Word works correctly ;)

@aaronbell
Copy link
Collaborator

@thlinard I reviewed all the fonts that didn't default properly and only came up with "Spartan" as lacking the elidable flag on the Regular weight. Are there other fonts you found as problematic?

@thlinard
Copy link
Contributor

@aaronbell To Spartan you could add:

  • Dancing Script
  • Hepta Slab
  • Maven Pro
  • Mulish and Mulish Italic (Samsa doesn't consider them to be truly elidable, probably due to a poor definition of the Italic axis, in format 2)
  • Oswald
  • Podkova
  • Roboto Slab
  • Rokkit
  • Yanone Kaffeesatz

@aaronbell
Copy link
Collaborator

Thanks! Interestingly those fonts did not exhibit any issues in Mac Word, but probably best to update. I’ll add them :)

@m4rc1e m4rc1e closed this as completed Jul 20, 2021
@aaronbell
Copy link
Collaborator

I’m going to keep this open for tracking

@aaronbell aaronbell reopened this Jul 20, 2021
@thlinard
Copy link
Contributor

thlinard commented Aug 5, 2021

@aaronbell 4 VFs are missing in your summary table:

  1. Assistant
  2. Commissioner (for the "Reviewed" column I think: a small issue in Word, but it's probably Word's fault)
  3. Fraunces (Please correct SOFT value for Thin Italic instance undercasetype/Fraunces#259: the Thin Italic fvar instance has <coord axis="SOFT" value="100.0"/> while all other instances, Italic or Regular, have <coord axis="SOFT" value="0.0"/>)
  4. Literata (Bold twice in Word, because there is a SemiBold entry in fvar and not in STAT)

@aaronbell
Copy link
Collaborator

I believe that all fonts that needed correction have been corrected, and PRs opened for each. Given that, I'm going to go ahead and close this issue :)

@thlinard
Copy link
Contributor

thlinard commented Oct 1, 2021

Hi @aaronbell

Same question as #3267: shoulddn't the issue be closed only when the fix is released, rather than just at the PR stage?

And also, as #3267: many PRs missing (Archivo Narrow, Fraunces, all the Noto VFs — a lot !)? (or am I missing something?)

Congratulations for the job done, anyway. 😃

@aaronbell
Copy link
Collaborator

Sure. I mistakenly thought all of the affected fonts were either merged, or soon to be merged, and I forgot that there's a number of other fonts that still need update! Whoops.

On the various fonts:

  1. Archive Narrow – As Archivo currently supports the width axis, I believe that this one can be delisted (I think that's part of the reason I passed by it). @RosaWagner Thoughts?
  2. Fraunces – Am not sure why I skipped that one. I recall looking at it, but it isn't on my sheet. It looks like it is a fairly minor fix. I'll take a look next week at some point.
  3. Noto is being handled by someone else :)

@aaronbell aaronbell reopened this Oct 1, 2021
@RosaWagner
Copy link
Contributor

RosaWagner commented Oct 2, 2021

  1. Some fonts are used in Google docs, we can’t de-list them (that’s my understanding about this case)
  2. I don’t know. This font is fairly complicated and can’t be touched easily. Separated issue has to be opened, and we can see according to priorities.
  3. To answer a question in the other linked issue: it makes the workflow less time-consuming if the issue gets closed automatically once the PR is merged. Please don’t ask to reopened, onboarders need a healthy working system and this means being able to cross elements off their check list (aka the opened issues) to reduce mental charge. Everything that happens once the PR is merged, is not of their responsibility anymore and therefore shouldn’t be laying around opened as a to-do. Obviously it can be re-thought if necessary, but it would mean that the person tracking the fonts would have to close the issues manually when the font has been produced, which would be highly annoying, would lead to mistakes or, as we can appreciate already, would result in issues never getting closed although they are fixed. For now it is working great, so let’s continue this way :)

@thlinard
Copy link
Contributor

thlinard commented Oct 2, 2021

  1. If Archivo Narrow is to be kept, it must be fixed (of course, it's up to you to determine the priority). Edit: done.
  2. Indeed, an separate issue has to be opened: the present issue, STAT table errors in variable fonts #2391, is a "meta-issue", serving as a point of reference for dozens of other issues with similar flaws (hence Aaron's table: STAT table errors in variable fonts #2391 (comment)). The particular issue is in fact already open: Please correct SOFT value for Thin Italic instance undercasetype/Fraunces#259.
  3. "once the PR is merged": OK, good criterion. The PRs depending on this issue which have not yet merged:

@aaronbell
Copy link
Collaborator

It appears that the STAT tables in the Noto fonts still have not been fixed. Is there a timeline for that?

@tiroj
Copy link

tiroj commented Jun 14, 2023

Can anyone confirm that the InDesign issue reported here is still an active problem? I am trying to figure out if a behaviour I am experiencing in testing a new variable font is that same as discussed here, or something new.

@davelab6
Copy link
Member

A user wrote in.

For some reason the Public Sans Variable fonts, when both the regular and italic variants are installed/activated, will appear incorrectly in various apps, such as Adobe Ps, Ai etc. Instead of both the upright and italics being shown in the font selector drop-down menu after each other, only either the regular or italic family appears, and only those are usable at a time.

Basically the issue is the same as what's described here:
https://forum.glyphsapp.com/t/variable-font-naming-for-italic-face-conflict-with-regular/24190/1

It seems to be some sort of variable instance/naming conflict, but I can't confirm because I can't figure out what's causing this problem.

However, this isn't an Adobe problem as it's consistent across other programs. I believe this is a typeface design/metadata problem, since some other variable fonts do appear correctly every single time, such as Work Sans Variable. Just like in the forum post linked above, I'm also unable to find out why this is happening. I've found other fonts that also behave the same incorrect way, such as Roboto Mono Variable

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

9 participants