Created
December 26, 2022 22:24
-
-
Save NoTimeForHero/d5aa6c7ddcc6115af432bc59546e6f44 to your computer and use it in GitHub Desktop.
React Simple Atom Storage
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// import { useEffect, useState } from 'react'; | |
import { useEffect, useState } from 'preact/compat'; | |
interface Atom<T> { | |
key: Symbol | |
} | |
interface StoreValue { | |
value: any, | |
listeners: Set<() => void> | |
} | |
const store = new Map<Symbol,StoreValue>(); | |
export const createAtom = <T>(initialValue: T) : Atom<T> => { | |
const key = Symbol(); | |
store.set(key, { | |
value: initialValue, | |
listeners: new Set() | |
}); | |
return { key }; | |
} | |
export const useAtom = <T>(atom: Atom<T>) => { | |
const entry = store.get(atom.key); | |
if (!entry) throw new Error('Atom not found!'); | |
const [_, rerender] = useState<unknown>(); | |
const value = entry?.value; | |
const setValue = (value: T) => { | |
entry.value = value; | |
entry.listeners.forEach(x => x()); | |
} | |
useEffect(() => { | |
const listener = () => rerender({}); | |
entry.listeners.add(listener); | |
return () => { | |
entry.listeners.delete(listener); | |
} | |
}) | |
return [value, setValue] as const; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment