Migration Guide
Learn how to migrate from Formsnap v1 to v2.
Formsnap v2 introduces significant changes to support Svelte 5, requiring updates to your existing codebase. This guide covers all breaking changes and provides clear migration paths for each component.
Note
sveltekit-superforms
integration remains unchanged from v1, requiring no modifications to Superforms-specific code.
Core Changes
Formsnap v2 adopts Svelte 5's new snippet pattern, replacing slots, slot props and other traditional patterns. This change affects multiple components throughout the library.
asChild -> child Snippet
The asChild
prop has been replaced with the more flexible child
snippet pattern.
Before (v1)
In v1, the asChild
prop could be used on any component that rendered an HTML element under the hood to opt out of rendering the element to provide your own.
After (v2)
In v2, this prop has been removed completely in favor of the child
snippet, which is available for all components that render an HTML element, and exposes a props
snippet prop that you can spread onto your own element/component.
You can learn more about the child
snippet in the child documentation.
el
-> ref
Element References: Element references now use the $bindable
ref
prop.
Before (v1)
In v1, you could bind to the el
prop of any component that rendered an element to receive a reference to that HTML element.
After (v2)
In v2, this prop has been replaced by the ref
prop, which is available for all components that render an HTML element, and exposes a $bindable
reference to the underlying HTML element.
Slot Props -> Snippet Props
The various *Field*
components now use snippet props instead of slot props to provide access to their inner state.
Before (v1)
In v1, the various *Field
components provided a number of slot props for your convenience, such as value
, errors
, tainted
, and constraints
.
After (v2)
In v2, the *Field*
components now provide those values via snippets props to the children
snippet that you can use when needing to access those values.
Component Changes
Control
The Control
component now uses snippet props for attribute passing.
Before (v1)
The Control
component in v1 simply expose an attrs
slot prop that was spread onto the control element, like so:
After (v2)
In v2, the Control
component now provides those attributes via the controlProps
function which retrieves the props from the closest Control
component's context.
Alternatively, you can access the props via the Control
component's children
snippet props.
This change comes with Svelte's deprecation of <slot />
and slot props in favor of Snippets.
Composition API Changes
Form Field Access
getFormField
is now deprecated in favor of useFormField
with a new reactive getter pattern.
Before (v1)
In v1, the getFormField
function was used to get a reference to a form field's state for more advance composition patterns.
After (v2)
In v2, the getFormField
function is marked as deprecated and is aliased to the new useFormField
.
useFormField
returns getters for the various reactive states, rather than an object of stores. This means you should not destructure the object returned by useFormField
and instead use the getters directly.
See the useFormField documentation for more information.
Form Control Access
getFormControl
is deprecated in favor of useFormControl
with the new reactive getter pattern.
Before (v1)
In v1, the getFormControl
function was used to hook into the state of the closest parent Control
component. This function is useful for building custom components that may encompass both the control and the label.
After (v2)
In v2, the getFormControl
function is marked as deprecated and is aliased to the new useFormControl
.
useFormControl
returns getters for the various reactive states, rather than an object of stores. This means you should not destructure the object returned by useFormControl
and instead use the getters directly.
Migration Checklist
- Update all
asChild
instances to use thechild
snippet - Replace
el
bindings withref
- Update Field components to use
children
snippet props instead of slot props - Replace
getFormField
withuseFormField
- Replace
getFormControl
withuseFormControl