|
| 1 | +import { |
| 2 | + DependencyList, |
| 3 | + EffectCallback, |
| 4 | + useRef, |
| 5 | + useEffect, |
| 6 | + useDebugValue, |
| 7 | +} from 'react' |
| 8 | +import useWillUnmount from './useWillUnmount' |
| 9 | +import useMounted from './useMounted' |
| 10 | + |
| 11 | +export type EffectHook = (effect: EffectCallback, deps?: DependencyList) => void |
| 12 | + |
| 13 | +export type IsEqual<TDeps extends DependencyList> = ( |
| 14 | + nextDeps: TDeps, |
| 15 | + prevDeps: TDeps, |
| 16 | +) => boolean |
| 17 | + |
| 18 | +export type CustomEffectOptions<TDeps extends DependencyList> = { |
| 19 | + isEqual: IsEqual<TDeps> |
| 20 | + effectHook?: EffectHook |
| 21 | +} |
| 22 | + |
| 23 | +type CleanUp = { |
| 24 | + (): void |
| 25 | + cleanup?: ReturnType<EffectCallback> |
| 26 | +} |
| 27 | + |
| 28 | +/** |
| 29 | + * a useEffect() hook with customized depedency comparision |
| 30 | + * |
| 31 | + * @param effect The effect callback |
| 32 | + * @param dependencies A list of dependencies |
| 33 | + * @param isEqual A function comparing the next and previous dependencyLists |
| 34 | + */ |
| 35 | +function useCustomEffect<TDeps extends DependencyList = DependencyList>( |
| 36 | + effect: EffectCallback, |
| 37 | + dependencies: TDeps, |
| 38 | + isEqual: IsEqual<TDeps>, |
| 39 | +): void |
| 40 | +/** |
| 41 | + * a useEffect() hook with customized depedency comparision |
| 42 | + * |
| 43 | + * @param effect The effect callback |
| 44 | + * @param dependencies A list of dependencies |
| 45 | + * @param options |
| 46 | + * @param options.isEqual A function comparing the next and previous dependencyLists |
| 47 | + * @param options.effectHook the underlying effect hook used, defaults to useEffect |
| 48 | + */ |
| 49 | +function useCustomEffect<TDeps extends DependencyList = DependencyList>( |
| 50 | + effect: EffectCallback, |
| 51 | + dependencies: TDeps, |
| 52 | + options: CustomEffectOptions<TDeps>, |
| 53 | +): void |
| 54 | +function useCustomEffect<TDeps extends DependencyList = DependencyList>( |
| 55 | + effect: EffectCallback, |
| 56 | + dependencies: TDeps, |
| 57 | + isEqualOrOptions: IsEqual<TDeps> | CustomEffectOptions<TDeps>, |
| 58 | +) { |
| 59 | + const isMounted = useMounted() |
| 60 | + const { isEqual, effectHook = useEffect } = |
| 61 | + typeof isEqualOrOptions === 'function' |
| 62 | + ? { isEqual: isEqualOrOptions } |
| 63 | + : isEqualOrOptions |
| 64 | + |
| 65 | + const dependenciesRef = useRef<TDeps>() |
| 66 | + dependenciesRef.current = dependencies |
| 67 | + |
| 68 | + const cleanupRef = useRef<CleanUp | null>(null) |
| 69 | + |
| 70 | + effectHook(() => { |
| 71 | + // If the ref the is `null` it's either the first effect or the last effect |
| 72 | + // ran and was cleared, meaning _this_ update should run, b/c the equality |
| 73 | + // check failed on in the cleanup of the last effect. |
| 74 | + if (cleanupRef.current === null) { |
| 75 | + const cleanup = effect() |
| 76 | + |
| 77 | + cleanupRef.current = () => { |
| 78 | + if (isMounted() && isEqual(dependenciesRef.current!, dependencies)) { |
| 79 | + return |
| 80 | + } |
| 81 | + |
| 82 | + cleanupRef.current = null |
| 83 | + if (cleanup) cleanup() |
| 84 | + } |
| 85 | + } |
| 86 | + |
| 87 | + return cleanupRef.current |
| 88 | + }) |
| 89 | + |
| 90 | + useDebugValue(effect) |
| 91 | +} |
| 92 | + |
| 93 | +export default useCustomEffect |
0 commit comments