Skip to content

Instantly share code, notes, and snippets.

@deltaepsilon
Created October 23, 2024 14:00
Show Gist options
  • Save deltaepsilon/501646efb78dc3f8014ce9fb48a9ee00 to your computer and use it in GitHub Desktop.
Save deltaepsilon/501646efb78dc3f8014ce9fb48a9ee00 to your computer and use it in GitHub Desktop.
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