Recoil has been making waves in the React community lately, and for good reason. As a state management library, it offers a fresh approach to handling complex application states. I’ve been diving deep into Recoil recently, and I’m excited to share what I’ve learned.
At its core, Recoil introduces the concept of atoms and selectors. Atoms are units of state that components can subscribe to, while selectors allow you to derive state based on other atoms or selectors. This approach feels intuitive and aligns well with React’s component-based architecture.
One thing I love about Recoil is how it simplifies sharing state across components. Gone are the days of prop drilling or complex context setups. With Recoil, you can easily access and modify state from any component, regardless of where it sits in the component tree.
Let’s take a look at a simple example to get a feel for how Recoil works:
import { atom, useRecoilState } from 'recoil';
const counterState = atom({
key: 'counterState',
default: 0,
});
function Counter() {
const [count, setCount] = useRecoilState(counterState);
return (
<div>
<p>Count: {count}</p>
<button onClick={() => setCount(count + 1)}>Increment</button>
</div>
);
}
In this example, we define a counterState atom and use the useRecoilState hook to read and update its value. It’s clean, simple, and doesn’t require any complex setup.
But Recoil really shines when dealing with more complex state scenarios. Let’s say we want to derive some state based on our counter. We can use a selector for that:
import { selector, useRecoilValue } from 'recoil';
const evenOddSelector = selector({
key: 'evenOddSelector',
get: ({get}) => {
const count = get(counterState);
return count % 2 === 0 ? 'Even' : 'Odd';
},
});
function EvenOddDisplay() {
const evenOdd = useRecoilValue(evenOddSelector);
return <p>The count is {evenOdd}</p>;
}
Here, we’re using a selector to derive whether the count is even or odd. The EvenOddDisplay component can then use this derived state without needing to know about the underlying counterState.
One of the things that initially drew me to Recoil was its focus on performance. By default, Recoil only re-renders components that actually need to update based on state changes. This can lead to significant performance improvements, especially in larger applications.
Recoil also plays nicely with React’s concurrent mode. It’s built with async operations in mind, making it a great choice for applications that need to handle complex data fetching scenarios.
Speaking of async operations, Recoil has a neat feature called loadable. This allows you to handle async state in a way that feels natural and React-like. Here’s a quick example:
const userDataQuery = selector({
key: 'userDataQuery',
get: async () => {
const response = await fetch('https://api.example.com/user');
return response.json();
},
});
function UserData() {
const userData = useRecoilValueLoadable(userDataQuery);
switch (userData.state) {
case 'hasValue':
return <div>Welcome, {userData.contents.name}!</div>;
case 'loading':
return <div>Loading...</div>;
case 'hasError':
return <div>Error: {userData.contents.message}</div>;
}
}
This approach to handling async state feels much cleaner than traditional methods involving useEffect and multiple state variables.
Another feature of Recoil that I’ve found incredibly useful is atom families. These allow you to create a collection of atoms that share similar behavior. It’s perfect for scenarios where you need multiple instances of similar state, like a list of todos:
const todoListState = atomFamily({
key: 'todoListState',
default: (id) => ({
id,
text: '',
isComplete: false,
}),
});
function TodoItem({ id }) {
const [todo, setTodo] = useRecoilState(todoListState(id));
return (
<div>
<input
type="text"
value={todo.text}
onChange={(e) => setTodo({ ...todo, text: e.target.value })}
/>
<input
type="checkbox"
checked={todo.isComplete}
onChange={(e) => setTodo({ ...todo, isComplete: e.target.checked })}
/>
</div>
);
}
This approach allows each todo item to have its own state, while still being part of a larger collection.
One of the things I appreciate about Recoil is how it encourages a modular approach to state management. You can split your state into logical units, making it easier to reason about and maintain as your application grows.
Recoil also provides some powerful debugging tools out of the box. The useRecoilSnapshot hook allows you to take a snapshot of your entire Recoil state at any point in time. This can be incredibly useful for debugging complex state issues or implementing time-travel debugging.
function DebugObserver() {
const snapshot = useRecoilSnapshot();
useEffect(() => {
console.log('The following atoms were modified:');
for (const node of snapshot.getNodes_UNSTABLE({isModified: true})) {
console.log(node.key, snapshot.getLoadable(node));
}
}, [snapshot]);
return null;
}
This DebugObserver component will log any state changes to the console, giving you a clear picture of what’s happening in your application.
One aspect of Recoil that I find particularly exciting is its potential for code splitting and lazy loading. Because Recoil state is defined outside of your React components, it’s easy to split your state definitions into separate files and load them on demand. This can lead to significant performance improvements, especially for larger applications.
Recoil also plays well with TypeScript, which is a big plus in my book. The type inference for atoms and selectors works great out of the box, and you can easily add your own type definitions for more complex state structures.
When it comes to testing, Recoil doesn’t disappoint. The RecoilRoot component makes it easy to set up a test environment, and the various hooks provided by Recoil allow you to easily manipulate and observe state in your tests.
import { RecoilRoot, useRecoilValue } from 'recoil';
import { render, act } from '@testing-library/react';
import { myAtom } from './atoms';
test('myAtom updates correctly', () => {
let result;
function TestComponent() {
result = useRecoilValue(myAtom);
return null;
}
render(
<RecoilRoot>
<TestComponent />
</RecoilRoot>
);
expect(result).toBe(initialValue);
act(() => {
set(myAtom, newValue);
});
expect(result).toBe(newValue);
});
This testing approach feels natural and allows you to focus on testing your state logic without getting bogged down in implementation details.
One thing to keep in mind when using Recoil is that it’s still a relatively young library. While it’s been adopted by many developers and has a growing ecosystem, it doesn’t have the same level of community support as some more established state management solutions. However, I’ve found the documentation to be excellent, and the core concepts are intuitive enough that I was able to get up and running quickly.
As with any state management solution, it’s important to consider whether Recoil is the right fit for your project. For smaller applications, you might find that React’s built-in state management capabilities are sufficient. However, for larger applications or those with complex state requirements, Recoil can be a game-changer.
I’ve been using Recoil in a few projects recently, and I’ve been impressed with how it’s simplified my state management code. The ability to easily share and derive state has made it much easier to reason about my application’s data flow.
One pattern I’ve found particularly useful is combining Recoil with React’s context API. While Recoil is great for managing application state, there are still scenarios where you might want to use context for dependency injection or for state that’s truly local to a particular part of your component tree.
Here’s an example of how you might combine Recoil with context:
import { atom, useRecoilState } from 'recoil';
import { createContext, useContext } from 'react';
const ThemeContext = createContext('light');
const userPreferencesState = atom({
key: 'userPreferencesState',
default: {
fontSize: 16,
language: 'en',
},
});
function App() {
const [theme, setTheme] = useState('light');
const [userPreferences, setUserPreferences] = useRecoilState(userPreferencesState);
return (
<ThemeContext.Provider value={theme}>
<UserPreferences preferences={userPreferences} setPreferences={setUserPreferences} />
<ThemeToggle setTheme={setTheme} />
<Content />
</ThemeContext.Provider>
);
}
function Content() {
const theme = useContext(ThemeContext);
const [userPreferences] = useRecoilState(userPreferencesState);
return (
<div style={{ backgroundColor: theme === 'light' ? 'white' : 'black', fontSize: userPreferences.fontSize }}>
{/* Content goes here */}
</div>
);
}
In this example, we’re using Recoil to manage user preferences that need to be accessible throughout the app, while using context for the theme which is set at the top level and passed down.
As I’ve worked more with Recoil, I’ve also come to appreciate its approach to state persistence. While Recoil doesn’t provide built-in persistence, it’s straightforward to implement using effects:
const localStorageEffect = key => ({setSelf, onSet}) => {
const savedValue = localStorage.getItem(key)
if (savedValue != null) {
setSelf(JSON.parse(savedValue));
}
onSet((newValue, _, isReset) => {
isReset
? localStorage.removeItem(key)
: localStorage.setItem(key, JSON.stringify(newValue));
});
};
const persistedState = atom({
key: 'persistedState',
default: { /* default value */ },
effects: [
localStorageEffect('persist-key'),
]
});
This approach allows you to easily persist specific parts of your state to localStorage (or any other storage mechanism), while keeping the rest of your state ephemeral.
One area where I think Recoil really shines is in handling derived state. The selector API makes it easy to create computed values based on other parts of your state. This can lead to more declarative code and can help prevent bugs caused by state getting out of sync.
For example, let’s say we have a list of todos and we want to display the number of completed todos:
const todoListState = atom({
key: 'todoListState',
default: [],
});
const completedTodosCountState = selector({
key: 'completedTodosCountState',
get: ({get}) => {
const todoList = get(todoListState);
return todoList.filter(todo => todo.isComplete).length;
},
});
function TodoStats() {
const completedTodosCount = useRecoilValue(completedTodosCountState);
return <div>Completed Todos: {completedTodosCount}</div>;
}
This approach ensures that the completed todos count is always in sync with the actual todo list, without requiring any manual updates.
As I’ve worked more with Recoil, I’ve also come to appreciate its flexibility when it comes to state updates. While the basic useRecoilState hook is great for simple updates, Recoil also provides more powerful tools for complex update scenarios.
The useRecoilCallback hook, for example, allows you to perform reads and writes