Using web fonts? Be careful

Searching for better design and stronger identity, custom fonts have become something present in almost all modern sites.

While the result is great, this feature also comes with one or more network requests and other compromises like FOUT and FOIT.

What are those weird acronyms? Well, you're going to see them in any post or article related to web fonts today so let's do a quick description of both.

FOUT (Flash Of Unstyled Text) describes the behavior of some browsers showing you the content of the page with a fallback font until the custom one is loaded. When the font is ready the switch is triggered, like Firefox does.

FOIT (Flash Of Invisible Text) is when the browser chooses not to display the text if it has a custom font assigned that isn't ready for use yet, like Chrome and Chrome Mobile.

Hitting the user experience spot

Both scenarios are not desirable. FOUT will cause your page to shift because of the physical differences between the font that's being displayed at the moment and the custom one. FOIT will literally hide your text content when the custom font has been loaded while the browser repaints the whole site.

You will have to choose between shift and hide.

The first one, while it's not perfect, prioritizes showing something to the user while the font is being loaded. Content first approaches were the trending thing in front end development last year with above-the-fold and critical CSS being the star of the show, so it makes sense to go that way.

If there's a font that's taking a while to load it is better to show content to the user until everything is ready. Google's PageSpeed Insight will penalize you if you don't follow this approach, for example.

After hearing about this, I just loaded the stylesheet containing the @font-face declaration by JavaScript, wait for the onload event to fire and add a class to the body or html tag to change the font-family of the site by CSS.

var fontStyleSheet = document.createElement('link');

fontStyleSheet.onload = function () {
  document.body.classList.add('font-loaded');
};

fontStyleSheet.href = '//fonts.googleapis.com/css?family=Roboto';
document.head.appendChild(fontStyleSheet);

And guess what? It didn't work. There was still a gap where invisible text was shown, apparently this happens because I changed the font family of the site when the stylesheet was ready which is ok, but the browser still was rendering the new font.

Solutions available today

A couple of developers noticed these issues and created libraries that provide events that truly indicate the loading and active state of the font. The first one, fontfaceonload by Zach Leatherman.

You still need to add your fonts via @font-face declaration or include the link tag in your page and later listen to a specific family of fonts to load.

FontFaceOnload('Oswald', {
  success: function () {
    // when font is available, add class to body
    document.body.classList.add('font-loaded');
  }
});

The second one is fontfaceobserver by Bram Stein. According to its creator, this one uses scroll events to detect when the font is available immediately.

Again, include your custom font by CSS but this time an observer object is created.

var observer = new FontFaceObserver('Oswald');

observer.check().then(function () {
  document.body.classList.add('font-loaded');
});

You might have noticed that this library uses a check method and the reserved word then as the Promise API does to run code after the font is ready.

This last one is highly recommended if you're loading a font from Google, Typekit or Fontdeck. It's actually co-developed by Typekit and Google developers and its name is webfontloader.

You won't need a @font-face declaration in your styles here, just JavaScript.

WebFont.load({
  google: {
    families: ['Oswald']
  }
});

The script will automatically add classes to the body of the document, but if you feel more comfortable using your own class names you can disable this feature and do it yourself.

WebFont.load({
  google: {
    families: ['Oswald']
  },
  classes: false,
  active: function () {
    document.body.classList.add('font-loaded');
  }
});

In the wild

The best way to test these libraries is to see them working on the web, so I've created a small site with three sample pages, one for each script. The sample pages don't load any resources that are not related to font management, they measure the time the script gets to the page and detects the font is ready for use.

After running these sites a few times to get some numbers about the speed of each one, I can say that there are no big differences. The average mark was 270ms to load the library, run the code and show the font with a normal internet home connection.

About weights, fontfaceonload is 1.06KB, fontfaceobserver is 1.83KB and webfontloader 4.46KB, all minified and gzipped. The last one is heavier because it supports various font API out of the box.

Then I've moved to WebPageTest to have a better insight of the performance differences between them and I saw that fontfaceonload wasn't solving the FOIT issue more than once thought as a user I didn't catch that shift at all. I might have missed some step or done something wrong, though I followed the documentation from the repository. The other two worked as expected and have become the choice of most developers.

You Only Load Once

Now that we have nice solutions, we are causing a FOUT on purpose to always show text content to the user, but now created a new problem.

Once the user got to your homepage and starts navigating through your site is going to experience the same effect in every page because, though the font has been previously loaded and cached by the browser, we are not switching to it until we detect it by JavaScript and add a class to the html or body tag.

This is not hard to solve, once you load the font you can place a cookie.

var observer = new FontFaceObserver('Oswald');

observer.check().then(function () {
  document.body.classList.add('font-loaded');
  document.cookie = 'fonts-loaded';
});

Then add the class dynamically, for example in PHP if your engine is Wordpress, to your site. What about the ones that use static generators, like this one? Well you have no solution around this.

Yes, boooooh.

Good thing is new specs to control the font rendering behavior are being discussed.

If you are more curious about FOUT and FIOT Chris Coyer wrote a good post about them. Also, Scott Jehl from Filament Group did a great post covering loading web fonts using their own site as a test case. Go check it out for more precise information about this.

Wrap-up

I love web fonts and how they make your web feel more unique, but we don't have to forget that content and user experience come first. If you're going to include custom fonts in your site use font events libraries to load them.

Do you want me to speak at your conference or write for your publication?

Click here to contact me for collaborations.