While Svelte 5 is a complete rewrite, we have done our best to ensure that most codebases can upgrade with a minimum of hassle. That said, there are a few small breaking changes which may require action on your part. They are listed here.
Components are no longer classespermalink
In Svelte 3 and 4, components are classes. In Svelte 5 they are functions and should be instantiated differently. If you need to manually instantiate components, you should use mount or createRoot (imported from svelte) instead. If you see this error using SvelteKit, try updating to the latest version of SvelteKit first, which adds support for Svelte 5. If you're using Svelte without SvelteKit, you'll likely have a main.js file (or similar) which you need to adjust:
 import { createRoot } from 'svelte';
import App from './App.svelte'
 const app = new App({ target: document.getElementById("app") });
 const app = createRoot(App, { target: document.getElementById("app") });
export default app;
createRoot returns an object with a $set and $destroy method on it. It does not come with an $on method you may know from the class component API. Instead, pass them via the events property on the options argument. If you don't need to interact with the component instance after creating it, you can use mount instead, which saves some bytes.
Note that using
eventsis discouraged — instead, use callbacks
As a stop-gap-solution, you can also use createClassComponent or asClassComponent (imported from svelte/legacy) instead to keep the same API after instantiating. If this component is not under your control, you can use the legacy.componentApi compiler option for auto-applied backwards compatibility (note that this adds a bit of overhead to each component).
Server API changespermalink
Similarly, components no longer have a render method when compiled for server side rendering. Instead, pass the function to render from svelte/server:
 import { render } from 'svelte/server';
import App from './App.svelte';
 const { html, head } = App.render({ message: 'hello' });
 const { html, head } = render(App, { props: { message: 'hello' } });
render also no longer returns CSS; it should be served separately from a CSS file.
bind:this changespermalink
Because components are no longer classes, using bind:this no longer returns a class instance with $set, $on and $destroy methods on it. It only returns the instance exports (export function/const) and, if you're using the accessors option, a getter/setter-pair for each property.
Whitespace handling changedpermalink
Previously, Svelte employed a very complicated algorithm to determine if whitespace should be kept or not. Svelte 5 simplifies this which makes it easier to reason about as a developer. The rules are:
- Whitespace between nodes is collapsed to one whitespace
- Whitespace at the start and end of a tag is removed completely
- Certain exceptions apply such as keeping whitespace inside pretags
As before, you can disable whitespace trimming by setting the preserveWhitespace option in your compiler settings or on a per-component basis in <svelte:options>.
More recent browser requiredpermalink
Svelte now use Mutation Observers instead of IFrames to measure dimensions for bind:clientWidth/clientHeight/offsetWidth/offsetHeight. It also no longer listens to the change event on range inputs. Lastly, the legacy option was removed (or rather, replaced with a different set of settings).
Changes to compiler optionspermalink
- The false/true(already deprecated previously) and the"none"values were removed as valid values from thecssoption
- The legacyoption was repurposed
- The hydratableoption has been removed. Svelte components are always hydratable now
- The tagoption was removed. Use<svelte:options customElement="tag-name" />inside the component instead
- The loopGuardTimeout,format,sveltePath,errorModeandvarsReportoptions were removed
The children prop is reservedpermalink
Content inside component tags becomes a snippet prop called children. You cannot have a separate prop by that name.
Other breaking changespermalink
Stricter @const assignment validationpermalink
Assignments to destructured parts of a @const declaration are no longer allowed. It was an oversight that this was ever allowed.
CSS hash position no longer deterministicpermalink
Previously Svelte would always insert the CSS hash last. This is no longer guaranteed in Svelte 5. This is only breaking if you have very weird css selectors.
beforeUpdate changepermalink
beforeUpdate no longer runs twice on initial render if it modifies a variable referenced in the template.
contenteditable behavior changepermalink
If you have a contenteditable node with a corresponding binding and a reactive value inside it (example: <div contenteditable=true bind:textContent>count is {count}</div>), then the value inside the contenteditable will not be updated by updates to count because the binding takes full control over the content immediately and it should only be updated through it.
oneventname attributes no longer accept string valuespermalink
In Svelte 4, it was possible to specify event attributes on HTML elements as a string:
<button onclick="alert('hello')">...</button>This is not recommended, and is no longer possible in Svelte 5, where properties like onclick replace on:click as the mechanism for adding event handlers.