Skip to content

Commit 20c6df1

Browse files
committed
preload option, some docs
1 parent 81186b6 commit 20c6df1

File tree

5 files changed

+78
-31
lines changed

5 files changed

+78
-31
lines changed

.changeset/odd-bears-greet.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@sveltejs/kit': patch
3+
---
4+
5+
[feat] preload fonts and add preload customization

documentation/docs/30-advanced/20-hooks.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -72,14 +72,16 @@ You can add call multiple `handle` functions with [the `sequence` helper functio
7272

7373
- `transformPageChunk(opts: { html: string, done: boolean }): MaybePromise<string | undefined>` — applies custom transforms to HTML. If `done` is true, it's the final chunk. Chunks are not guaranteed to be well-formed HTML (they could include an element's opening tag but not its closing tag, for example) but they will always be split at sensible boundaries such as `%sveltekit.head%` or layout/page components.
7474
- `filterSerializedResponseHeaders(name: string, value: string): boolean` — determines which headers should be included in serialized responses when a `load` function loads a resource with `fetch`. By default, none will be included.
75+
- `preload(input: { type: 'js' | 'css' | 'font' | 'asset', path: string }): boolean` — determines what should be added to the `<head>` tag to preload it. Preloading can improve performance because things are downloaded sooner, but they can also hurt core web vitals because too many things may be downloaded unnecessarily. By default, `js`, `css` and `font` files will be preloaded. `asset` files are not preloaded at all currently, but we may add this later after evaluating feedback.
7576

7677
```js
7778
/// file: src/hooks.server.js
7879
/** @type {import('@sveltejs/kit').Handle} */
7980
export async function handle({ event, resolve }) {
8081
const response = await resolve(event, {
8182
transformPageChunk: ({ html }) => html.replace('old', 'new'),
82-
filterSerializedResponseHeaders: (name) => name.startsWith('x-')
83+
filterSerializedResponseHeaders: (name) => name.startsWith('x-'),
84+
preload: ({ type, path }) => type === 'js' || path.includes('/important/')
8385
});
8486

8587
return response;

packages/kit/src/runtime/server/index.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -20,11 +20,15 @@ import { Redirect } from '../control.js';
2020

2121
/* global __SVELTEKIT_ADAPTER_NAME__ */
2222

23-
/** @param {{ html: string }} opts */
23+
/** @type {import('types').RequiredResolveOptions['transformPageChunk']} */
2424
const default_transform = ({ html }) => html;
2525

26+
/** @type {import('types').RequiredResolveOptions['filterSerializedResponseHeaders']} */
2627
const default_filter = () => false;
2728

29+
/** @type {import('types').RequiredResolveOptions['preload']} */
30+
const default_preload = ({ type }) => type !== 'asset';
31+
2832
/** @type {import('types').Respond} */
2933
export async function respond(request, options, state) {
3034
let url = new URL(request.url);
@@ -185,7 +189,8 @@ export async function respond(request, options, state) {
185189
/** @type {import('types').RequiredResolveOptions} */
186190
let resolve_opts = {
187191
transformPageChunk: default_transform,
188-
filterSerializedResponseHeaders: default_filter
192+
filterSerializedResponseHeaders: default_filter,
193+
preload: default_preload
189194
};
190195

191196
/**
@@ -211,7 +216,8 @@ export async function respond(request, options, state) {
211216

212217
resolve_opts = {
213218
transformPageChunk: opts.transformPageChunk || default_transform,
214-
filterSerializedResponseHeaders: opts.filterSerializedResponseHeaders || default_filter
219+
filterSerializedResponseHeaders: opts.filterSerializedResponseHeaders || default_filter,
220+
preload: opts.preload || default_preload
215221
};
216222
}
217223

packages/kit/src/runtime/server/page/render.js

+37-27
Original file line numberDiff line numberDiff line change
@@ -224,36 +224,43 @@ export async function render_response({
224224

225225
for (const dep of stylesheets) {
226226
const path = prefixed(dep);
227-
const attributes = [];
228227

229-
if (csp.style_needs_nonce) {
230-
attributes.push(`nonce="${csp.nonce}"`);
231-
}
228+
if (resolve_opts.preload({ type: 'css', path })) {
229+
const attributes = [];
232230

233-
if (inline_styles.has(dep)) {
234-
// don't load stylesheets that are already inlined
235-
// include them in disabled state so that Vite can detect them and doesn't try to add them
236-
attributes.push('disabled', 'media="(max-width: 0)"');
237-
} else {
238-
const preload_atts = ['rel="preload"', 'as="style"'].concat(attributes);
239-
link_header_preloads.add(`<${encodeURI(path)}>; ${preload_atts.join(';')}; nopush`);
240-
}
231+
if (csp.style_needs_nonce) {
232+
attributes.push(`nonce="${csp.nonce}"`);
233+
}
234+
235+
if (inline_styles.has(dep)) {
236+
// don't load stylesheets that are already inlined
237+
// include them in disabled state so that Vite can detect them and doesn't try to add them
238+
attributes.push('disabled', 'media="(max-width: 0)"');
239+
} else {
240+
const preload_atts = ['rel="preload"', 'as="style"'].concat(attributes);
241+
link_header_preloads.add(`<${encodeURI(path)}>; ${preload_atts.join(';')}; nopush`);
242+
}
241243

242-
attributes.unshift('rel="stylesheet"');
243-
head += `\n\t\t<link href="${path}" ${attributes.join(' ')}>`;
244+
attributes.unshift('rel="stylesheet"');
245+
head += `\n\t\t<link href="${path}" ${attributes.join(' ')}>`;
246+
}
244247
}
245248

246249
for (const dep of fonts) {
247-
const ext = dep.slice(dep.lastIndexOf('.') + 1);
248-
const attributes = [
249-
'rel="preload"',
250-
'as="font"',
251-
`type="font/${ext}"`,
252-
`href="${prefixed(dep)}"`,
253-
'crossorigin'
254-
];
255-
256-
head += `\n\t\t<link ${attributes.join(' ')}>`;
250+
const path = prefixed(dep);
251+
252+
if (resolve_opts.preload({ type: 'font', path })) {
253+
const ext = dep.slice(dep.lastIndexOf('.') + 1);
254+
const attributes = [
255+
'rel="preload"',
256+
'as="font"',
257+
`type="font/${ext}"`,
258+
`href="${path}"`,
259+
'crossorigin'
260+
];
261+
262+
head += `\n\t\t<link ${attributes.join(' ')}>`;
263+
}
257264
}
258265

259266
if (page_config.csr) {
@@ -280,9 +287,12 @@ export async function render_response({
280287

281288
for (const dep of modulepreloads) {
282289
const path = prefixed(dep);
283-
link_header_preloads.add(`<${encodeURI(path)}>; rel="modulepreload"; nopush`);
284-
if (state.prerendering) {
285-
head += `\n\t\t<link rel="modulepreload" href="${path}">`;
290+
291+
if (resolve_opts.preload({ type: 'js', path })) {
292+
link_header_preloads.add(`<${encodeURI(path)}>; rel="modulepreload"; nopush`);
293+
if (state.prerendering) {
294+
head += `\n\t\t<link rel="modulepreload" href="${path}">`;
295+
}
286296
}
287297
}
288298

packages/kit/types/index.d.ts

+24
Original file line numberDiff line numberDiff line change
@@ -238,6 +238,12 @@ export interface KitConfig {
238238
};
239239
}
240240

241+
/**
242+
* This function runs every time the SvelteKit server receives a [request](https://kit.svelte.dev/docs/web-standards#fetch-apis-request) and
243+
* determines the [response](https://kit.svelte.dev/docs/web-standards#fetch-apis-response).
244+
* It receives an `event` object representing the request and a function called `resolve`, which renders the route and generates a `Response`.
245+
* This allows you to modify response headers or bodies, or bypass SvelteKit entirely (for implementing routes programmatically, for example).
246+
*/
241247
export interface Handle {
242248
(input: {
243249
event: RequestEvent;
@@ -590,8 +596,26 @@ export interface RequestHandler<
590596
}
591597

592598
export interface ResolveOptions {
599+
/**
600+
* Applies custom transforms to HTML. If `done` is true, it's the final chunk. Chunks are not guaranteed to be well-formed HTML
601+
* (they could include an element's opening tag but not its closing tag, for example)
602+
* but they will always be split at sensible boundaries such as `%sveltekit.head%` or layout/page components.
603+
* @param input the html chunk and the info if this is the last chunk
604+
*/
593605
transformPageChunk?(input: { html: string; done: boolean }): MaybePromise<string | undefined>;
606+
/**
607+
* Determines which headers should be included in serialized responses when a `load` function loads a resource with `fetch`.
608+
* By default, none will be included.
609+
* @param name header name
610+
* @param value header value
611+
*/
594612
filterSerializedResponseHeaders?(name: string, value: string): boolean;
613+
/**
614+
* Determines what should be added to the `<head>` tag to preload it.
615+
* By default, `js`, `css` and `font` files will be preloaded.
616+
* @param input the type of the file and its path
617+
*/
618+
preload?(input: { type: 'font' | 'css' | 'js' | 'asset'; path: string }): boolean;
595619
}
596620

597621
export class Server {

0 commit comments

Comments
 (0)