UNPKG

@tanstack/svelte-db

Version:

Svelte integration for @tanstack/db

238 lines (186 loc) 5.7 kB
--- name: svelte-db description: > Svelte 5 bindings for TanStack DB. useLiveQuery with Svelte 5 runes ($state, $derived, $effect). Dependency arrays use getter functions (() => value) for reactive props. Direct destructuring breaks reactivity — use dot notation or wrap with $derived. Conditional queries via returning undefined/null. Import from @tanstack/svelte-db (re-exports all of @tanstack/db). type: framework library: db framework: svelte library_version: '0.6.0' requires: - db-core sources: - 'TanStack/db:docs/framework/svelte/overview.md' - 'TanStack/db:packages/svelte-db/src/useLiveQuery.svelte.ts' --- This skill builds on db-core. Read it first for collection setup, query builder, and mutation patterns. # TanStack DB — Svelte 5 ## Setup ```svelte <script lang="ts"> import { useLiveQuery, eq, not } from "@tanstack/svelte-db" const todosQuery = useLiveQuery((q) => q .from({ todo: todoCollection }) .where(({ todo }) => not(todo.completed)) .orderBy(({ todo }) => todo.created_at, "asc") ) </script> {#if todosQuery.isLoading} <div>Loading...</div> {:else} <ul> {#each todosQuery.data as todo (todo.id)} <li>{todo.text}</li> {/each} </ul> {/if} ``` `@tanstack/svelte-db` re-exports everything from `@tanstack/db`. ## Hook ### useLiveQuery Returns a reactive object with getter properties: ```ts // Query function — access via dot notation const query = useLiveQuery((q) => q.from({ todo: todoCollection })) // query.data, query.isLoading, query.status, etc. // With reactive deps — use GETTER FUNCTIONS let minPriority = $state(5) const query = useLiveQuery( (q) => q .from({ todo: todoCollection }) .where(({ todo }) => gt(todo.priority, minPriority)), [() => minPriority], ) // Config object const query = useLiveQuery({ query: (q) => q.from({ todo: todoCollection }), gcTime: 60000, }) // Pre-created collection const query = useLiveQuery(preloadedCollection) ``` ## Svelte-Specific Patterns ### Destructuring with $derived ```ts // WRONG — breaks reactivity const { data, isLoading } = useLiveQuery(...) // CORRECT — use dot notation const query = useLiveQuery(...) // Access: query.data, query.isLoading // CORRECT — wrap with $derived if you need destructuring const query = useLiveQuery(...) const { data, isLoading } = $derived(query) ``` ### Props in dependency arrays ```svelte <script lang="ts"> let { userId }: { userId: number } = $props() const todosQuery = useLiveQuery( (q) => q.from({ todo: todoCollection }) .where(({ todo }) => eq(todo.userId, userId)), [() => userId] ) </script> ``` ## Includes (Hierarchical Data) When a query uses includes (subqueries in `select`), each child field is a live `Collection` by default. Subscribe to it with `useLiveQuery` in a subcomponent: ```svelte <!-- ProjectList.svelte --> <script lang="ts"> import { useLiveQuery, eq } from "@tanstack/svelte-db" import IssueList from "./IssueList.svelte" const projectsQuery = useLiveQuery((q) => q.from({ p: projectsCollection }).select(({ p }) => ({ id: p.id, name: p.name, issues: q .from({ i: issuesCollection }) .where(({ i }) => eq(i.projectId, p.id)) .select(({ i }) => ({ id: i.id, title: i.title })), })) ) </script> {#each projectsQuery.data as project (project.id)} <div> {project.name} <IssueList issuesCollection={project.issues} /> </div> {/each} ``` ```svelte <!-- IssueList.svelte — subscribes to the child Collection --> <script lang="ts"> import { useLiveQuery } from "@tanstack/svelte-db" let { issuesCollection } = $props() const issuesQuery = useLiveQuery(issuesCollection) </script> {#each issuesQuery.data as issue (issue.id)} <li>{issue.title}</li> {/each} ``` With `toArray()`, child results are plain arrays and the parent re-emits on child changes: ```ts import { toArray, eq } from '@tanstack/svelte-db' const projectsQuery = useLiveQuery((q) => q.from({ p: projectsCollection }).select(({ p }) => ({ id: p.id, name: p.name, issues: toArray( q .from({ i: issuesCollection }) .where(({ i }) => eq(i.projectId, p.id)) .select(({ i }) => ({ id: i.id, title: i.title })), ), })), ) // project.issues is a plain array — no subcomponent needed ``` See db-core/live-queries/SKILL.md for full includes rules (correlation conditions, nested includes, aggregates). ## Common Mistakes ### CRITICAL Passing values instead of getter functions in deps Wrong: ```ts let filter = $state('active') const query = useLiveQuery( (q) => q .from({ todo: todoCollection }) .where(({ todo }) => eq(todo.status, filter)), [filter], ) ``` Correct: ```ts let filter = $state('active') const query = useLiveQuery( (q) => q .from({ todo: todoCollection }) .where(({ todo }) => eq(todo.status, filter)), [() => filter], ) ``` In Svelte 5, deps must be getter functions. Passing values directly captures them at creation time — changes won't trigger re-execution. Source: docs/framework/svelte/overview.md ### HIGH Direct destructuring breaks reactivity Wrong: ```ts const { data, isLoading } = useLiveQuery((q) => q.from({ todo: todoCollection }), ) ``` Correct: ```ts const query = useLiveQuery((q) => q.from({ todo: todoCollection })) // Use query.data, query.isLoading ``` Direct destructuring captures values at creation time. Use dot notation or wrap with `$derived`. Source: packages/svelte-db/src/useLiveQuery.svelte.ts See also: db-core/live-queries/SKILL.md — for query builder API. See also: db-core/mutations-optimistic/SKILL.md — for mutation patterns.