Data mutations
Tell react to update
Just like setState(), we must make React aware of the any mutations so it can rerender.
Controller from useController provides this functionality in a type-safe manner. Controller.fetch() lets us trigger async mutations.
import { useSuspense } from '@data-client/react';import { TodoResource } from './api/Todo';function TodoDetail({ id }: { id: number }) {const todo = useSuspense(TodoResource.get, { id });const ctrl = useController();const updateWith = title => () =>ctrl.fetch(TodoResource.partialUpdate, { id }, { title });return (<div><div>{todo.title}</div><button onClick={updateWith('🥑')}>🥑</button><button onClick={updateWith('💖')}>💖</button></div>);}render(<TodoDetail id={1} />);
Reactive Data Client uses the fetch response to safely update all components. This not only more than doubles performance, but dramatically reduces server load that comes up sequential fetches.
Tracking imperative loading/error state
useLoading() enhances async functions by tracking their loading and error states.
import { useController } from '@data-client/react';
import { useLoading } from '@data-client/hooks';
function ArticleEdit() {
const ctrl = useController();
const [handleSubmit, loading, error] = useLoading(
data => ctrl.fetch(todoUpdate, { id }, data),
[ctrl],
);
return <ArticleForm onSubmit={handleSubmit} loading={loading} />;
}
React 18 version with useTransition
import { useTransition } from 'react';
import { useController } from '@data-client/react';
import { useLoading } from '@data-client/hooks';
function ArticleEdit() {
const ctrl = useController();
const [loading, startTransition] = useTransition();
const handleSubmit = data =>
startTransition(() => ctrl.fetch(todoUpdate, { id }, data));
return <ArticleForm onSubmit={handleSubmit} loading={loading} />;
}
Zero delay mutations
Controller.fetch call the mutation endpoint, and update React based on the response. While useTransition improves the experience, the UI still ultimately waits on the fetch completion to update.
For many cases like toggling todo.completed, incrementing an upvote, or dragging and drop a frame this can be too slow!
export class Todo extends Entity {id = 0;userId = 0;title = '';completed = false;pk() {return `${this.id}`;}}export const TodoResource = createResource({urlPrefix: 'https://jsonplaceholder.typicode.com',path: '/todos/:id',schema: Todo,optimistic: true,});
getOptimisticResponse is just like setState with an updater function. Using snap for access to the store to get the previous value, as well as the fetch arguments, we return the expected fetch response.
export const increment = new RestEndpoint({
path: '/api/count/increment',
method: 'POST',
name: 'increment',
getOptimisticResponse(snap) {
const { data } = snap.getResponse(getCount);
if (!data) throw new AbortOptimistic();
return {
count: data.count + 1,
};
},
});
Reactive Data Client ensures data integrity against any possible networking failure or race condition, so don't worry about network failures, multiple mutation calls editing the same data, or other common problems in asynchronous programming.