UNPKG

@carbon/react

Version:

React components for the Carbon Design System

70 lines (61 loc) 1.93 kB
/** * Copyright IBM Corp. 2016, 2023 * * This source code is licensed under the Apache-2.0 license found in the * LICENSE file in the root directory of this source tree. */ 'use strict'; var React = require('react'); var usePrefix = require('./usePrefix.js'); var useIsomorphicEffect = require('./useIsomorphicEffect.js'); const usePresence = (ref, isOpen) => { const prefix = usePrefix.usePrefix(); const [exitState, setExitState] = React.useState(isOpen ? 'idle' : 'finished'); const isExiting = exitState === 'active'; // element is exiting if (!isOpen && exitState === 'idle') { setExitState('active'); } // element exit was interrupted if (isOpen && exitState !== 'idle') { setExitState('idle'); } const handleAnimationEnd = React.useCallback(() => { setExitState('finished'); }, []); useIsomorphicEffect.default(() => { if (!ref.current || !isExiting) return; // resolve for JSDOM if (!('getAnimations' in ref.current)) { handleAnimationEnd(); return; } // cover all animations that start with the presence prefix const animations = ref.current.getAnimations({ subtree: true }).filter(animation => animation instanceof CSSAnimation && animation.animationName.startsWith(`${prefix}--presence`)); if (!animations.length) { handleAnimationEnd(); return; } let cancelled = false; Promise.all(animations.map(animation => animation.finished)).finally(() => { if (cancelled) return; handleAnimationEnd(); }); return () => { cancelled = true; }; }, [ref, isExiting, prefix, handleAnimationEnd]); return { /** * Indicates whether the ref object is supposed to be mounted */ isPresent: isOpen || exitState !== 'finished', /** * Indicates whether the ref object is currently exiting */ isExiting }; }; exports.usePresence = usePresence;