Shared Hook State with SWR

SWR is a React hook for data fetching that features a cache for requests. This is generally used to share the response from API calls and deduplicate requests, but SWR is flexible enough to support another use case: shared hook state. 1

Let's look at an example of a useUsername hook:

const useUsername = () => {
return useState('')
}
const UsernameInput = () => {
const [username, setUsername] = useUsername()
return (
<div>
<input value={username} onChange={setUsername} />
</div>
)
}
const DisplayUsername = () => {
const [username] = useUsername()
return (
<span>Username: {username}</span>
)
}

This won't work, because each time we call useUsername, we receive a new instance of state. Updating the input won't affect what our DisplayUsername component renders.

Solving with Context

With React context, we typically lift the username state to the highest level and useContext to read the value in our components:

const UsernameContext = createContext()
const App = () => (
<UsernameProvider>
{/* ... */}
</UsernameProvider>
)
const UsernameProvider = ({ children }) => {
const [username, setUsername] = useState('')
return (
<UsernameContext.Provider value={[username, setUsername]}>
{children}
</UsernameContext.Provider>
)
}
const DisplayUsername = () => {
const [username] = useContext(UsernameContext)
return (
<span>Username: {username}
)
}

This will work, but in big applications you'll end up with a lot of context.

Solving with SWR

We can simulate useState with SWR by using the mutate function as our setState, and the config.initialData option the initial state. Now when we call mutate, the updated data will be reflected everywhere the hook is used.

import useSWR from 'swr'
const useUsername = () => {
const { data: username, mutate: setUsername } = useSWR('username', {
initialData: ''
})
return [username, setUsername]
}

It works, no context required. Every useUsername will share the same state, and calling setUsername will update the state across all uses of the hook.

useSharedState

We can go one step further and build a shared addition to useState:

const useSharedState = (key, initial) => {
const { data: state, mutate: setState } = useSWR(key, {
initialData: initial
})
return [state, setState]
}

Use it like useState, but pass a key as the first argument:

const [username, setUsername] = useSharedState('username', 'paco')
const [os, setOS] = useSharedState('os', 'macos')

SWR is mainly used to manage remote data requests, but it also provides a powerful cache and API that you can use to share local state between hooks. Kudos to Shu for his excellent work!


  1. Two components using the same React hook don't share state by default.

  2. SWR uses a client-side only cache, so your data won't persist between sessions or windows unless you keep external state like localStorage, at which point SWR may not be useful.