Created
October 23, 2024 14:00
-
-
Save deltaepsilon/501646efb78dc3f8014ce9fb48a9ee00 to your computer and use it in GitHub Desktop.
This file contains 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 localforage from 'localforage'; | |
import { useCallback, useEffect, useState } from 'react'; | |
type Setter = (value: any) => Promise<any>; | |
/** | |
* Usage example: | |
* | |
* const IS_DEV_KEY = 'is-dev'; | |
* | |
* const { values: [isDev], setItem } = useLocalforage<[boolean]>([IS_DEV_KEY]); | |
*/ | |
export function useLocalforage<T extends Array<any>>(keys: string[], options: LocalForageOptions = {}) { | |
const channelId = `localforage-${options.storeName ?? 'tah'}-${options.name ?? 'default'}`; | |
const [store, setStore] = useState<LocalForage | null>(null); | |
const [values, setValues] = useState<T>([] as unknown as T); | |
const [setters, setSetters] = useState<Setter[]>([] as Setter[]); | |
const refresh = useCallback(async () => { | |
if (store) { | |
const values = await Promise.all(keys.map((key) => store.getItem(key))); | |
const setters = keys.map((key) => (value: any) => store.setItem(key, value)); | |
setValues(values as T); | |
setSetters(setters); | |
return values; | |
} | |
// eslint-disable-next-line react-hooks/exhaustive-deps | |
}, [keys.join('/'), !!store]); | |
const setItem = useCallback( | |
async (key: string, value: any) => { | |
if (store) { | |
const result = await store.setItem(key, value); | |
const channel = getChannel(channelId, 'send'); | |
channel.postMessage({ key, value }); | |
return result; | |
} | |
}, | |
[channelId, store] | |
); | |
useEffect(() => (refresh(), undefined), [refresh]); | |
// eslint-disable-next-line react-hooks/exhaustive-deps | |
useEffect( | |
() => (!store && setStore(localforage.createInstance(options)), undefined), | |
[options] // eslint-disable-line react-hooks/exhaustive-deps | |
); | |
useEffect(() => { | |
const channel = getChannel(channelId, 'receive'); | |
function listener({ data }: any) { | |
if (keys.includes(data.key)) { | |
const { key, value } = data; | |
setValues((values) => { | |
const index = keys.indexOf(key); | |
if (index > -1) { | |
values[index] = value; | |
} | |
return structuredClone(values) as T; | |
}); | |
} | |
} | |
channel.addEventListener('message', listener); | |
return () => { | |
channel.removeEventListener('message', listener); | |
/** | |
* Leave channel open, or figure out a way to handle it as a singleton. | |
*/ | |
// closeChannel(channelId, 'send'); | |
// closeChannel(channelId, 'receive'); | |
}; | |
}, [setValues, keys, channelId]); | |
return { values, refresh, setters, setItem, store }; | |
} | |
const CHANNELS = new Map<string, BroadcastChannel>(); | |
function getChannel(channelId: string, sendOrReceive: 'send' | 'receive') { | |
const id = `${channelId}-${sendOrReceive}`; | |
if (!CHANNELS.has(id)) { | |
CHANNELS.set(id, new BroadcastChannel(channelId)); | |
} | |
return CHANNELS.get(id) as BroadcastChannel; | |
} | |
function closeChannel(channelId: string, sendOrReceive: 'send' | 'receive') { | |
const id = `${channelId}-${sendOrReceive}`; | |
const channel = CHANNELS.get(id); | |
if (channel) { | |
CHANNELS.delete(id); | |
channel.close(); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment