-
Notifications
You must be signed in to change notification settings - Fork 412
/
Copy pathprefetch.mjs
85 lines (75 loc) · 2.51 KB
/
prefetch.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/**
* Portions copyright 2018 Google Inc.
* Inspired by Gatsby's prefetching logic, with those portions
* remaining MIT. Additions include support for Fetch API,
* XHR switching, SaveData and Effective Connection Type checking.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**/
/**
* Checks if a feature on `link` is natively supported.
* Examples of features include `prefetch` and `preload`.
* @param {Object} link Link object.
* @return {Boolean} whether the feature is supported
*/
function hasPrefetch(link) {
link = document.createElement('link');
return link.relList && link.relList.supports && link.relList.supports('prefetch');
}
/**
* Fetches a given URL using `<link rel=prefetch>`
* @param {string} url - the URL to fetch
* @return {Object} a Promise
*/
function viaDOM(url) {
return new Promise((res, rej, link) => {
link = document.createElement(`link`);
link.rel = `prefetch`;
link.href = url;
link.onload = res;
link.onerror = rej;
document.head.appendChild(link);
});
};
/**
* Fetches a given URL using XMLHttpRequest
* @param {string} url - the URL to fetch
* @return {Object} a Promise
*/
function viaXHR(url) {
return new Promise((res, rej, req) => {
req = new XMLHttpRequest();
req.open(`GET`, url, req.withCredentials=true);
req.onload = () => {
(req.status === 200) ? res() : rej();
};
req.send();
});
}
/**
* Fetches a given URL using the Fetch API. Falls back
* to XMLHttpRequest if the API is not supported.
* @param {string} url - the URL to fetch
* @return {Object} a Promise
*/
export function priority(url) {
// TODO: Investigate using preload for high-priority
// fetches. May have to sniff file-extension to provide
// valid 'as' values. In the future, we may be able to
// use Priority Hints here.
//
// As of 2018, fetch() is high-priority in Chrome
// and medium-priority in Safari.
return window.fetch ? fetch(url, {credentials: `include`}) : viaXHR(url);
}
export const supported = hasPrefetch() ? viaDOM : viaXHR;