Skip to content

Instantly share code, notes, and snippets.

@deltaepsilon
Created January 6, 2025 01:15
Show Gist options
  • Save deltaepsilon/eb888e84c7209069faf2d3ceaf6085bb to your computer and use it in GitHub Desktop.
Save deltaepsilon/eb888e84c7209069faf2d3ceaf6085bb to your computer and use it in GitHub Desktop.
A bare-bones data-management hook for React. It's literally just localforage and the Broadcast API.
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[],
defaults = [] as any[],
options: LocalForageOptions = { name: 'arcus' }
) {
const channelId = `localforage-${options.storeName ?? 'arcus'}-${options.name ?? 'default'}`;
const [store, setStore] = useState<LocalForage | null>(null);
const [values, setValues] = useState<T>(defaults as unknown as T);
const [setters, setSetters] = useState<Setter[]>([] as Setter[]);
const sendKeyValueToChannel = useCallback(
([key, value]: [string, any]) => {
const channel = getChannel(channelId, 'send');
channel.postMessage({ key, value });
},
[channelId]
);
const refresh = useCallback(async () => {
if (store) {
const values = await Promise.all(
keys.map((key, index) =>
store.getItem(key).then((value) => {
if (value === null) {
value = defaults[index];
}
return value;
})
)
);
const setters = keys.map((key, index) => async (value: any | Function) => {
const defaultValue = defaults[index];
if (typeof value === 'function') {
value = value((await store.getItem(key)) || defaultValue);
} else {
value = value === undefined ? defaultValue : value;
}
await store.setItem(key, value);
sendKeyValueToChannel([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);
sendKeyValueToChannel([key, value]);
return result;
}
},
[sendKeyValueToChannel, 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