@mittwald/react-use-promise
v2.6.0
Published
Simple and declarative use of Promises in your React components. Observe their state and refresh them in various advanced ways.
Downloads
3,308
Readme
React Use Promise
Simple and declarative use of Promises in your React components. Observe their state and refresh them in various advanced ways.
Now with built-in support for 🌐 HTTP!
import { Suspense } from "react";
import { usePromise, refresh } from "@mittwald/react-use-promise";
// Async loader 👇 function
const loadNewsItem = async (id) => {
const res = await fetch(
`https://hacker-news.firebaseio.com/v0/item/${id}.json`,
);
return res.json();
};
const NewsItem = ({ id }) => {
const news = usePromise(loadNewsItem, [id], {
// ✨ Use the async loader 👆 function with its ID-parameter
tags: [`news/${id}`],
// Use tags 🏷️ with support for "tree structures" 🌳
});
// Do not care about any loading states – just use the result 🤩
return (
<li>
{news.by}: {news.title}
</li>
);
};
const App = () => {
const reloadAllNews = () => {
// ✨ Reload Promises 🔄 by using tags and glob pattern *️⃣
refresh({ tag: "news/*" });
};
return (
// The Suspense component will render a fallback️
// if any child component is in loading state ⏱️
<Suspense fallback={<div>Loading...</div>}>
<NewsList>
<NewsItem id={1000} />
<NewsItem id={1001} />
<NewsItem id={1002} />
</NewsList>
<button onClick={reloadAllNews}>Reload news</button>
</Suspense>
);
};
Features
- Simple and declarative use of Promises, loading- and error views
- Built-in support to load data with HTTP resources
- Works with data fetching via Next.js server actions
- Auto-refresh after timeout or on window focus
- Type-Safe API
- No "double-loading" when using "same" Promise in different places in your app
- Caching with support for custom tags
- Opt-out Suspense-based loading
- Cache invalidation with glob support
- Observable loading state
- Set up lazy-loading "async resources"
with the alternative
getAsyncResource
-API and pass them as prop to child components - "Resourceify" any async function
- Fully tested and well-structured TypeScript code-base
Table of Contents
- Installing
- Terminology
- API
- Caching
- Tags
- Refreshing resources
- HTTP Resources
- Lazy loading with Async Resources
- Defining loading views
- Opt-Out Suspense
- Opt-Out Loading
- Error handling
- Best practices
- Migration guides
Installing
With npm:
$ npm install @mittwald/react-use-promise
With Yarn:
$ yarn add @mittwald/react-use-promise
Terminology
(Async) resource
An async resource (or just resource) represents something that has to be loaded asynchronously. Asynchronous loading involves different states, like "loading" or "loading is done", that should be reflected in the UI. Basically async resources encapsulating Promises and equipping them with relevant features like observing the Promises state or caching the result.
Async loader (function)
The async loader function is simply a function returning a Promise and thus acts as the basic input for async resources.
API
usePromise
Get the result of a Promise by passing an async loader with the relevant loader
parameters and an optional configuration to the usePromise
hook.
usePromise(asyncLoader, loaderParameters, options?)
Returns: the value of the async resource
For possible options see Options section.
import { usePromise } from "@mittwald/react-use-promise";
// 1. Inline loader function
const response = usePromise(() => axios("/user/12345"), []);
// 2. Loader function with parameters
const getUser = (id) => axios(`/user/${id}`);
const response = usePromise(getUser, [12345]); // 👈 loader parameters as array
// 3. With options
const response = usePromise(getUser, [12345], {
autoRefresh: {
seconds: 30,
},
});
getAsyncResource
This is an alternative and more advanced API to the usePromise
hook. For
details see
Lazy Loading with Async Resources. The
usePromise
hook uses the getAsyncResource
-API under the
hood and is basically just a shortcut for
getAsyncResource(asyncLoader, loaderParameters, options).use()
.
Get an async resource by passing an async loader with the relevant loader
parameters and an optional configuration to the getAsyncResource
function.
getAsyncResource(asyncLoader, loaderParameters, options?)
Returns: Async resource
For possible options see Options section.
import { getAsyncResource } from "@mittwald/react-use-promise";
// 1. Inline loader function
const userResource = getAsyncResource(() => axios(`/user/123`), []);
// 2. Loader function with parameters
const getUser = (id) => axios(`/user/${id}`);
const userResource = getAsyncResource(getUser, [123]);
// 3. With options
const userResource = getAsyncResource(getUser, [123], {
tags: ["api"],
});
// 4. Usage in factory function
const getUserResource = (id) => getAsyncResource(getUser, [id]);
Async resource
The async resource returned by getAsyncResource
has the following API.
.use(options?)
For possible options see Options section.
Returns: the value of the async resource, or the result object when
useSuspense: false
(see Opt-out Suspense)
Calling the use
method will actually start the loading process and returns the
value of the async resource, once it is loaded. Everytime the resource is
refreshed, the used value automatically updates itself.
import getUserResource from "../resources/user";
const Username = ({ id }) => {
const user = getUserResource(id).use(); // 👈 using the resource value
return <>{user.name}</>;
};
.refresh()
Calling the refresh
method will clear the cached resource value and trigger a
reload, if the resource is being used in any mounted component.
import getScoreResource from "../resources/score";
const Score = ({ matchId }) => {
const scoreResource = getScoreResource(matchId);
const score = scoreResource.use();
const reloadScore = () => scoreResource.refresh(); // 👈 refresh the resource
return (
<ScoreBox>
<Score>
{score.home} - {score.guest}
</Score>
<button onClick={reloadScore}>Reload</button>
</ScoreBox>
);
};
.watchState()
Returns: "void" | "loading" | "loaded" | "error"
You can watch the resources state by calling the watchState
method.
import getScoreResource from "../resources/score";
const Score = ({ matchId }) => {
const scoreResource = getScoreResource(matchId);
const scoreResourceState = scoreResource.watchState();
const score = scoreResource.use();
const scoreIsLoading = scoreResourceState === "loading";
return (
<ScoreBox>
<Score>
{score.home} - {score.guest}
</Score>
<LoadingSpinner visible={scoreIsLoading} />
</ScoreBox>
);
};
Options
You can configure usePromise
and getAsyncResource
with the following
options:
autoRefresh – only supported by hooks (use*
)
Type:
Duration like object
Default: undefined
When a duration is configured, the resource will automatically be refreshed in the provided interval. If the same resource has multiple auto-refresh intervals, the shortest interval will be used.
autoRefresh: {
seconds: 30;
}
refreshOnWindowFocus – only supported by hooks (use*
)
Type: boolean
Default: false
Set this option to true
, if the resource should automatically be refreshed, if
the window is re-focused.
keepValueWhileLoading – only supported by hooks (use*
)
Type: boolean
Default: true
If true
, the previously loaded value will be returned during refresh is in
progress. The loading view will only be triggered during initial load.
If false
, the loading view will always be triggered – during initial load and
refresh as well.
tags
Type: string[]
Default: undefined
With this option you can assign tags to resources. Tags allow you to be expressive and flexible when accessing resources, e.g. when you need to refresh multiple resources at once.
You can find details about how to use tags in the Tags section.
tags: ["hackernews", "getRequest"];
loaderId
Type: string
Default: "null"
It is very unlikely that you ever need to use this option, but to get around the "same code" issue (see "Caveats of default storage key generation"), you can set an explicit loader ID, that identifies the loader function.
useSuspense – only supported by hooks (use*
)
Type: boolean
Default: true
Set this to false
to opt-out Suspense-based loading behavior. See
Opt-Out Suspense for more details.
refresh
If you do not have direct access to the resource that should be refreshed, you can use the global refresh method.
refresh()
Refreshes all resources.
import { refresh } from "@mittwald/react-use-promise";
export const Menu = () => (
<Menu>
<MenuItem url="/dashboard" />
<MenuItem url="/projects" />
<MenuItem url="/profile" />
<MenuButton onClick={() => refresh()}>Refresh all data</MenuButton>
</Menu>
);
refresh(options)
The refresh method takes an option object to do a more selective refresh.
The following options are supported:
tag
(string
): Refreshes all resources matching the given tag. Glob pattern are supported here. See also the Hierarchical Tags section.error
(true | error
): Set this option totrue
, to refresh all resource with errors. Set this option to an error instance, to refresh all resources with a matching error. See the Error handling section for more details.
resourceify
resourceify
creates a factory function for async resources based on given
async loader functions.
resourceify(asyncLoader)
Example of how to create factory functions for your async loaders:
import { resourceify } from "@mittwald/react-use-promise";
const getUser = (id) => axios(`/user/${id}`);
// Creates a function to get user resources
const getUserResource = resourceify(getUser);
// `myUser` is an async resource
const myUser = getUserResource(["me"], { refresh: { seconds: 30 } });
myUser.use();
// Factory method for HTTP GET-Requests via Axios
import axios from "axios";
export const getHttpResource = resourceify(axios.get);
Caching
Caching the result of async loader functions is essential to make this library work. Without it, components will end-up in an endless render-loop, since the re-render after finished loading will trigger the loader function again.
To break this loop, the result (and even the error) of the loader function is cached inside the resource instance. If a cached result exists, the loader must not be called again and the cached value is used instead.
Challenges concerning Caching
This caching approach comes with two essential issues one has to care about:
- creating unique cache keys to store results
- providing flexible ways of cache invalidation
Resource store
Every time when usePromise
resp. getAsyncResource
is called, either a new
resource is created or an existing resource is taken from the resource store. If
a resources has loaded once, it exists in the store and contains the cached
result of the async loader function.
It is noticeable that not the raw result is cached in some "result cache" – it is the resource the keeps the cached result which itself is stored in the resource store.
Creating unique storage keys
Basically you do not have to care about storage keys at all, but there are some odd situations where the default storage key generation fails, and produces conflicting keys. If so, you probably might not get the expected resource. When you are experiencing such issues, you better take a look at this section.
Same resource, same storage key
When creating storage keys, the same key should be formed for async resources considered as "same". An async resource is "the same" compared to another resource, if:
- the loader function is the same
- the parameters used to load the resource are the same
Examples:
const getUser = (id) => axios(`/user/${id}`);
const getPost = (id) => axios(`/post/${id}`);
// Same storage key 🍎🍎
const response1 = usePromise(getUser, [12345]);
const response2 = usePromise(getUser, [12345]);
// Same storage key 🍐🍐
const response3 = usePromise(() => getUser(12345), []);
const response4 = usePromise(() => getUser(12345), []);
// Different storage keys 🍎🍋
const response5 = usePromise(getUser, [12345]);
const response6 = usePromise(getUser, [56789]);
// Different storage keys 🍎🍉
const response7 = usePromise(getUser, [12345]);
const response8 = usePromise(getPost, [12345]);
// Different storage keys 🍎🌶️
const response9 = usePromise(() => getUser(12345), []);
const response10 = usePromise(() => getUser(12344 + 1), []);
Caveats of default storage key generation
In some odd situations the default storage key generation fails and may generate conflicting keys. To evaluate these situations, you should know some details about the default behaviour, which is basically implemented like this:
import { hash } from "object-code";
const storageKeyObject = {
loaderFunction,
parameters,
};
const storageKey = hash(storageKeyObject);
As you can see, the implementation heavily relies on the
object-code
library, which
is a blazing fast hash code generator that supports every possible javascript value.
When it comes to "hashing" a function (in this case the async loader function),
object-code
uses toString()
to get the function code and uses it for
further hashing. This is the point where problems may arise. When two function
using the same code, they must not necessarily behave the same. Think of
functions using variables from their parent scope (🤯).
The following example may result in inconsistencies. (But only if users and posts share same the IDs.)
// File Username.jsx
import repo from "./user";
const loadUserById = (id) => repo.loadById(id);
// same code 👆, but different repo 😧
export const Username = ({ id }) => {
const user = usePromise(loadUserById, [id]);
//👆 may be a cached post
return <>{user.name}</>;
};
// File PostTitle.jsx
import repo from "./post";
const loadPostById = (id) => repo.loadById(id);
// same code 👆, but different repo 😧
export const PostTitle = ({ id }) => {
const post = usePromise(loadPostById, [id]);
//👆 may be a cached user
return <>{post.title}</>;
};
Using explicit loader IDs
It is very unlikely that you ever need to use this option, but to get around the "same code" issue demonstrated above, you can set explicit loader IDs in the options object.
This example shows how to set explicit loader IDs:
// ...
const post = usePromise(loadPostById, [id], {
loaderId: "loadPostById", // explicit loader ID 😮💨
});
// ...
Tags
You can assign tags to resources. A tag is simply a string that classifies the resource. In advanced scenarios you might want to assign multiple tags to one resource, expressing the resource matches different classifications.
Using relevant tags creates the possibility to be very expressive and flexible when it comes to accessing multiple resources at once, e.g. when you need to refresh them.
For example, you might use tags to refresh some specific resources, when a backend event occurs.
// File components/Chat.jsx
import { usePromise } from "@mittwald/react-use-promise";
const Chat = ({ chatId }) => {
const messages = usePromise(loadChatMessages, [chatId], {
tags: [`chat/${chatId}`],
});
return (
<div>
{messages.map((msg) => (
<ChatMessage message={msg} key={msg.id} />
))}
</div>
);
};
// File setup.js
import { refresh } from "@mittwald/react-use-promise";
backendEventListener.on("chatUpdated", (chatId) => {
refresh({
tag: `chat/${chatId}`,
});
});
Hierarchical Tags
Tags are supporting a tree structure, compared to paths in filesystems or URLs
(e.g. chat/12345/messages
). This creates the possibility to structure your
resources into hierarchical classes. Combined with
glob support in cache invalidation,
you can invalidate resources matching a certain level in the tree.
import { refresh } from "@mittwald/react-use-promise";
refresh({
tag: `chat/12345/**/*`,
});
// Matches everything for the chat 12345
// - chat/12345/messages/1
// - chat/12345/messages/2
// - chat/12345/metaData
refresh({
tag: `chat/12345/messages/*`,
});
// Matches all messages for the chat 12345
// - chat/12345/messages/1
// - chat/12345/messages/2
Refreshing resources
If a resource has a cached result, the async loader function must not be called
again and the cached value is used instead. To invalidate the cached value you
can call the refresh()
method on the resource or use the global
refresh()
method. As a result the async loader function will
automatically be called again and a component update is triggered when the
result is available.
Refreshing resources that are not used in any mounted component, will suspend the reload until the resource is actually used the next time.
You might also use the auto-refresh mechanisms after timeout or refocus window.
import { refresh } from "@mittwald/react-use-promise";
🌐 HTTP Resources
A major use case of this library might be to load data from some HTTP-API. This
is exactly why a preconfigured HTTP resource is included in this package. The
simplest way is to use the useHttpData()
method to GET data from a given URL.
import { useHttpData } from "@mittwald/react-use-promise/http";
const NewsItem = ({ id }) => {
const news = useHttpData(
`https://hacker-news.firebaseio.com/v0/item/${id}.json`,
);
return (
<li>
{news.by}: {news.title}
</li>
);
};
Important: As this feature requires the axios
package you need to include
it in your dependencies!
yarn add axios
Request config
Under the hood the popular HTTP client Axios is used to perform the request. To make all Axios config options available to you, the complete config object can be passed as the second parameter to all HTTP functions.
const videoGames = useHttpData<Article[]>(`/articles`, {
params: {
catagory: "video-games",
},
});
Default request config
To set the default config for each request (e.g the baseURL
or an
Authorization
header) you can use the
axios.defaults
object.
import axios from "axios";
axios.defaults.baseURL = "https://api.foo.org/v2";
Caching and refreshing of HTTP Resources
HTTP Resources are cached by the identity of the request config – meaning, different request configs result in different HTTP Resources.
Cache tags added to HTTP Resources:
http/method/(HTTP METHOD)
http/uri/(REQUEST URI)
This lets you refresh HTTP Resources with a certain path.
import { refresh } from "@mittwald/react-use-promise";
function refreshArticles() {
refresh({
tag: "http/uri/**/articles",
});
}
HTTP API
useHttpData(url, requestConfig?, usePromiseOptions?)
Fetches data from a given URL with an optional request configuration.
Returns: the parsed body data of the response
url
The URL to perform the request on
requestConfig?
usePromiseOptions?
For possible options see Options section.
useHttp(url, requestConfig?, usePromiseOptions?)
Get the response from a given URL with an optional request configuration.
Returns: an Axios response object
url
The URL to perform the request on
requestConfig?
usePromiseOptions?
For possible options see Options section.
getHttpResource(url, requestConfig?, usePromiseOptions?)
Creates an Async Resource that performs an HTTP request as the loader function.
Returns: an Axios response object
url
The URL to perform the request on
requestConfig?
usePromiseOptions?
For possible options see Options section.
😴 Lazy loading with Async Resources
When constructing your React app, you might get to the point, where you must decide whether to
- load something from your API and pass the result into some child component,
- or the child component should load the data itself by some ID from its props.
Since loading async resources involves different loading states, it is a good advice, to react on these state changes, as deep as possible in the component tree, to avoid needless re-renderings of parent components.
💅 Presentational vs. 📦 Container Components
The components deep in the tree are usually some small components, like the
<Avatar />
component showing an avatar image. These kind of components do have
a clear focus on the visual representation and should not be polluted by some
loading state handling. They are often referred to as Presentational
Components. They can be developed as "standalone" and showcased in Storybooks,
without the need of any API.
When the time has come to bring your Presentational Components to life, you can "wrap" the component in a container that "connects" it with real data. This is the job of Container Components. They care about loading data and compose Presentational Components to display the data.
Example Presentational Component:
// Props needed for presentation
interface Props {
imageUrl?: string;
size?: number;
}
const Avatar: FC<Props> = ({ imageUrl, size }) => {
return <Image round url={imageUrl} size={size} />;
};
Example Container Component:
interface Props {
// ID is needed to load the user
userId: string;
// optional presentational props are possible
size?: number;
}
const UserAvatar: FC<Props> = ({ userId, size }) => {
const profile = usePromise(api.loadUserProfile, [userId]);
return <Avatar imageUrl={profile.avatarUrl} size={size} />;
};
Async Resources as prop
As an alternative to the user ID, you might pass an Async User Resource as prop to your Container Component. This makes the component more flexible, because it is not tied to the actual loading method.
For instance if user profiles can be loaded via multiple API methods, the
<UserAvatar />
component can still be used without any changes.
It is noticeable that creating Async Resources in any parent component does
not trigger the loading process. Therefore, the .use()
method can be used
in the child components, where the data is actually needed.
Alternative Container Component with lazy loading:
// ...
import { AsyncResource, getAsyncResource } from "@mittwald/react-use-promise";
interface Props {
// Async User Resource passed as prop
user: AsyncResource<UserData>;
size?: number;
}
const UserAvatar: FC<Props> = ({ user, size }) => {
// .use() 🔔 triggers loading when needed
const profile = user.use();
return <Avatar imageUrl={profile.avatarUrl} size={size} />;
};
const UserProfileMenu: FC = () => {
// 😴 Creating the Async Resource does not trigger loading
const myProfile = getAsyncResource(api.loadMyUserProfile, []);
return (
<Menu>
<MenuHeader>
<UserAvatar user={myProfile} size={128} />
</MenuHeader>
{/* menu items*/}
</Menu>
);
};
Defining loading views
When using @mittwald/react-use-promise
(or other Suspense-based approaches)
you have to define a loading boundary (Reacts <Suspense />
component) to
display the loading view. Loading boundaries are quite similar to error
boundaries. They "catch" active loading processes in any child component and
render a fallback instead. The benefits of this pattern are:
- No pollution of handling loading states over and over again in your components
- Definition of loading views, where you need them in the component tree
- Flexible level of loading views: Sometimes it is less distracting to use a single loading view for a larger section of your app, but sometimes small pieces of async data should not keep off parts of your app from rendering.
- Declarative and easy to understand
Example of how to define loading views:
const App = () => (
<Suspense fallback={<FullScreenLoadingSpinner />}>
{/* ... */}
<MainLayout>
<Suspense fallback={<MainMenuLoadingSkeleton />}>
<MainMenu />
</Suspense>
<Suspense fallback={<MainContentLoadingSkeleton />}>
<MainContent />
</Suspense>
</MainLayout>
{/* ... */}
</Suspense>
);
Gotchas when defining "built-in" loading views
When you are using .use()
or usePromise()
in your component, it can not
define its own loading boundary – at least not for directly used Async
Resources.
In this example the fallback component will not be shown:
const UserAvatar = ({ userResource, size }) => {
const user = userResource.use();
// 👆 any code below this line will not be executed 🙅 until the async loader is done.
const loadingView = <AvatarSkeleton size={size} />;
return (
<Suspense fallback={loadingView}>
{/* This fallback 👆 is not rendered for the used resource from above. 😢 */}
<Avatar imageUrl={user.avatarUrl} size={size} />
</Suspense>
);
};
The following approaches can help you to solve this issue:
- Split up your component into two separate components – one private with the regular rendering, and another one wrapping it with a loading boundary while forwarding props to it.
const Component = ({ userResource, size }) => {
const user = userResource.use();
return <Avatar imageUrl={user.avatarUrl} size={size} />;
};
export const UserAvatar = (props) => {
const loadingView = <AvatarSkeleton size={props.size} />;
return (
<Suspense fallback={loadingView}>
<Component {...props} />
</Suspense>
);
};
- Opt-out Suspense-based loading behavior.
const UserAvatar = ({ userResource, size }) => {
const user = userResource.use({ useSuspense: false });
// `user` is an object with `isSet` and `value` properties
if (!user.isSet) {
return <AvatarSkeleton size={size} />;
}
// `value` only exists when `isSet === true`
return <Avatar imageUrl={user.value.avatarUrl} size={size} />;
};
- Use a Higher Order Component (HOC) that enhances your regular implementation with a loading boundary. This HOC could optionally add an error boundary.
const UserAvatar = withLoadingBoundary(({ userResource, size }) => {
const user = userResource.use();
return <Avatar imageUrl={user.avatarUrl} size={size} />;
}, AvatarSkeleton);
- Use a render component where you access the resource:
const UserAvatar = ({ userResource, size }) => {
const loadingView = <AvatarSkeleton size={size} />;
return (
<Suspense fallback={loadingView}>
<Render>
{() => {
const user = userResource.use();
return <Avatar imageUrl={user.avatarUrl} size={size} />;
}}
</Render>
</Suspense>
);
};
Opt-Out Suspense
When you want to react explicitly on the loading-state, the suspense-based loading behavior is a little bit obstructive, e.g. when you want to define components with built-in loading views.
You can opt-out Suspense by setting the useSuspense
option to false
. When
Suspense is disabled, calling .use()
resp. usePromise()
will not trigger any
<Suspense />
component. Instead, a result object is returned, containing
loading information and the eventual value.
Example of how the opt-out behaves:
const message = usePromise(loadMessage, ["12345"], { useSuspense: false });
if (!message.hasValue) {
return <LoadingView />;
}
return (
<MessageView message={message.value} activitySpinner={message.isLoading} />
);
Return object with disabled Suspense
The object returned by .use()
resp. usePromise()
has the following
properties, when Suspense is disabled.
isLoading
: Istrue
when the resource is loading or reloading.false
otherwise.hasValue
: Istrue
when a resource value is available, including the "old" value when reloading.false
otherwise.value
: Contains the resource value. Is only available ifhasValue
istrue
.maybeValue
: Contains the resource value, when the resource has loaded.undefined
otherwise.
Examples of result objects:
// Loading the first time or reloading with keepValueWhileLoading disabled
({
isLoading: true,
maybeValue: undefined,
hasValue: false,
});
// When value has loaded
({
isLoading: false,
maybeValue: "Foo",
value: "Foo",
hasValue: true,
});
// When value is reloading
({
isLoading: true,
maybeValue: "Foo",
value: "Foo",
hasValue: true,
});
Opt-Out Loading
You probably know that you should
not call hooks conditionally.
But what if the loader parameters you want to use e.g. in usePromise
are
optional properties, and the loader function requires them? In this case just
use null
as parameters and loading is disabled!
😈 Bad:
const Username = ({ id }) => {
if (!id) {
return null;
}
// 💥 This will throw a React error because `usePromise` is called conditionally!
const user = usePromise(loadUseProfile, [id]);
return <>{user.name}</>;
};
😇 Good:
const Username = ({ id }) => {
// When required loader parameters can be undefined: use null 👇
const user = usePromise(loadUseProfile, id ? [id] : null);
// ⚠️ In this case, `user` can also be `undefined`!
return <>{user ? user.name : "Unknown"}</>;
};
Error handling
Errors that occur in any async loader function can be handled with regular error boundaries. With only one exception. When you have implemented a retry-render-mechanism in your error fallback view, the erroneous resource has to be refreshed as well before re-rendering.
When you use the popular
react-error-boundary
package, you can utilize its onReset
property to refresh all erroneous
resources, while resetting the error view.
Example of refreshing erroneous resources on reset:
// ...
import { refresh } from "@mittwald/react-use-promise";
import { ErrorBoundary } from "react-error-boundary";
const App = () => (
<MainLayout>
<MainMenu />
<ErrorBoundary
fallback={<MainContentErrorFallback />}
onReset={() => {
refresh({ error: true });
}}
>
<Suspense fallback={<MainContentLoadingSkeleton />}>
<MainContent />
</Suspense>
</ErrorBoundary>
</MainLayout>
);
Best practices
- Use async resources as deep in the component tree as possible (see Lazy loading with Async Resources).
- In larger applications use factory methods for your async resources. The
resourceify
function helps you to quickly set up your factory methods. - Add tags to your resources for a nuanced refreshing. You can even listen on backend-events (maybe pushed via a socket-connection) to invalidate the resources used in your session.
- Use models for a coherent interaction with your business models. This is a good practice in general, when your app has to deal with a large business model.
// File resources/user.ts
import { getUserProfile } from "../resources/user";
import { resourceify } from "@mittwald/react-use-promise";
export const getUserProfile = resourceify(apiClient.loadUserProfile);
// File models/UserProfile.ts
import { getUserProfile } from "../resources/user";
// ...
export class UserProfile {
public readonly id: string;
public readonly firstName: string;
public readonly lastName: string;
public constructor(data: UserProfileApiData) {
this.id = data.id;
this.firstName = data.firstName;
this.lastName = data.lastName;
}
public static useLoadById(id: string): UserProfile {
const data = getUserProfile([id]).use();
return new UserProfile(data);
}
public getFullName(): string {
return `${this.firstName} ${this.lastName}`;
}
public async updateName(firstName: string, lastName: string): Promise<void> {
await apiClient.updateProfile({
firstName,
lastName,
});
}
}
// File components/UserProfileName.tsx
import { UserProfile } from "../models/UserProfile";
export const UserProfileName: FC<{ id: string }> = (props) => {
const user = UserProfile.useLoadById(props.id);
return <>{user.getFullName()}</>;
};
Migration guides
V1 to V2
- For more naming consistency the
.watch()
method of the Async Resource has changed to.use()
. Replace all usages ofwatch()
withuse()
.