2020-09-18 21:53:13 +08:00

39 lines
1.4 KiB
TypeScript

/* eslint-disable no-redeclare */
import { useRecoilState, RecoilState } from 'recoil'
import produce, { Draft } from 'immer'
import { useMemo } from 'react'
export function useRecoilObjectWithImmer<T> (value: RecoilState<T>) {
const [copy, rawSet] = useRecoilState(value)
const set = useMemo(() => {
function set<K extends keyof Draft<T>> (key: K, value: Draft<T>[K]): void
function set (data: Partial<T>): void
function set (f: (draft: Draft<T>) => void | T): void
function set<K extends keyof Draft<T>> (data: any, value?: Draft<T>[K]): void {
if (typeof data === 'string') {
rawSet(pre => produce(pre, (draft: Draft<T>) => {
const key = data as K
const v = value
draft[key] = v!
}))
} else if (typeof data === 'function') {
const fn = data as (draft: Draft<T>) => void | T
rawSet(pre => produce(pre, fn) as T)
} else if (typeof data === 'object') {
rawSet(pre => produce(pre, (draft: Draft<T>) => {
const obj = data as Draft<T>
for (const key of Object.keys(obj)) {
const k = key as keyof Draft<T>
draft[k] = obj[k]
}
}))
}
}
return set
}, [rawSet])
return [copy, set] as [T, typeof set]
}