Skip to content

Instantly share code, notes, and snippets.

@acorn1010
Last active December 27, 2024 01:59
Show Gist options
  • Save acorn1010/9f4621d3dfc33052ffd84f6c2a06d4d6 to your computer and use it in GitHub Desktop.
Save acorn1010/9f4621d3dfc33052ffd84f6c2a06d4d6 to your computer and use it in GitHub Desktop.
Easier Zustand store
import {SetStateAction, useCallback} from 'react';
import {create} from "zustand";
export type EqualityFn<T> = (left: T | null | undefined, right: T | null | undefined) => boolean;
export type StoreType<State> = {
use<K extends keyof State>(
key: K,
defaultValue?: State[K],
equalityFn?: EqualityFn<State[K]>,
): [State[K], (value: SetStateAction<State[K]>) => void];
useAll: () => State;
delete: <K extends keyof State>(key: K) => void;
get: <K extends keyof State>(key: K) => State[K];
getAll: () => State;
has: <K extends keyof State>(key: K) => boolean;
setAll: (state: State) => void;
update: (state: Partial<State>) => void;
set: <K extends keyof State>(key: K, value: SetStateAction<State[K]>) => void;
reset: () => void;
};
// eslint-disable-next-line @typescript-eslint/ban-types
const isFunction = (fn: unknown): fn is Function => (typeof fn === 'function');
/**
* Create a global state
*
* It returns a set of functions
* - `use`: Works like React.useState. "Registers" the component as a listener on that key
* - `get`: retrieves a key without a re-render
* - `set`: sets a key. Causes re-renders on any listeners
* - `getAll`: retrieves the entire state (all keys) as an object without a re-render
* - `reset`: resets the state back to its initial value
*
* @example
* import { createStore } from 'create-store';
*
* const store = createStore({ count: 0 });
*
* const Component = () => {
* const [count, setCount] = store.use('count');
* ...
* };
*/
export const createGlobalStore = <State extends object>(initialState: State) => {
// NOTE: Not using structuredClone because browser support only goes about 2 years back.
const store = create<State>(() => deepClone(initialState));
const setter = <T extends keyof State>(key: T, value: SetStateAction<State[T]>) => {
if (isFunction(value)) {
store.setState(prevValue => ({[key]: value(prevValue[key])} as unknown as Partial<State>));
} else {
store.setState({[key]: value} as unknown as Partial<State>);
}
};
return {
/** Works like React.useState. "Registers" the component as a listener on that key. */
use<K extends keyof State>(
key: K,
defaultValue?: State[K],
equalityFn?: EqualityFn<State[K]>): [State[K], (value: SetStateAction<State[K]>) => void] {
// If state isn't defined for a given defaultValue, set it.
if (defaultValue !== undefined && !(key in store.getState())) {
setter(key, defaultValue);
}
const result = store((state) => state[key], equalityFn || Object.is);
const keySetter = useCallback(
(value: SetStateAction<State[K]>) => setter(key, value),
[key],
);
return [result! as State[K], keySetter];
},
/** Listens on the entire state, causing a re-render when anything in the state changes. */
useAll: () => store(state => state),
/** Deletes a `key` from state, causing a re-render for anything listening. */
delete<K extends keyof State>(key: K) {
store.setState(prevState => {
const rest = {...prevState};
delete rest[key];
return rest as Partial<State>;
}, true);
},
/** Retrieves the current `key` value. Does _not_ listen on state changes (meaning no re-renders). */
get<K extends keyof State>(key: K) {
return store.getState()[key];
},
/** Retrieves the entire state. Does _not_ listen on state changes (meaning no re-renders). */
getAll: () => store.getState(),
/** Returns `true` if `key` is in the state. */
has<K extends keyof State>(key: K) {
return key in store.getState();
},
/** Sets a `key`, triggering a re-render for all listeners. */
set: setter,
/** Sets the entire state, removing any keys that aren't present in `state`. */
setAll: (state: State) => store.setState(state, true),
/** Updates the keys in `state`, leaving any keys / values not in `state` unchanged. */
update: (state: Partial<State>) => store.setState(state, false),
/** Resets the entire state back to its initial state when the store was created. */
reset: () => store.setState(deepClone(initialState), true),
};
};
/**
* Returns a wrapped `store` that can't be modified. Useful when you want to
* control who is able to write to a store.
*/
export function createReadonlyStore<T extends ReturnType<typeof createGlobalStore>>(
store: T) {
type State = ReturnType<T['getAll']>;
return {
get: store.get,
getAll: store.getAll,
use: <K extends keyof State>(key: K, equalityFn?: EqualityFn<State[K]>) =>
(store.use as any)(key, undefined, equalityFn)[0] as State[K] | undefined | null,
useAll: store.useAll,
};
}
/**
* Deeply copies objects. Borrowed from just-clone, but with some nicer types.
* See: https://github.com/angus-c/just/blob/master/packages/collection-clone/index.cjs
*/
function deepClone<T>(obj: T): T {
let result = obj;
const type = {}.toString.call(obj).slice(8, -1);
if (type === 'Set') {
return new Set([...obj as Set<any>].map(value => deepClone(value))) as any;
}
if (type === 'Map') {
return new Map([...obj as Set<any>].map(kv => [deepClone(kv[0]), deepClone(kv[1])])) as any;
}
if (type === 'Date') {
return new Date((obj as Date).getTime()) as any;
}
if (type === 'RegExp') {
return RegExp((obj as RegExp).source as string, getRegExpFlags(obj as RegExp)) as any;
}
if (type === 'Array' || type === 'Object') {
result = Array.isArray(obj) ? [] : {} as any;
for (const key in obj) {
// include prototype properties
result[key] = deepClone(obj[key]);
}
}
// primitives and non-supported objects (e.g. functions) land here
return result;
}
function getRegExpFlags(regExp: RegExp): string {
if ((typeof regExp.source as any).flags === 'string') {
return (regExp.source as any).flags;
}
const flags = [];
regExp.global && flags.push('g');
regExp.ignoreCase && flags.push('i');
regExp.multiline && flags.push('m');
regExp.sticky && flags.push('y');
regExp.unicode && flags.push('u');
return flags.join('');
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment