⚡️Faster subsequent page-loads by prefetching in-viewport links during idle time
Faster subsequent page-loads by prefetching or prerendering in-viewport links during idle time
Quicklink attempts to make navigations to subsequent pages load faster. It:
navigator.connection.effectiveType
) or has data-saver enabled (using navigator.connection.saveData
)<link rel=prefetch>
or XHR) or prerenders (using Speculation Rules API) URLs to the links. Provides some control over the request priority (can switch to fetch()
if supported).This project aims to be a drop-in solution for sites to prefetch or prerender links based on what is in the user’s viewport. It also aims to be small (< 2KB minified/gzipped).
npm install quicklink
You can also grab quicklink
from unpkg.com/quicklink.
Once initialized, quicklink
will automatically prefetch URLs for links that are in-viewport during idle time.
Quickstart:
<!-- Include quicklink from dist -->
<script src="dist/quicklink.umd.js"></script>
<!-- Initialize (you can do this whenever you want) -->
<script>
quicklink.listen();
</script>
For example, you can initialize after the load
event fires:
<script>
window.addEventListener('load', () => {
quicklink.listen();
});
</script>
ES Module import:
import {listen, prefetch} from 'quicklink';
First, install the packages with Node.js and npm:
npm install quicklink webpack-route-manifest --save-dev
Then, configure Webpack route manifest into your project, as explained here.
This will generate a map of routes and chunks called rmanifest.json
. It can be obtained at:
site_url/rmanifest.json
window.__rmanifest
Import quicklink
React HOC where want to add prefetching functionality.
Wrap your routes with the withQuicklink()
HOC.
Example:
import {withQuicklink} from 'quicklink/dist/react/hoc.js';
const options = {
origins: [],
};
<Suspense fallback={<div>Loading...</div>}>
<Route path='/' exact component={withQuicklink(Home, options)} />
<Route path='/blog' exact component={withQuicklink(Blog, options)} />
<Route path='/blog/:title' component={withQuicklink(Article, options)} />
<Route path='/about' exact component={withQuicklink(About, options)} />
</Suspense>;
Returns: Function
A “reset” function is returned, which will empty the active IntersectionObserver
and the cache of URLs that have already been prefetched or prerendered. This can be used between page navigations and/or when significant DOM changes have occurred.
Boolean
false
Whether to switch from the default prefetching mode to the prerendering mode for the links inside the viewport.
Note: The prerendering mode (when this option is set to true) will fallback to the prefetching mode if the browser does not support prerender.
Boolean
false
Whether to activate both the prefetching and prerendering mode at the same time.
Number
0
The amount of time each link needs to stay inside the viewport before being prefetched, in milliseconds.
HTMLElement|NodeList<A>
document.body
The DOM element to observe for in-viewport links to prefetch or the NodeList of Anchor Elements.
Number
Infinity
The total requests that can be prefetched or prerendered while observing the options.el
container.
Number
0
The area percentage of each link that must have entered the viewport to be fetched, in its decimal form (e.g. 0.25 = 25%).
Number
Infinity
The concurrency limit for simultaneous requests while observing the options.el
container.
Number
2000
The requestIdleCallback
timeout, in milliseconds.
Note: The browser must be idle for the configured duration before prefetching.
Function
requestIdleCallback
A function used for specifying a timeout
delay.
This can be swapped out for a custom function like networkIdleCallback (see demos).
By default, this uses requestIdleCallback
or the embedded polyfill.
Boolean
false
Whether or not the URLs within the options.el
container should be treated as high priority.
When true
, quicklink will attempt to use the fetch()
API if supported (rather than link[rel=prefetch]
).
Array<String>
[location.hostname]
A static array of URL hostnames that are allowed to be prefetched.
Defaults to the same domain origin, which prevents any cross-origin requests.
Important: An empty array ([]
) allows all origins to be prefetched.
RegExp
or Function
or Array
[]
Determine if a URL should be prefetched.
When a RegExp
tests positive, a Function
returns true
, or an Array
contains the string, then the URL is not prefetched.
Note: An
Array
may containString
,RegExp
, orFunction
values.
Important: This logic is executed after origin matching!
Function
An optional error handler that will receive any errors from prefetched requests.
By default, these errors are silently ignored.
Function
An optional function to generate the URL to prefetch. It receives an Element as the argument.
Returns: Promise
The urls
provided are always passed through Promise.all
, which means the result will always resolve to an Array.
Important: You much
catch
you own request error(s).
String
or Array<String>
true
One or many URLs to be prefetched.
Note: Each
url
value is resolved from the current location.
Boolean
false
Whether or not the URL(s) should be treated as “high priority” targets.
By default, calls to prefetch()
are low priority.
Note: This behaves identically to
listen()
'spriority
option.
Returns: Promise
Important: You much
catch
you own request error(s).
String
or Array<String>
true
One or many URLs to be prerendered.
Note: Speculative Rules API supports same-site cross origin Prerendering with opt-in header.
quicklink
:
IntersectionObserver
to be supported (see Can I Use). We recommend conditionally polyfilling this feature with a service like Polyfill.io:<script src="https://polyfill.io/v3/polyfill.min.js?features=IntersectionObserver"></script>
Alternatively, see the Intersection Observer polyfill.
Defaults to 2 seconds (via requestIdleCallback
). Here we override it to 4 seconds:
quicklink.listen({
timeout: 4000,
});
Defaults to document
otherwise.
quicklink.listen({
el: document.querySelectorAll('a.linksToPrefetch'),
});
Defaults to document
otherwise.
quicklink.listen({
el: document.getElementById('carousel'),
});
prefetch()
URLsIf you would prefer to provide a static list of URLs to be prefetched, instead of detecting those in-viewport, customizing URLs is supported.
// Single URL
quicklink.prefetch('2.html');
// Multiple URLs
quicklink.prefetch(['2.html', '3.html', '4.js']);
// Multiple URLs, with high priority
// Note: Can also be use with single URL!
quicklink.prefetch(['2.html', '3.html', '4.js'], true);
prerender()
URLsIf you would prefer to provide a static list of URLs to be prerendered, instead of detecting those in-viewport, customizing URLs is supported.
// Single URL
quicklink.prerender('2.html');
// Multiple URLs
quicklink.prerender(['2.html', '3.html', '4.js']);
Defaults to low-priority (rel=prefetch
or XHR). For high-priority (priority: true
), attempts to use fetch()
or falls back to XHR.
Note: This runs
prefetch(..., true)
with URLs found within theoptions.el
container.
quicklink.listen({priority: true});
Provide a list of hostnames that should be prefetch-able. Only the same origin is allowed by default.
Important: You must also include your own hostname!
quicklink.listen({
origins: [
// add mine
'my-website.com',
'api.my-website.com',
// add third-parties
'other-website.com',
'example.com',
// ...
],
});
Enables all cross-origin requests to be made.
quicklink.listen({
origins: true,
// or
origins: [],
});
These filters run after the origins
matching has run. Ignores can be useful for avoiding large file downloads or for responding to DOM attributes dynamically.
// Same-origin restraint is enabled by default.
//
// This example will ignore all requests to:
// - all "/api/*" pathnames
// - all ".zip" extensions
// - all <a> tags with "noprefetch" attribute
//
quicklink.listen({
ignores: [
/\/api\/?/,
uri => uri.includes('.zip'),
(uri, elem) => elem.hasAttribute('noprefetch'),
],
});
You may also wish to ignore prefetches to URLs which contain a URL fragment (e.g. index.html#top
). This can be useful if you (1) are using anchors to headings in a page or (2) have URL fragments setup for a single-page application, and which to avoid firing prefetches for similar URLs.
Using ignores
this can be achieved as follows:
quicklink.listen({
ignores: [
uri => uri.includes('#'),
// or RegExp: /#(.+)/
// or element matching: (uri, elem) => !!elem.hash
],
});
The hrefFn method allows to build the URL to prefetch (e.g. API endpoint) on the fly instead of the prefetching the href
attribute URL.
quicklink.listen({
hrefFn(element) {
return element.href.replace('html', 'json');
},
});
The prefetching provided by quicklink
can be viewed as a progressive enhancement. Cross-browser support is as follows:
Set()
and Array.from()
shims. Projects like es6-shim are an alternative you can consider.Certain features have layered support:
navigator.connection.effectiveType
) is only available in Chrome 61+ and Opera 57+{priority: true}
and the Fetch API isn’t available, XHR will be used instead.A prefetch
method can be individually imported for use in other projects.
This method includes the logic to respect Data Saver and 2G connections. It also issues requests thru fetch()
, XHRs, or link[rel=prefetch]
depending on (a) the isPriority
value and (b) the current browser’s support.
After installing quicklink
as a dependency, you can use it as follows:
<script type="module">
import {prefetch} from 'quicklink';
prefetch(['1.html', '2.html']).catch(error => {
// Handle own errors
});
</script>
href
attributeHere’s a WebPageTest run for our demo improving page-load performance by up to 4 seconds via quicklink’s prefetching. A video comparison of the before/after prefetching is on YouTube.
For demo purposes, we deployed a version of the Google Blog on
Firebase hosting. We then deployed another version of it, adding quicklink to the homepage and benchmarked navigating from the homepage to an article that was
automatically prefetched. The prefetched version loaded faster.
Please note: this is by no means an exhaustive benchmark of the pros and cons of in-viewport link prefetching. Just a demo of the potential improvements the approach can offer. Your own mileage may heavily vary.
Cross-origin prefetching (e.g a.com/foo.html
prefetches b.com/bar.html
) has a number of limitations. One such limitation is with session-stitching. b.com
may expect a.com
’s navigation requests to include session information (e.g a temporary ID - e.g b.com/bar.html?hash=<>×tamp=<>
), where this information is used to customize the experience or log information to analytics. If session-stitching requires a timestamp in the URL, what is prefetched and stored in the HTTP cache may not be the same as the one the user ultimately navigates to. This introduces a challenge as it can result in double prefetches.
To workaround this problem, you can consider passing along session information via the ping attribute (separately) so the origin can stitch a session together asynchronously.
Sites that rely on ads as a source of monetization should not prefetch ad-links, to avoid unintentionally counting clicks against those ad placements, which can lead to inflated Ad CTR (click-through-rate).
Ads appear on sites mostly in two ways:
Inside iframes: By default, most ad-servers render ads within iframes. In these cases, those ad-links won’t be prefetched by Quicklink, unless a developer explicitly passes in the URL of an ads iframe. The reason is that the library look-up for in-viewport elements is restricted to those of the top-level origin.
Outside iframes:: In cases when the site shows same-origin ads, displayed in the top-level document (e.g. by hosting the ads themselves and by displaying the ads in the page directly), the developer needs to explicitly tell Quicklink to avoid prefetching these links. This can be achieved by passing the URL or subpath of the ad-link, or the element containing it to the custom ignore patterns list.
Intersection Observer
to prefetch all of the links that are in view and provided heavy inspiration for this project.Licensed under the Apache-2.0 license.