When you want to fetch something in Svelte, the recommended method is to put it in the load function.
I wrote an article about async fetching and why this could be important to work outside the load function.
Generally speaking, when SSR is not involved, you have better control of race conditions, error handling, and implementation outside the load function. While I agree in MOST situations you should use the load functions (with SvelteKit), this is not true for ALL situations.
Here is a simple resource
function to handle this:
// resource.svelte.ts
export let resource = <T>(
fn: () => Promise<T>,
initialValue?: T
) => {
const _rune = $state<{ value: T | undefined }>({
value: initialValue
});
$effect(() => {
fn().then((data) => {
_rune.value = data;
});
});
return _rune;
};
And you would use it like so in your component:
import { resource } from '$lib/resource.svelte';
...
const todo = resource<Todo>(() =>
fetch(`https://jsonplaceholder.typicode.com/todos/${id}`)
.then((response) => response.json())
);
And show the value with:
{todo.value}
Similar Concepts in Other Frameworks
- React Query
- Vue watch
- SolidJS createResource
- Qwik useResource$
- Angular 19 resource
- ngxtension derivedAsync
Now you can easily create a signal from an async resource! I hope something like this can get implemented in Svelte like $resource
one day.
J
Top comments (6)
This is awesome, thanks man
As a correction, the
load
function and SSR only exist in Sveltekit, so maybe the title should say Sveltekit. Remember that one can do Svelte in non-Sveltekit projects.Right, but this works in both Svelte and SvelteKit.
There is no
load
function unless you're in Sveltekit.Again, agreed, but you can use my
resource
function in both.Works in both though