Project Name | Stars | Downloads | Repos Using This | Packages Using This | Most Recent Commit | Total Releases | Latest Release | Open Issues | License | Language |
---|---|---|---|---|---|---|---|---|---|---|
Mobx | 26,495 | 9,267 | 3,766 | 8 days ago | 236 | September 07, 2022 | 42 | mit | TypeScript | |
Simple, scalable state management. | ||||||||||
Jotai | 13,580 | 123 | 2 days ago | 121 | September 19, 2022 | 7 | mit | TypeScript | ||
👻 Primitive and flexible state management for React | ||||||||||
Boardgame.io | 9,568 | 32 | 18 | 17 days ago | 200 | May 07, 2022 | 96 | mit | TypeScript | |
State Management and Multiplayer Networking for Turn-Based Games | ||||||||||
Awesome React Hooks | 8,307 | 10 months ago | 8 | cc0-1.0 | ||||||
Awesome React Hooks | ||||||||||
Rematch | 8,175 | 210 | 109 | 9 months ago | 76 | November 09, 2021 | 19 | mit | TypeScript | |
The Redux Framework | ||||||||||
React Final Form | 7,257 | 500 | 384 | 17 days ago | 76 | April 01, 2022 | 390 | mit | JavaScript | |
🏁 High performance subscription-based form state management for React | ||||||||||
Mobx State Tree | 6,651 | 479 | 259 | 3 months ago | 107 | August 09, 2022 | 201 | mit | TypeScript | |
Full-featured reactive state management without the boilerplate | ||||||||||
Easy Peasy | 4,969 | 37 | 50 | 5 days ago | 254 | September 17, 2022 | 18 | mit | JavaScript | |
Vegetarian friendly state for React | ||||||||||
Router | 4,792 | 3 | 5 | a day ago | 127 | January 31, 2022 | 43 | mit | TypeScript | |
🤖 Type-safe router w/ built-in caching & URL state management for JS/TS, React, Preact, Solid, Vue, Svelte and Angular | ||||||||||
Effector | 4,213 | 26 | 109 | a day ago | 242 | April 15, 2022 | 140 | mit | TypeScript | |
Business logic with ease ☄️ |
visit jotai.org or npm i jotai
Jotai scales from a simple useState replacement to an enterprise TypeScript application.
An atom represents a piece of state. All you need is to specify an initial value, which can be primitive values like strings and numbers, objects, and arrays. You can create as many primitive atoms as you want.
import { atom } from 'jotai'
const countAtom = atom(0)
const countryAtom = atom('Japan')
const citiesAtom = atom(['Tokyo', 'Kyoto', 'Osaka'])
const mangaAtom = atom({ 'Dragon Ball': 1984, 'One Piece': 1997, Naruto: 1999 })
It can be used like React.useState
:
import { useAtom } from 'jotai'
function Counter() {
const [count, setCount] = useAtom(countAtom)
return (
<h1>
{count}
<button onClick={() => setCount((c) => c + 1)}>one up</button>
...
A new read-only atom can be created from existing atoms by passing a read
function as the first argument. get
allows you to fetch the contextual value
of any atom.
const doubledCountAtom = atom((get) => get(countAtom) * 2)
function DoubleCounter() {
const [doubledCount] = useAtom(doubledCountAtom)
return <h2>{doubledCount}</h2>
}
You can combine multiple atoms to create a derived atom.
const count1 = atom(1)
const count2 = atom(2)
const count3 = atom(3)
const sum = atom((get) => get(count1) + get(count2) + get(count3))
Or if you like fp patterns ...
const atoms = [count1, count2, count3, ...otherAtoms]
const sum = atom((get) => atoms.map(get).reduce((acc, count) => acc + count))
You can make the read function an async function too.
const urlAtom = atom('https://json.host.com')
const fetchUrlAtom = atom(async (get) => {
const response = await fetch(get(urlAtom))
return await response.json()
})
function Status() {
// Re-renders the component after urlAtom is changed and the async function above concludes
const [json] = useAtom(fetchUrlAtom)
...
Specify a write function at the second argument. get
will return the current
value of an atom. set
will update the value of an atom.
const decrementCountAtom = atom(
(get) => get(countAtom),
(get, set, _arg) => set(countAtom, get(countAtom) - 1)
)
function Counter() {
const [count, decrement] = useAtom(decrementCountAtom)
return (
<h1>
{count}
<button onClick={decrement}>Decrease</button>
...
Just do not define a read function.
const multiplyCountAtom = atom(null, (get, set, by) =>
set(countAtom, get(countAtom) * by)
)
function Controls() {
const [, multiply] = useAtom(multiplyCountAtom)
return <button onClick={() => multiply(3)}>triple</button>
}
Just make the write function an async function and call set
when you're ready.
const fetchCountAtom = atom(
(get) => get(countAtom),
async (_get, set, url) => {
const response = await fetch(url)
set(countAtom, (await response.json()).count)
}
)
function Controls() {
const [count, compute] = useAtom(fetchCountAtom)
return (
<button onClick={() => compute('http://count.host.com')}>compute</button>
...