UNPKG

@tanstack/react-db

Version:

React integration for @tanstack/db

115 lines (114 loc) 4.28 kB
import { Collection, Context, GetResult, InferResultType, InitialQueryBuilder, LiveQueryCollectionConfig, NonSingleResult, QueryBuilder, SingleResult } from '@tanstack/db'; /** * Create a live query with React Suspense support * @param queryFn - Query function that defines what data to fetch * @param deps - Array of dependencies that trigger query re-execution when changed * @returns Object with reactive data and state - data is guaranteed to be defined * @throws Promise when data is loading (caught by Suspense boundary) * @throws Error when collection fails (caught by Error boundary) * @example * // Basic usage with Suspense * function TodoList() { * const { data } = useLiveSuspenseQuery((q) => * q.from({ todos: todosCollection }) * .where(({ todos }) => eq(todos.completed, false)) * .select(({ todos }) => ({ id: todos.id, text: todos.text })) * ) * * return ( * <ul> * {data.map(todo => <li key={todo.id}>{todo.text}</li>)} * </ul> * ) * } * * function App() { * return ( * <Suspense fallback={<div>Loading...</div>}> * <TodoList /> * </Suspense> * ) * } * * @example * // Single result query * const { data } = useLiveSuspenseQuery( * (q) => q.from({ todos: todosCollection }) * .where(({ todos }) => eq(todos.id, 1)) * .findOne() * ) * // data is guaranteed to be the single item (or undefined if not found) * * @example * // With dependencies that trigger re-suspension * const { data } = useLiveSuspenseQuery( * (q) => q.from({ todos: todosCollection }) * .where(({ todos }) => gt(todos.priority, minPriority)), * [minPriority] // Re-suspends when minPriority changes * ) * * @example * // With Error boundary * function App() { * return ( * <ErrorBoundary fallback={<div>Error loading data</div>}> * <Suspense fallback={<div>Loading...</div>}> * <TodoList /> * </Suspense> * </ErrorBoundary> * ) * } * * @remarks * **Important:** This hook does NOT support disabled queries (returning undefined/null). * Following TanStack Query's useSuspenseQuery design, the query callback must always * return a valid query, collection, or config object. * * ❌ **This will cause a type error:** * ```ts * useLiveSuspenseQuery( * (q) => userId ? q.from({ users }) : undefined // ❌ Error! * ) * ``` * * ✅ **Use conditional rendering instead:** * ```ts * function Profile({ userId }: { userId: string }) { * const { data } = useLiveSuspenseQuery( * (q) => q.from({ users }).where(({ users }) => eq(users.id, userId)) * ) * return <div>{data.name}</div> * } * * // In parent component: * {userId ? <Profile userId={userId} /> : <div>No user</div>} * ``` * * ✅ **Or use useLiveQuery for conditional queries:** * ```ts * const { data, isEnabled } = useLiveQuery( * (q) => userId ? q.from({ users }) : undefined, // ✅ Supported! * [userId] * ) * ``` */ export declare function useLiveSuspenseQuery<TContext extends Context>(queryFn: (q: InitialQueryBuilder) => QueryBuilder<TContext>, deps?: Array<unknown>): { state: Map<string | number, GetResult<TContext>>; data: InferResultType<TContext>; collection: Collection<GetResult<TContext>, string | number, {}>; }; export declare function useLiveSuspenseQuery<TContext extends Context>(config: LiveQueryCollectionConfig<TContext>, deps?: Array<unknown>): { state: Map<string | number, GetResult<TContext>>; data: InferResultType<TContext>; collection: Collection<GetResult<TContext>, string | number, {}>; }; export declare function useLiveSuspenseQuery<TResult extends object, TKey extends string | number, TUtils extends Record<string, any>>(liveQueryCollection: Collection<TResult, TKey, TUtils> & NonSingleResult): { state: Map<TKey, TResult>; data: Array<TResult>; collection: Collection<TResult, TKey, TUtils>; }; export declare function useLiveSuspenseQuery<TResult extends object, TKey extends string | number, TUtils extends Record<string, any>>(liveQueryCollection: Collection<TResult, TKey, TUtils> & SingleResult): { state: Map<TKey, TResult>; data: TResult | undefined; collection: Collection<TResult, TKey, TUtils> & SingleResult; };