• Stars
    star
    369
  • Rank 115,686 (Top 3 %)
  • Language
    TypeScript
  • License
    MIT License
  • Created over 3 years ago
  • Updated 3 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Delayed hydration for progressively enhanced apps. Reduced blocking time and improved Google Lighthouse scores. ⚡️

nuxt-delay-hydration

NPM Downloads


Status: Stable v2 v0 ✅ , v3 main
Made possible by my Sponsor Program 💖
Follow me @harlan_zw 🐦

⚠️ This is a "hack" to trick Google Lighthouse into thinking your site is faster than it otherwise would be.

  • It should only ever be used for progressively enhanced websites.
  • It may not provide any real performance or SEO benefit (test it with CrUX, not Google Lighthouse).

Features

  • 🔥 Reduce your sites "Blocking Time"
  • 🚦 Filter it to run only on specific pages
  • 🍃 Minimal config
  • 🔁 (optional) Replay pre-hydration clicks

Why delay hydration?

Delaying hydration is a technique to hint to Google that our scripts are not required for our app to function.

By delaying hydration we improve the Google Lighthouse score by reducing your "Blocking Time" metric.

Previously you may have used vue-lazy-hydration which works well. However, it is just a more verbose way of providing hints to Google, like we're doing with this module.


What is a progressively enhanced app?

A progressively enhanced app is one that is designed to work without JavaScript, and then progressively enhanced with JavaScript.

Your scripts should not be required to use your website, this is what we're hinting to Google by delaying the hydration.

To do that you can ensure:

  • Full HTML served on the initial response
  • Scripts don't trigger a CLS
  • Avoid using scripts to set images, will affect the LCP

How this module works
A promise is injected into your app, the location is based on the mode. The promise is resolved as soon as either of these events has fired:
  • an interaction event (mouse move, scroll, click, etc)
  • an idle callback with a fixed timeout

The idle CPU time hints to Google that these scripts are not required for your app to run.

For example:

  • if a Google bot visits the page and has no interaction, out of the box the hydration won't occur until the browser idle callback + 6 seconds
  • if a user visits the page and moves their cursor or scrolls, the hydration will be triggered immediately. The chance of interacting with the non-hydration app will be minimised

Keep in mind, this is a hacky solution. Until Google can recognise progressive script enhancements, we'll need to rely on this approach.


Install

If you're using Nuxt 2.x, please follow the docs on the v0 branch.

yarn add -D nuxt-delay-hydration
# npm i -D nuxt-delay-hydration
# pnpm add -D nuxt-delay-hydration

Requirement: Progressively enhanced SSR or SSG Nuxt app.


Usage

// nuxt.config.ts
export default {
  modules: [
    'nuxt-delay-hydration',
  ],
  delayHydration: {
    // enables nuxt-delay-hydration in dev mode for testing
    debug: process.env.NODE_ENV === 'development'
  }
}

Note: The module will not run in development unless you have enabled debug.

Choosing a mode

By default, no mode is selected, you will need to select how you would the module to work.

Type: init | mount| manual | false

Default: false

Type Description Use Case
false default Disable the module Testing
init Delays all scripts from loading. Zero or minimal plugins/modules.
mount recommended Delays Nuxt while it's mounting. Plugins and some third-party scripts will work. Minimal non-critical plugins and third-party plugins.
manual Delay is provided by the DelayHydration component. All other apps

Regardless of the mode you choose, please read further optimisations.

Init Mode

This mode delays all scripts from loading until the hydration promise is resolved.

It does this by hooking into the HTML rendering, removing all script tags and adding them back after the hydration promise is resolved.

This will provide the biggest speed improvements however is the riskiest.

Pros: Provides the biggest blocking time reduction

Cons: Risky if you have critical third party scripts

Benchmark: ~90-100% reduction

export default {
  delayHydration: {
    mode: 'init'
  }
}

Mount Mode

This mode delays Nuxt while it's mounting. Plugins and some third-party scripts will work.

This delays your layout and page components.

Pros: Safer and still provides good improvements

Cons: May still break certain layouts if they are js dependent.

Benchmark: ~70% reduction

export default {
  delayHydration: {
    mode: 'mount'
  }
}

Manual Mode

Using the manual mode, you manually specify what part of your app you'd like to delay. Useful for when you need some part of the page to always hydrate immediately, such as a navigation drawer.

Pros: Safest way to optimise

Cons: Speed improvement based on usage

export default {
  delayHydration: {
    mode: 'manual'
  }
}

DelayHydration component

Once you have set the mode, you need to use the component.

<template>
  <div>
    <DelayHydration>
      <div>
        <LazyMyExpensiveComponent />
      </div>
    </DelayHydration>
  </div>
</template>

Guides

Debugging

Debug mode
It's recommended that you do thorough testing on your app with the module before deploying it into production.

To make sure the module is doing what you expect, there is a debug mode, which when enabled will log behaviour in the console.

It might be a good idea to always debug on your local environment, in that instance you could do:

export default {
  delayHydration: {
    debug: process.env.NODE_ENV === 'development',
  },
}

Visualising the hydration status
It can be unclear at times whether your app has been hydrated or not if it's quite static, this can make debugging hard.

To make things easier, there is a component HydrationStatus which will tell you what's going on.

<template>
  <div>
    <MyHeader />
    <DelayHydration>
      <div>
        <!-- Show the hydration status, only for debugging -->
        <HydrationStatus />
        <main>
          <nuxt />
        </main>
        <my-footer />
      </div>
    </DelayHydration>
  </div>
</template>

Performance Auditing

Use my audit tool: https://unlighthouse.dev/

Replaying hydration click

What is this and how to enable
One of the issues with delaying hydration is that a user interaction event can occur before your scripts are loaded, leading to a user having to click on something multiple times for it to do what they expect. Think of a hamburger that is triggered using Javascript, if your app isn't hydrated then clicking it won't do anything.

The best fix for this is to write your HTML in a way that doesn't need Javascript to be interactive.

However, there are use cases where you need to use Javascript and responding to the first click is important. In those instances, you can enable the replay of the click.

export default {
  delayHydration: {
    replayClick: true
  },
}

This is an experimental configuration, you should test this option yourself before implementing it into your production app.

Further Optimisations

Load heavy components async
When you load in a heavy component synchronously, the javascript will be bundled in with the main application payload.

This will decrease all of your performance metrics. It's recommended you use async imports for these components.

Run nuxi analyze to find large components. When loading them, prefix them with Lazy.

Advanced Configuration

Configuration should be provided on the delayHydration key within your Nuxt config.

If you're finding the lab or field data is not performing, you may want to tinker with this advanced configuration.

Filtering routes

Using the include and exclude options, you can specify which routes you'd like to delay hydration on.

// nuxt.config.ts
export default {
  delayHydration: {
    include: [
      '/blog/**',
    ],
    exclude: [
      '/admin/**'
    ],
  },
}

You can provide a glob pattern similar to route rules or a regex.

Event Hydration

hydrateOnEvents

  • Type: string[]
  • Default: [ 'mousemove', 'scroll', 'keydown', 'click', 'touchstart', 'wheel' ]

Controls which browser events should trigger the hydration to resume. By default, it is quite aggressive to avoid possible user experience issues.

replayClick

  • Type: boolean
  • Default: false

If the trigger for hydration was a click, you can replay it. Replaying it will re-execute the event when it is presumed your app is hydrated.

For example, if a user clicks a hamburger icon and hydration is required to open the menu, it would replay the click once hydrated.

⚠️ This is experimental, use with caution.

Idle Hydration

idleCallbackTimeout

  • Type: number (milliseconds)
  • Default: 7000

When waiting for an idle callback, it's possible to define a max amount of time to wait in milliseconds. This is useful when there are a lot of network requests happening.

postIdleTimeout

  • Type: { mobile: number, desktop: number } (milliseconds)
  • Default: { mobile: 5000, desktop: 4000, }

How many to wait (in milliseconds) after the idle callback before we resume the hydration. This extra timeout is required to avoid the standard "blocking", we need to provide real idle time to lighthouse.

Mobile should always be higher than desktop as the CPU capacity will generally be a lot less than a desktop.

Note: The default will likely be customised in the future based on further benchmarking.

Debugging

debug

  • Type: boolean
  • Default: false

Log details in the console on when hydration is blocked and when and why it becomes unblocked.

Benchmarks

Live examples

Credits

Sponsors

License

MIT License © 2022 - Present Harlan Wilton

More Repositories

1

unlighthouse

Scan your entire site with Google Lighthouse in 2 minutes (on average). Open source, fully configurable with minimal setup.
JavaScript
3,840
star
2

nuxt-seo

The complete SEO solution for Nuxt.
TypeScript
1,053
star
3

request-indexing

Find your missing on pages Google and request them to be indexed using the Web Search Indexing API.
TypeScript
313
star
4

nuxt-webpack-optimisations

Make your Nuxt.js webpack builds faster ⚡
TypeScript
269
star
5

nuxt-og-image

Enlightened runtime images generated with Vue templates.
TypeScript
268
star
6

harlanzw.com

My personal website built with Nuxt 3 and Nuxt Content 2.
Vue
195
star
7

nuxt-simple-sitemap

Powerfully flexible XML Sitemaps that integrate seamlessly, for Nuxt.
TypeScript
166
star
8

nuxt-schema-org

The quickest and easiest way to build Schema.org graphs for Nuxt.
TypeScript
145
star
9

zhead

All of the TypeScript definitions for <head>.
TypeScript
122
star
10

nuxt-simple-robots

(Migrated to @nuxtjs/robots) Tame the robots crawling and indexing your Nuxt site.
TypeScript
114
star
11

nuxt-link-checker

Find and magically fix links that may be negatively effecting your Nuxt sites SEO.
TypeScript
87
star
12

nuxt-seo-utils

SEO utilities to improve your Nuxt sites discoverability and shareability.
TypeScript
75
star
13

unrouted

Unrouted is a minimal, composable router built for speed, portability and DX
TypeScript
67
star
14

nuxt-site-config

Unifying site config with powerful and flexible APIs, for module authors and users.
TypeScript
65
star
15

harlanzw.com-vitepress

My personal blog built using VitePress and TailwindCSS
Vue
44
star
16

vue-cli-plugin-import-components

🔌 Automatically import components in your Vue CLI apps.
TypeScript
39
star
17

packrup

Simple utils to pack arrays, objects and strings to a flat object (and back again).
TypeScript
17
star
18

schema-org-graph

Build Schema.org graphs for JavaScript Runtimes (Browser, Node, etc). Improve your sites SEO with quick and easy Rich Results.
TypeScript
16
star
19

laradock-cli

[Unmaintained] Your new best friend for Laradock.
Dockerfile
16
star
20

changelogd

🪵 Aggressively find a packages changelog (or releases) between versions.
TypeScript
9
star
21

wp-keystone

A Wordpress Boilerplate combining the best libraries and practices
PHP
8
star
22

laravel-swiftype

Swiftype Integration for Laravel
PHP
7
star
23

nuxt-mycelium

🍄 TBA 🤫
TypeScript
7
star
24

to-vite-and-beyond-a-history-and-future-of-bundling

Vue
6
star
25

WP-Development-Environment

WorldPress Plugin - Development Environment
PHP
5
star
26

nuxt-seo-ui

Deprecated - Nuxt components with Schema.org and SEO integrations
TypeScript
5
star
27

talk-supercharged-head-management

NuxtNation 2022 Talk - VueUse Head v1
Vue
3
star
28

laravel-modern-mail

TBA
PHP
2
star
29

static

TypeScript
1
star
30

blockfolio-php

Blockfolio SDK for PHP - Unofficial
PHP
1
star
31

penguin-pass

Laravel / PHP Package for kid friendly passwords
PHP
1
star
32

nuxt-og-image-playground

Testing out nitro deployments with nuxt-og-imge.
Vue
1
star