UNPKG

react-aria

Version:
187 lines (172 loc) 8.92 kB
import {focusWithoutScrolling as $1969ac565cfec8d0$export$de79e2c695e052f3} from "../utils/focusWithoutScrolling.mjs"; import {getEventTarget as $23f2114a1b82827e$export$e58f029f0fbfdb29} from "../utils/shadowdom/DOMFunctions.mjs"; import {getInteractionModality as $8f5a2122b0992be3$export$630ff653c5ada6a9} from "../interactions/useFocusVisible.mjs"; import $7pO52$intlStringsmjs from "./intlStrings.mjs"; import {mergeProps as $bbaa08b3cd72f041$export$9d1611c77c2fe928} from "../utils/mergeProps.mjs"; import {useFocusWithin as $2c9edc598a03d523$export$420e68273165f4ec} from "../interactions/useFocusWithin.mjs"; import {useHover as $e969f22b6713ca4a$export$ae780daf29e6d456} from "../interactions/useHover.mjs"; import {useLandmark as $6b88d4883d4c7fe3$export$4cc632584fd87fae} from "../landmark/useLandmark.mjs"; import {useLayoutEffect as $c4867b2f328c2698$export$e5c5a5f917a5871c} from "../utils/useLayoutEffect.mjs"; import {useLocalizedStringFormatter as $cf2482eff2eeeec2$export$f12b703ca79dfbb1} from "../i18n/useLocalizedStringFormatter.mjs"; import {useRef as $7pO52$useRef, useCallback as $7pO52$useCallback, useEffect as $7pO52$useEffect} from "react"; function $parcel$interopDefault(a) { return a && a.__esModule ? a.default : a; } /* * Copyright 2025 Adobe. All rights reserved. * This file is licensed to you under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. You may obtain a copy * of the License at http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed under * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ function $17a75c27599f3197$export$b8cbbb20a51697de(props, state, ref) { let stringFormatter = (0, $cf2482eff2eeeec2$export$f12b703ca79dfbb1)((0, ($parcel$interopDefault($7pO52$intlStringsmjs))), '@react-aria/toast'); let { landmarkProps: landmarkProps } = (0, $6b88d4883d4c7fe3$export$4cc632584fd87fae)({ role: 'region', 'aria-label': props['aria-label'] || stringFormatter.format('notifications', { count: state.visibleToasts.length }) }, ref); let isHovered = (0, $7pO52$useRef)(false); let isFocused = (0, $7pO52$useRef)(false); let updateTimers = (0, $7pO52$useCallback)(()=>{ if (isHovered.current || isFocused.current) state.pauseAll(); else state.resumeAll(); }, [ state ]); let { hoverProps: hoverProps } = (0, $e969f22b6713ca4a$export$ae780daf29e6d456)({ onHoverStart: ()=>{ isHovered.current = true; updateTimers(); }, onHoverEnd: ()=>{ isHovered.current = false; updateTimers(); } }); // Manage focus within the toast region. // If a focused containing toast is removed, move focus to the next toast, or the previous toast if there is no next toast. let toasts = (0, $7pO52$useRef)([]); let prevVisibleToasts = (0, $7pO52$useRef)(state.visibleToasts); let focusedToast = (0, $7pO52$useRef)(null); (0, $c4867b2f328c2698$export$e5c5a5f917a5871c)(()=>{ // If no toast has focus, then don't do anything. if (focusedToast.current === -1 || state.visibleToasts.length === 0 || !ref.current) { toasts.current = []; prevVisibleToasts.current = state.visibleToasts; return; } toasts.current = [ ...ref.current.querySelectorAll('[role="alertdialog"]') ]; // If the visible toasts haven't changed, we don't need to do anything. if (prevVisibleToasts.current.length === state.visibleToasts.length && state.visibleToasts.every((t, i)=>t.key === prevVisibleToasts.current[i].key)) { prevVisibleToasts.current = state.visibleToasts; return; } // Get a list of all toasts by index and add info if they are removed. let allToasts = prevVisibleToasts.current.map((t, i)=>({ ...t, i: i, isRemoved: !state.visibleToasts.some((t2)=>t.key === t2.key) })); let removedFocusedToastIndex = allToasts.findIndex((t)=>t.i === focusedToast.current && t.isRemoved); // If the focused toast was removed, focus the next or previous toast. if (removedFocusedToastIndex > -1) { // In pointer modality, move focus out of the toast region. // Otherwise auto-dismiss timers will appear "stuck". if ((0, $8f5a2122b0992be3$export$630ff653c5ada6a9)() === 'pointer' && lastFocused.current?.isConnected) (0, $1969ac565cfec8d0$export$de79e2c695e052f3)(lastFocused.current); else { let i = 0; let nextToast; let prevToast; while(i <= removedFocusedToastIndex){ if (!allToasts[i].isRemoved) prevToast = Math.max(0, i - 1); i++; } while(i < allToasts.length){ if (!allToasts[i].isRemoved) { nextToast = i - 1; break; } i++; } // in the case where it's one toast at a time, both will be undefined, but we know the index must be 0 if (prevToast === undefined && nextToast === undefined) prevToast = 0; // prioritize going to newer toasts if (prevToast >= 0 && prevToast < toasts.current.length) (0, $1969ac565cfec8d0$export$de79e2c695e052f3)(toasts.current[prevToast]); else if (nextToast >= 0 && nextToast < toasts.current.length) (0, $1969ac565cfec8d0$export$de79e2c695e052f3)(toasts.current[nextToast]); } } prevVisibleToasts.current = state.visibleToasts; }, [ state.visibleToasts, ref ]); let lastFocused = (0, $7pO52$useRef)(null); let { focusWithinProps: focusWithinProps } = (0, $2c9edc598a03d523$export$420e68273165f4ec)({ onFocusWithin: (e)=>{ isFocused.current = true; lastFocused.current = e.relatedTarget; updateTimers(); }, onBlurWithin: ()=>{ isFocused.current = false; lastFocused.current = null; updateTimers(); } }); // When the number of visible toasts becomes 0 or the region unmounts, // restore focus to the last element that had focus before the user moved focus // into the region. FocusScope restore focus doesn't update whenever the focus // moves in, it only happens once, so we correct it. // Because we're in a hook, we can't control if the user unmounts or not. (0, $7pO52$useEffect)(()=>{ if (state.visibleToasts.length === 0 && lastFocused.current?.isConnected) { if ((0, $8f5a2122b0992be3$export$630ff653c5ada6a9)() === 'pointer') (0, $1969ac565cfec8d0$export$de79e2c695e052f3)(lastFocused.current); else lastFocused.current.focus(); lastFocused.current = null; } }, [ ref, state.visibleToasts.length ]); (0, $7pO52$useEffect)(()=>{ return ()=>{ if (lastFocused.current?.isConnected) { if ((0, $8f5a2122b0992be3$export$630ff653c5ada6a9)() === 'pointer') (0, $1969ac565cfec8d0$export$de79e2c695e052f3)(lastFocused.current); else lastFocused.current.focus(); lastFocused.current = null; } }; }, [ ref ]); return { regionProps: (0, $bbaa08b3cd72f041$export$9d1611c77c2fe928)(landmarkProps, hoverProps, focusWithinProps, { tabIndex: -1, // Mark the toast region as a "top layer", so that it: // - is not aria-hidden when opening an overlay // - allows focus even outside a containing focus scope // - doesn’t dismiss overlays when clicking on it, even though it is outside // @ts-ignore 'data-react-aria-top-layer': true, // listen to focus events separate from focuswithin because that will only fire once // and we need to follow all focus changes onFocus: (e)=>{ let target = (0, $23f2114a1b82827e$export$e58f029f0fbfdb29)(e).closest('[role="alertdialog"]'); focusedToast.current = toasts.current.findIndex((t)=>t === target); }, onBlur: ()=>{ focusedToast.current = -1; } }) }; } export {$17a75c27599f3197$export$b8cbbb20a51697de as useToastRegion}; //# sourceMappingURL=useToastRegion.mjs.map