feat: add useAnimatedMount
This commit is contained in:
parent
d10fec90ad
commit
3ccfa3b519
@ -1,3 +1,4 @@
|
|||||||
export * from "./vcn";
|
export * from "./vcn";
|
||||||
export * from "./Slot";
|
export * from "./Slot";
|
||||||
export * from "./useDocument";
|
export * from "./useDocument";
|
||||||
|
export * from "./useAnimatedMount";
|
||||||
|
84
packages/react/lib/useAnimatedMount.ts
Normal file
84
packages/react/lib/useAnimatedMount.ts
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
import { type MutableRefObject, useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
function getCalculatedTransitionDuration(
|
||||||
|
ref: MutableRefObject<HTMLElement>,
|
||||||
|
): number {
|
||||||
|
let transitionDuration: {
|
||||||
|
value: number;
|
||||||
|
unit: string;
|
||||||
|
} | null;
|
||||||
|
if (ref.current.computedStyleMap !== undefined) {
|
||||||
|
transitionDuration = ref.current
|
||||||
|
.computedStyleMap()
|
||||||
|
.get("transition-duration") as { value: number; unit: string };
|
||||||
|
} else {
|
||||||
|
const style = /(\d+(\.\d+)?)(.+)/.exec(
|
||||||
|
window.getComputedStyle(ref.current).transitionDuration,
|
||||||
|
);
|
||||||
|
if (!style) return 0;
|
||||||
|
transitionDuration = {
|
||||||
|
value: Number.parseFloat(style[1] ?? "0"),
|
||||||
|
unit: style[3] ?? style[2] ?? "s",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
transitionDuration.value *
|
||||||
|
({
|
||||||
|
s: 1000,
|
||||||
|
ms: 1,
|
||||||
|
}[transitionDuration.unit] ?? 1)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Component Mount Component Appear Component Disappear Component Unmount
|
||||||
|
* v v v v
|
||||||
|
* |-|=================|------------------------|======================|-|
|
||||||
|
*/
|
||||||
|
|
||||||
|
function useAnimatedMount(
|
||||||
|
visible: boolean,
|
||||||
|
ref: MutableRefObject<HTMLElement | null>,
|
||||||
|
callbacks?: { onMount: () => void; onUnmount: () => void },
|
||||||
|
) {
|
||||||
|
const [state, setState] = useState<{
|
||||||
|
isMounted: boolean;
|
||||||
|
isRendered: boolean;
|
||||||
|
}>({ isMounted: visible, isRendered: visible });
|
||||||
|
|
||||||
|
const umountCallback = useCallback(() => {
|
||||||
|
setState((p) => ({ ...p, isRendered: false }));
|
||||||
|
|
||||||
|
const calculatedTransitionDuration = ref.current
|
||||||
|
? getCalculatedTransitionDuration(ref as MutableRefObject<HTMLElement>)
|
||||||
|
: 0;
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
setState((p) => ({ ...p, isMounted: false }));
|
||||||
|
callbacks?.onUnmount?.();
|
||||||
|
}, calculatedTransitionDuration);
|
||||||
|
}, [ref, callbacks]);
|
||||||
|
|
||||||
|
const mountCallback = useCallback(() => {
|
||||||
|
setState((p) => ({ ...p, isMounted: true }));
|
||||||
|
callbacks?.onMount?.();
|
||||||
|
requestAnimationFrame(function onMount() {
|
||||||
|
if (!ref.current) return requestAnimationFrame(onMount);
|
||||||
|
setState((p) => ({ ...p, isRendered: true }));
|
||||||
|
});
|
||||||
|
}, [ref.current, callbacks]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
console.log(state);
|
||||||
|
if (!visible && state.isRendered) {
|
||||||
|
umountCallback();
|
||||||
|
} else if (visible && !state.isMounted) {
|
||||||
|
mountCallback();
|
||||||
|
}
|
||||||
|
}, [state, visible, mountCallback, umountCallback]);
|
||||||
|
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { useAnimatedMount };
|
Loading…
x
Reference in New Issue
Block a user