How to use Liveblocks Storage with React
In this guide, we’ll be learning how to use Liveblocks Storage with React using
the hooks from the @liveblocks/react package.
This guide uses TypeScript. Liveblocks can definitely be used without TypeScript. We believe typings are helpful to make collaborative apps more robust, but if you’d prefer to skip the TypeScript syntax, feel free to write your code in JavaScript
Sync and persist the state across client
Some collaborative features require a single shared state between all users—an
example of this would be a
collaborative design tool, with each shape having
its own state, or a form with shared inputs. In Liveblocks, this is where
storage comes in. Room storage automatically updates for every user on
changes, and unlike presence, persists after users disconnect.
Storage types
Our storage uses special data structures (inspired by CRDTs) to resolve all conflicts, meaning that state is always accurate. There are multiple storage types available:
- LiveObject- Similar to a JavaScript object.
- LiveList- An array-like ordered collection of items.
- LiveMap- Similar to a JavaScript Map.
Defining initial storage
To use storage, first define a type named Storage in liveblocks.config.ts.
In this example we’ll define a LiveObject called scientist, containing
first and last name properties.
Then, define the initial structure within RoomProvider.
Using storage
Once the default structure is defined, we can then make use of our storage. The
useStorage hook allows us to access an immutable version of our storage
using a selector function.
The two input values will now automatically update in a realtime as firstName
and lastName are modified by other users.
useStorage returns null during the initial loading because the storage is
loaded from the server. It can quickly become cumbersome to handle null
whenever we use useStorage, but we have some good new for you;
@liveblocks/react contains a
Suspense version of all
of our hooks.
Updating storage
The best way to update storage is through mutations. The useMutation hook
allows you to create reusable callback functions that modify Liveblocks state.
For example, let’s create a mutation that can modify the scientist’s name.
Inside this mutation we’re accessing the storage root, a LiveObject like
scientist, and retrieving a mutable copy of scientist with
LiveObject.get. From there, we can set the updated name using
LiveObject.set.
We can then call this mutation, and pass nameType and newName arguments.
If we take a look at this in the context of a component, we can see how to
combine useStorage to display the names, and useMutation to modify
them. Note that useMutation takes a dependency array, and works similarly to
useCallback.
All changes made within useMutation are automatically batched and sent to the
Liveblocks together. useMutation can also be used to retrieve and modify
presence too, giving you access to multiple parameters, not just storage.
Find more information in the Mutations section of our documentation.
Nested data structures
With Liveblocks storage, it’s possible to nest data structures inside each
other, for example scientist could hold a LiveList containing a list of
pets.
Because the useStorage selector converts your data structure into a normal
immutable JavaScript structure (made from objects, arrays, maps), pets can be
accessed directly with useStorage.
You can even reach into a LiveObject or LiveList and extract a property.
Improving storage performance
useStorage is highly efficient and only triggers a rerender when the value
returned from the selector changes. For example, the following selectors will
only trigger rerenders when their respective values change, and are unaffected
by any other storage updates.
However, selector functions must return a stable result to be efficient—if a new object is created within the selector function, it will rerender on every storage change.
To account for this, we can pass a shallow equality check function, provided
by @liveblocks/react:
Find more information in the How selectors work section of our documentation.
Using Suspense
If you’d like to use Suspense in your application, make sure to re-export our
hooks from "@liveblocks/react/suspense".
And then put a Suspense component right below the RoomProvider. This version
of useStorage never returns null, the loading fallback will be handled by
Suspense fallback.
If you’re using a framework that supports Server Side Rendering like
Next.js, you cannot use Suspense directly like this.
Liveblocks does not load the storage on the server by default, so the components
using useStorage will never be able to render. To keep the benefits from
Suspense, you should use ClientSideSuspense from  @liveblocks/react
instead of the normal Suspense from React like this: