"use client"; import { Presence, Primitive, composeEventHandlers, createContextScope, dispatchDiscreteCustomEvent, useCallbackRef, useLayoutEffect2 } from "./chunk-QFTOZ7SM.js"; import { require_react_dom } from "./chunk-WERSD76P.js"; import { Slot, useComposedRefs } from "./chunk-TQG5UYZM.js"; import { require_jsx_runtime } from "./chunk-S77I6LSE.js"; import { require_react } from "./chunk-3TFVT2CW.js"; import { __toESM } from "./chunk-4MBMRILA.js"; // node_modules/@radix-ui/react-toast/dist/index.mjs var React8 = __toESM(require_react(), 1); var ReactDOM2 = __toESM(require_react_dom(), 1); // node_modules/@radix-ui/react-collection/dist/index.mjs var import_react = __toESM(require_react(), 1); // node_modules/@radix-ui/react-collection/node_modules/@radix-ui/react-context/dist/index.mjs var React = __toESM(require_react(), 1); var import_jsx_runtime = __toESM(require_jsx_runtime(), 1); function createContextScope2(scopeName, createContextScopeDeps = []) { let defaultContexts = []; function createContext3(rootComponentName, defaultContext) { const BaseContext = React.createContext(defaultContext); const index = defaultContexts.length; defaultContexts = [...defaultContexts, defaultContext]; function Provider2(props) { const { scope, children, ...context } = props; const Context = (scope == null ? void 0 : scope[scopeName][index]) || BaseContext; const value = React.useMemo(() => context, Object.values(context)); return (0, import_jsx_runtime.jsx)(Context.Provider, { value, children }); } function useContext22(consumerName, scope) { const Context = (scope == null ? void 0 : scope[scopeName][index]) || BaseContext; const context = React.useContext(Context); if (context) return context; if (defaultContext !== void 0) return defaultContext; throw new Error(`\`${consumerName}\` must be used within \`${rootComponentName}\``); } Provider2.displayName = rootComponentName + "Provider"; return [Provider2, useContext22]; } const createScope = () => { const scopeContexts = defaultContexts.map((defaultContext) => { return React.createContext(defaultContext); }); return function useScope(scope) { const contexts = (scope == null ? void 0 : scope[scopeName]) || scopeContexts; return React.useMemo( () => ({ [`__scope${scopeName}`]: { ...scope, [scopeName]: contexts } }), [scope, contexts] ); }; }; createScope.scopeName = scopeName; return [createContext3, composeContextScopes(createScope, ...createContextScopeDeps)]; } function composeContextScopes(...scopes) { const baseScope = scopes[0]; if (scopes.length === 1) return baseScope; const createScope = () => { const scopeHooks = scopes.map((createScope2) => ({ useScope: createScope2(), scopeName: createScope2.scopeName })); return function useComposedScopes(overrideScopes) { const nextScopes = scopeHooks.reduce((nextScopes2, { useScope, scopeName }) => { const scopeProps = useScope(overrideScopes); const currentScope = scopeProps[`__scope${scopeName}`]; return { ...nextScopes2, ...currentScope }; }, {}); return React.useMemo(() => ({ [`__scope${baseScope.scopeName}`]: nextScopes }), [nextScopes]); }; }; createScope.scopeName = baseScope.scopeName; return createScope; } // node_modules/@radix-ui/react-collection/dist/index.mjs var import_jsx_runtime2 = __toESM(require_jsx_runtime(), 1); function createCollection(name) { const PROVIDER_NAME2 = name + "CollectionProvider"; const [createCollectionContext, createCollectionScope2] = createContextScope2(PROVIDER_NAME2); const [CollectionProviderImpl, useCollectionContext] = createCollectionContext( PROVIDER_NAME2, { collectionRef: { current: null }, itemMap: /* @__PURE__ */ new Map() } ); const CollectionProvider = (props) => { const { scope, children } = props; const ref = import_react.default.useRef(null); const itemMap = import_react.default.useRef(/* @__PURE__ */ new Map()).current; return (0, import_jsx_runtime2.jsx)(CollectionProviderImpl, { scope, itemMap, collectionRef: ref, children }); }; CollectionProvider.displayName = PROVIDER_NAME2; const COLLECTION_SLOT_NAME = name + "CollectionSlot"; const CollectionSlot = import_react.default.forwardRef( (props, forwardedRef) => { const { scope, children } = props; const context = useCollectionContext(COLLECTION_SLOT_NAME, scope); const composedRefs = useComposedRefs(forwardedRef, context.collectionRef); return (0, import_jsx_runtime2.jsx)(Slot, { ref: composedRefs, children }); } ); CollectionSlot.displayName = COLLECTION_SLOT_NAME; const ITEM_SLOT_NAME = name + "CollectionItemSlot"; const ITEM_DATA_ATTR = "data-radix-collection-item"; const CollectionItemSlot = import_react.default.forwardRef( (props, forwardedRef) => { const { scope, children, ...itemData } = props; const ref = import_react.default.useRef(null); const composedRefs = useComposedRefs(forwardedRef, ref); const context = useCollectionContext(ITEM_SLOT_NAME, scope); import_react.default.useEffect(() => { context.itemMap.set(ref, { ref, ...itemData }); return () => void context.itemMap.delete(ref); }); return (0, import_jsx_runtime2.jsx)(Slot, { ...{ [ITEM_DATA_ATTR]: "" }, ref: composedRefs, children }); } ); CollectionItemSlot.displayName = ITEM_SLOT_NAME; function useCollection2(scope) { const context = useCollectionContext(name + "CollectionConsumer", scope); const getItems = import_react.default.useCallback(() => { const collectionNode = context.collectionRef.current; if (!collectionNode) return []; const orderedNodes = Array.from(collectionNode.querySelectorAll(`[${ITEM_DATA_ATTR}]`)); const items = Array.from(context.itemMap.values()); const orderedItems = items.sort( (a, b) => orderedNodes.indexOf(a.ref.current) - orderedNodes.indexOf(b.ref.current) ); return orderedItems; }, [context.collectionRef, context.itemMap]); return getItems; } return [ { Provider: CollectionProvider, Slot: CollectionSlot, ItemSlot: CollectionItemSlot }, useCollection2, createCollectionScope2 ]; } // node_modules/@radix-ui/react-dismissable-layer/dist/index.mjs var React4 = __toESM(require_react(), 1); // node_modules/@radix-ui/react-use-escape-keydown/dist/index.mjs var React3 = __toESM(require_react(), 1); function useEscapeKeydown(onEscapeKeyDownProp, ownerDocument = globalThis == null ? void 0 : globalThis.document) { const onEscapeKeyDown = useCallbackRef(onEscapeKeyDownProp); React3.useEffect(() => { const handleKeyDown = (event) => { if (event.key === "Escape") { onEscapeKeyDown(event); } }; ownerDocument.addEventListener("keydown", handleKeyDown, { capture: true }); return () => ownerDocument.removeEventListener("keydown", handleKeyDown, { capture: true }); }, [onEscapeKeyDown, ownerDocument]); } // node_modules/@radix-ui/react-dismissable-layer/dist/index.mjs var import_jsx_runtime3 = __toESM(require_jsx_runtime(), 1); var DISMISSABLE_LAYER_NAME = "DismissableLayer"; var CONTEXT_UPDATE = "dismissableLayer.update"; var POINTER_DOWN_OUTSIDE = "dismissableLayer.pointerDownOutside"; var FOCUS_OUTSIDE = "dismissableLayer.focusOutside"; var originalBodyPointerEvents; var DismissableLayerContext = React4.createContext({ layers: /* @__PURE__ */ new Set(), layersWithOutsidePointerEventsDisabled: /* @__PURE__ */ new Set(), branches: /* @__PURE__ */ new Set() }); var DismissableLayer = React4.forwardRef( (props, forwardedRef) => { const { disableOutsidePointerEvents = false, onEscapeKeyDown, onPointerDownOutside, onFocusOutside, onInteractOutside, onDismiss, ...layerProps } = props; const context = React4.useContext(DismissableLayerContext); const [node, setNode] = React4.useState(null); const ownerDocument = (node == null ? void 0 : node.ownerDocument) ?? (globalThis == null ? void 0 : globalThis.document); const [, force] = React4.useState({}); const composedRefs = useComposedRefs(forwardedRef, (node2) => setNode(node2)); const layers = Array.from(context.layers); const [highestLayerWithOutsidePointerEventsDisabled] = [...context.layersWithOutsidePointerEventsDisabled].slice(-1); const highestLayerWithOutsidePointerEventsDisabledIndex = layers.indexOf(highestLayerWithOutsidePointerEventsDisabled); const index = node ? layers.indexOf(node) : -1; const isBodyPointerEventsDisabled = context.layersWithOutsidePointerEventsDisabled.size > 0; const isPointerEventsEnabled = index >= highestLayerWithOutsidePointerEventsDisabledIndex; const pointerDownOutside = usePointerDownOutside((event) => { const target = event.target; const isPointerDownOnBranch = [...context.branches].some((branch) => branch.contains(target)); if (!isPointerEventsEnabled || isPointerDownOnBranch) return; onPointerDownOutside == null ? void 0 : onPointerDownOutside(event); onInteractOutside == null ? void 0 : onInteractOutside(event); if (!event.defaultPrevented) onDismiss == null ? void 0 : onDismiss(); }, ownerDocument); const focusOutside = useFocusOutside((event) => { const target = event.target; const isFocusInBranch = [...context.branches].some((branch) => branch.contains(target)); if (isFocusInBranch) return; onFocusOutside == null ? void 0 : onFocusOutside(event); onInteractOutside == null ? void 0 : onInteractOutside(event); if (!event.defaultPrevented) onDismiss == null ? void 0 : onDismiss(); }, ownerDocument); useEscapeKeydown((event) => { const isHighestLayer = index === context.layers.size - 1; if (!isHighestLayer) return; onEscapeKeyDown == null ? void 0 : onEscapeKeyDown(event); if (!event.defaultPrevented && onDismiss) { event.preventDefault(); onDismiss(); } }, ownerDocument); React4.useEffect(() => { if (!node) return; if (disableOutsidePointerEvents) { if (context.layersWithOutsidePointerEventsDisabled.size === 0) { originalBodyPointerEvents = ownerDocument.body.style.pointerEvents; ownerDocument.body.style.pointerEvents = "none"; } context.layersWithOutsidePointerEventsDisabled.add(node); } context.layers.add(node); dispatchUpdate(); return () => { if (disableOutsidePointerEvents && context.layersWithOutsidePointerEventsDisabled.size === 1) { ownerDocument.body.style.pointerEvents = originalBodyPointerEvents; } }; }, [node, ownerDocument, disableOutsidePointerEvents, context]); React4.useEffect(() => { return () => { if (!node) return; context.layers.delete(node); context.layersWithOutsidePointerEventsDisabled.delete(node); dispatchUpdate(); }; }, [node, context]); React4.useEffect(() => { const handleUpdate = () => force({}); document.addEventListener(CONTEXT_UPDATE, handleUpdate); return () => document.removeEventListener(CONTEXT_UPDATE, handleUpdate); }, []); return (0, import_jsx_runtime3.jsx)( Primitive.div, { ...layerProps, ref: composedRefs, style: { pointerEvents: isBodyPointerEventsDisabled ? isPointerEventsEnabled ? "auto" : "none" : void 0, ...props.style }, onFocusCapture: composeEventHandlers(props.onFocusCapture, focusOutside.onFocusCapture), onBlurCapture: composeEventHandlers(props.onBlurCapture, focusOutside.onBlurCapture), onPointerDownCapture: composeEventHandlers( props.onPointerDownCapture, pointerDownOutside.onPointerDownCapture ) } ); } ); DismissableLayer.displayName = DISMISSABLE_LAYER_NAME; var BRANCH_NAME = "DismissableLayerBranch"; var DismissableLayerBranch = React4.forwardRef((props, forwardedRef) => { const context = React4.useContext(DismissableLayerContext); const ref = React4.useRef(null); const composedRefs = useComposedRefs(forwardedRef, ref); React4.useEffect(() => { const node = ref.current; if (node) { context.branches.add(node); return () => { context.branches.delete(node); }; } }, [context.branches]); return (0, import_jsx_runtime3.jsx)(Primitive.div, { ...props, ref: composedRefs }); }); DismissableLayerBranch.displayName = BRANCH_NAME; function usePointerDownOutside(onPointerDownOutside, ownerDocument = globalThis == null ? void 0 : globalThis.document) { const handlePointerDownOutside = useCallbackRef(onPointerDownOutside); const isPointerInsideReactTreeRef = React4.useRef(false); const handleClickRef = React4.useRef(() => { }); React4.useEffect(() => { const handlePointerDown = (event) => { if (event.target && !isPointerInsideReactTreeRef.current) { let handleAndDispatchPointerDownOutsideEvent2 = function() { handleAndDispatchCustomEvent( POINTER_DOWN_OUTSIDE, handlePointerDownOutside, eventDetail, { discrete: true } ); }; var handleAndDispatchPointerDownOutsideEvent = handleAndDispatchPointerDownOutsideEvent2; const eventDetail = { originalEvent: event }; if (event.pointerType === "touch") { ownerDocument.removeEventListener("click", handleClickRef.current); handleClickRef.current = handleAndDispatchPointerDownOutsideEvent2; ownerDocument.addEventListener("click", handleClickRef.current, { once: true }); } else { handleAndDispatchPointerDownOutsideEvent2(); } } else { ownerDocument.removeEventListener("click", handleClickRef.current); } isPointerInsideReactTreeRef.current = false; }; const timerId = window.setTimeout(() => { ownerDocument.addEventListener("pointerdown", handlePointerDown); }, 0); return () => { window.clearTimeout(timerId); ownerDocument.removeEventListener("pointerdown", handlePointerDown); ownerDocument.removeEventListener("click", handleClickRef.current); }; }, [ownerDocument, handlePointerDownOutside]); return { // ensures we check React component tree (not just DOM tree) onPointerDownCapture: () => isPointerInsideReactTreeRef.current = true }; } function useFocusOutside(onFocusOutside, ownerDocument = globalThis == null ? void 0 : globalThis.document) { const handleFocusOutside = useCallbackRef(onFocusOutside); const isFocusInsideReactTreeRef = React4.useRef(false); React4.useEffect(() => { const handleFocus = (event) => { if (event.target && !isFocusInsideReactTreeRef.current) { const eventDetail = { originalEvent: event }; handleAndDispatchCustomEvent(FOCUS_OUTSIDE, handleFocusOutside, eventDetail, { discrete: false }); } }; ownerDocument.addEventListener("focusin", handleFocus); return () => ownerDocument.removeEventListener("focusin", handleFocus); }, [ownerDocument, handleFocusOutside]); return { onFocusCapture: () => isFocusInsideReactTreeRef.current = true, onBlurCapture: () => isFocusInsideReactTreeRef.current = false }; } function dispatchUpdate() { const event = new CustomEvent(CONTEXT_UPDATE); document.dispatchEvent(event); } function handleAndDispatchCustomEvent(name, handler, detail, { discrete }) { const target = detail.originalEvent.target; const event = new CustomEvent(name, { bubbles: false, cancelable: true, detail }); if (handler) target.addEventListener(name, handler, { once: true }); if (discrete) { dispatchDiscreteCustomEvent(target, event); } else { target.dispatchEvent(event); } } var Root = DismissableLayer; var Branch = DismissableLayerBranch; // node_modules/@radix-ui/react-portal/dist/index.mjs var React5 = __toESM(require_react(), 1); var import_react_dom = __toESM(require_react_dom(), 1); var import_jsx_runtime4 = __toESM(require_jsx_runtime(), 1); var PORTAL_NAME = "Portal"; var Portal = React5.forwardRef((props, forwardedRef) => { var _a; const { container: containerProp, ...portalProps } = props; const [mounted, setMounted] = React5.useState(false); useLayoutEffect2(() => setMounted(true), []); const container = containerProp || mounted && ((_a = globalThis == null ? void 0 : globalThis.document) == null ? void 0 : _a.body); return container ? import_react_dom.default.createPortal((0, import_jsx_runtime4.jsx)(Primitive.div, { ...portalProps, ref: forwardedRef }), container) : null; }); Portal.displayName = PORTAL_NAME; // node_modules/@radix-ui/react-use-controllable-state/dist/index.mjs var React6 = __toESM(require_react(), 1); function useControllableState({ prop, defaultProp, onChange = () => { } }) { const [uncontrolledProp, setUncontrolledProp] = useUncontrolledState({ defaultProp, onChange }); const isControlled = prop !== void 0; const value = isControlled ? prop : uncontrolledProp; const handleChange = useCallbackRef(onChange); const setValue = React6.useCallback( (nextValue) => { if (isControlled) { const setter = nextValue; const value2 = typeof nextValue === "function" ? setter(prop) : nextValue; if (value2 !== prop) handleChange(value2); } else { setUncontrolledProp(nextValue); } }, [isControlled, prop, setUncontrolledProp, handleChange] ); return [value, setValue]; } function useUncontrolledState({ defaultProp, onChange }) { const uncontrolledState = React6.useState(defaultProp); const [value] = uncontrolledState; const prevValueRef = React6.useRef(value); const handleChange = useCallbackRef(onChange); React6.useEffect(() => { if (prevValueRef.current !== value) { handleChange(value); prevValueRef.current = value; } }, [value, prevValueRef, handleChange]); return uncontrolledState; } // node_modules/@radix-ui/react-visually-hidden/dist/index.mjs var React7 = __toESM(require_react(), 1); var import_jsx_runtime5 = __toESM(require_jsx_runtime(), 1); var NAME = "VisuallyHidden"; var VisuallyHidden = React7.forwardRef( (props, forwardedRef) => { return (0, import_jsx_runtime5.jsx)( Primitive.span, { ...props, ref: forwardedRef, style: { // See: https://github.com/twbs/bootstrap/blob/master/scss/mixins/_screen-reader.scss position: "absolute", border: 0, width: 1, height: 1, padding: 0, margin: -1, overflow: "hidden", clip: "rect(0, 0, 0, 0)", whiteSpace: "nowrap", wordWrap: "normal", ...props.style } } ); } ); VisuallyHidden.displayName = NAME; // node_modules/@radix-ui/react-toast/dist/index.mjs var import_jsx_runtime6 = __toESM(require_jsx_runtime(), 1); var PROVIDER_NAME = "ToastProvider"; var [Collection, useCollection, createCollectionScope] = createCollection("Toast"); var [createToastContext, createToastScope] = createContextScope("Toast", [createCollectionScope]); var [ToastProviderProvider, useToastProviderContext] = createToastContext(PROVIDER_NAME); var ToastProvider = (props) => { const { __scopeToast, label = "Notification", duration = 5e3, swipeDirection = "right", swipeThreshold = 50, children } = props; const [viewport, setViewport] = React8.useState(null); const [toastCount, setToastCount] = React8.useState(0); const isFocusedToastEscapeKeyDownRef = React8.useRef(false); const isClosePausedRef = React8.useRef(false); if (!label.trim()) { console.error( `Invalid prop \`label\` supplied to \`${PROVIDER_NAME}\`. Expected non-empty \`string\`.` ); } return (0, import_jsx_runtime6.jsx)(Collection.Provider, { scope: __scopeToast, children: (0, import_jsx_runtime6.jsx)( ToastProviderProvider, { scope: __scopeToast, label, duration, swipeDirection, swipeThreshold, toastCount, viewport, onViewportChange: setViewport, onToastAdd: React8.useCallback(() => setToastCount((prevCount) => prevCount + 1), []), onToastRemove: React8.useCallback(() => setToastCount((prevCount) => prevCount - 1), []), isFocusedToastEscapeKeyDownRef, isClosePausedRef, children } ) }); }; ToastProvider.displayName = PROVIDER_NAME; var VIEWPORT_NAME = "ToastViewport"; var VIEWPORT_DEFAULT_HOTKEY = ["F8"]; var VIEWPORT_PAUSE = "toast.viewportPause"; var VIEWPORT_RESUME = "toast.viewportResume"; var ToastViewport = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, hotkey = VIEWPORT_DEFAULT_HOTKEY, label = "Notifications ({hotkey})", ...viewportProps } = props; const context = useToastProviderContext(VIEWPORT_NAME, __scopeToast); const getItems = useCollection(__scopeToast); const wrapperRef = React8.useRef(null); const headFocusProxyRef = React8.useRef(null); const tailFocusProxyRef = React8.useRef(null); const ref = React8.useRef(null); const composedRefs = useComposedRefs(forwardedRef, ref, context.onViewportChange); const hotkeyLabel = hotkey.join("+").replace(/Key/g, "").replace(/Digit/g, ""); const hasToasts = context.toastCount > 0; React8.useEffect(() => { const handleKeyDown = (event) => { var _a; const isHotkeyPressed = hotkey.length !== 0 && hotkey.every((key) => event[key] || event.code === key); if (isHotkeyPressed) (_a = ref.current) == null ? void 0 : _a.focus(); }; document.addEventListener("keydown", handleKeyDown); return () => document.removeEventListener("keydown", handleKeyDown); }, [hotkey]); React8.useEffect(() => { const wrapper = wrapperRef.current; const viewport = ref.current; if (hasToasts && wrapper && viewport) { const handlePause = () => { if (!context.isClosePausedRef.current) { const pauseEvent = new CustomEvent(VIEWPORT_PAUSE); viewport.dispatchEvent(pauseEvent); context.isClosePausedRef.current = true; } }; const handleResume = () => { if (context.isClosePausedRef.current) { const resumeEvent = new CustomEvent(VIEWPORT_RESUME); viewport.dispatchEvent(resumeEvent); context.isClosePausedRef.current = false; } }; const handleFocusOutResume = (event) => { const isFocusMovingOutside = !wrapper.contains(event.relatedTarget); if (isFocusMovingOutside) handleResume(); }; const handlePointerLeaveResume = () => { const isFocusInside = wrapper.contains(document.activeElement); if (!isFocusInside) handleResume(); }; wrapper.addEventListener("focusin", handlePause); wrapper.addEventListener("focusout", handleFocusOutResume); wrapper.addEventListener("pointermove", handlePause); wrapper.addEventListener("pointerleave", handlePointerLeaveResume); window.addEventListener("blur", handlePause); window.addEventListener("focus", handleResume); return () => { wrapper.removeEventListener("focusin", handlePause); wrapper.removeEventListener("focusout", handleFocusOutResume); wrapper.removeEventListener("pointermove", handlePause); wrapper.removeEventListener("pointerleave", handlePointerLeaveResume); window.removeEventListener("blur", handlePause); window.removeEventListener("focus", handleResume); }; } }, [hasToasts, context.isClosePausedRef]); const getSortedTabbableCandidates = React8.useCallback( ({ tabbingDirection }) => { const toastItems = getItems(); const tabbableCandidates = toastItems.map((toastItem) => { const toastNode = toastItem.ref.current; const toastTabbableCandidates = [toastNode, ...getTabbableCandidates(toastNode)]; return tabbingDirection === "forwards" ? toastTabbableCandidates : toastTabbableCandidates.reverse(); }); return (tabbingDirection === "forwards" ? tabbableCandidates.reverse() : tabbableCandidates).flat(); }, [getItems] ); React8.useEffect(() => { const viewport = ref.current; if (viewport) { const handleKeyDown = (event) => { var _a, _b, _c; const isMetaKey = event.altKey || event.ctrlKey || event.metaKey; const isTabKey = event.key === "Tab" && !isMetaKey; if (isTabKey) { const focusedElement = document.activeElement; const isTabbingBackwards = event.shiftKey; const targetIsViewport = event.target === viewport; if (targetIsViewport && isTabbingBackwards) { (_a = headFocusProxyRef.current) == null ? void 0 : _a.focus(); return; } const tabbingDirection = isTabbingBackwards ? "backwards" : "forwards"; const sortedCandidates = getSortedTabbableCandidates({ tabbingDirection }); const index = sortedCandidates.findIndex((candidate) => candidate === focusedElement); if (focusFirst(sortedCandidates.slice(index + 1))) { event.preventDefault(); } else { isTabbingBackwards ? (_b = headFocusProxyRef.current) == null ? void 0 : _b.focus() : (_c = tailFocusProxyRef.current) == null ? void 0 : _c.focus(); } } }; viewport.addEventListener("keydown", handleKeyDown); return () => viewport.removeEventListener("keydown", handleKeyDown); } }, [getItems, getSortedTabbableCandidates]); return (0, import_jsx_runtime6.jsxs)( Branch, { ref: wrapperRef, role: "region", "aria-label": label.replace("{hotkey}", hotkeyLabel), tabIndex: -1, style: { pointerEvents: hasToasts ? void 0 : "none" }, children: [ hasToasts && (0, import_jsx_runtime6.jsx)( FocusProxy, { ref: headFocusProxyRef, onFocusFromOutsideViewport: () => { const tabbableCandidates = getSortedTabbableCandidates({ tabbingDirection: "forwards" }); focusFirst(tabbableCandidates); } } ), (0, import_jsx_runtime6.jsx)(Collection.Slot, { scope: __scopeToast, children: (0, import_jsx_runtime6.jsx)(Primitive.ol, { tabIndex: -1, ...viewportProps, ref: composedRefs }) }), hasToasts && (0, import_jsx_runtime6.jsx)( FocusProxy, { ref: tailFocusProxyRef, onFocusFromOutsideViewport: () => { const tabbableCandidates = getSortedTabbableCandidates({ tabbingDirection: "backwards" }); focusFirst(tabbableCandidates); } } ) ] } ); } ); ToastViewport.displayName = VIEWPORT_NAME; var FOCUS_PROXY_NAME = "ToastFocusProxy"; var FocusProxy = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, onFocusFromOutsideViewport, ...proxyProps } = props; const context = useToastProviderContext(FOCUS_PROXY_NAME, __scopeToast); return (0, import_jsx_runtime6.jsx)( VisuallyHidden, { "aria-hidden": true, tabIndex: 0, ...proxyProps, ref: forwardedRef, style: { position: "fixed" }, onFocus: (event) => { var _a; const prevFocusedElement = event.relatedTarget; const isFocusFromOutsideViewport = !((_a = context.viewport) == null ? void 0 : _a.contains(prevFocusedElement)); if (isFocusFromOutsideViewport) onFocusFromOutsideViewport(); } } ); } ); FocusProxy.displayName = FOCUS_PROXY_NAME; var TOAST_NAME = "Toast"; var TOAST_SWIPE_START = "toast.swipeStart"; var TOAST_SWIPE_MOVE = "toast.swipeMove"; var TOAST_SWIPE_CANCEL = "toast.swipeCancel"; var TOAST_SWIPE_END = "toast.swipeEnd"; var Toast = React8.forwardRef( (props, forwardedRef) => { const { forceMount, open: openProp, defaultOpen, onOpenChange, ...toastProps } = props; const [open = true, setOpen] = useControllableState({ prop: openProp, defaultProp: defaultOpen, onChange: onOpenChange }); return (0, import_jsx_runtime6.jsx)(Presence, { present: forceMount || open, children: (0, import_jsx_runtime6.jsx)( ToastImpl, { open, ...toastProps, ref: forwardedRef, onClose: () => setOpen(false), onPause: useCallbackRef(props.onPause), onResume: useCallbackRef(props.onResume), onSwipeStart: composeEventHandlers(props.onSwipeStart, (event) => { event.currentTarget.setAttribute("data-swipe", "start"); }), onSwipeMove: composeEventHandlers(props.onSwipeMove, (event) => { const { x, y } = event.detail.delta; event.currentTarget.setAttribute("data-swipe", "move"); event.currentTarget.style.setProperty("--radix-toast-swipe-move-x", `${x}px`); event.currentTarget.style.setProperty("--radix-toast-swipe-move-y", `${y}px`); }), onSwipeCancel: composeEventHandlers(props.onSwipeCancel, (event) => { event.currentTarget.setAttribute("data-swipe", "cancel"); event.currentTarget.style.removeProperty("--radix-toast-swipe-move-x"); event.currentTarget.style.removeProperty("--radix-toast-swipe-move-y"); event.currentTarget.style.removeProperty("--radix-toast-swipe-end-x"); event.currentTarget.style.removeProperty("--radix-toast-swipe-end-y"); }), onSwipeEnd: composeEventHandlers(props.onSwipeEnd, (event) => { const { x, y } = event.detail.delta; event.currentTarget.setAttribute("data-swipe", "end"); event.currentTarget.style.removeProperty("--radix-toast-swipe-move-x"); event.currentTarget.style.removeProperty("--radix-toast-swipe-move-y"); event.currentTarget.style.setProperty("--radix-toast-swipe-end-x", `${x}px`); event.currentTarget.style.setProperty("--radix-toast-swipe-end-y", `${y}px`); setOpen(false); }) } ) }); } ); Toast.displayName = TOAST_NAME; var [ToastInteractiveProvider, useToastInteractiveContext] = createToastContext(TOAST_NAME, { onClose() { } }); var ToastImpl = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, type = "foreground", duration: durationProp, open, onClose, onEscapeKeyDown, onPause, onResume, onSwipeStart, onSwipeMove, onSwipeCancel, onSwipeEnd, ...toastProps } = props; const context = useToastProviderContext(TOAST_NAME, __scopeToast); const [node, setNode] = React8.useState(null); const composedRefs = useComposedRefs(forwardedRef, (node2) => setNode(node2)); const pointerStartRef = React8.useRef(null); const swipeDeltaRef = React8.useRef(null); const duration = durationProp || context.duration; const closeTimerStartTimeRef = React8.useRef(0); const closeTimerRemainingTimeRef = React8.useRef(duration); const closeTimerRef = React8.useRef(0); const { onToastAdd, onToastRemove } = context; const handleClose = useCallbackRef(() => { var _a; const isFocusInToast = node == null ? void 0 : node.contains(document.activeElement); if (isFocusInToast) (_a = context.viewport) == null ? void 0 : _a.focus(); onClose(); }); const startTimer = React8.useCallback( (duration2) => { if (!duration2 || duration2 === Infinity) return; window.clearTimeout(closeTimerRef.current); closeTimerStartTimeRef.current = (/* @__PURE__ */ new Date()).getTime(); closeTimerRef.current = window.setTimeout(handleClose, duration2); }, [handleClose] ); React8.useEffect(() => { const viewport = context.viewport; if (viewport) { const handleResume = () => { startTimer(closeTimerRemainingTimeRef.current); onResume == null ? void 0 : onResume(); }; const handlePause = () => { const elapsedTime = (/* @__PURE__ */ new Date()).getTime() - closeTimerStartTimeRef.current; closeTimerRemainingTimeRef.current = closeTimerRemainingTimeRef.current - elapsedTime; window.clearTimeout(closeTimerRef.current); onPause == null ? void 0 : onPause(); }; viewport.addEventListener(VIEWPORT_PAUSE, handlePause); viewport.addEventListener(VIEWPORT_RESUME, handleResume); return () => { viewport.removeEventListener(VIEWPORT_PAUSE, handlePause); viewport.removeEventListener(VIEWPORT_RESUME, handleResume); }; } }, [context.viewport, duration, onPause, onResume, startTimer]); React8.useEffect(() => { if (open && !context.isClosePausedRef.current) startTimer(duration); }, [open, duration, context.isClosePausedRef, startTimer]); React8.useEffect(() => { onToastAdd(); return () => onToastRemove(); }, [onToastAdd, onToastRemove]); const announceTextContent = React8.useMemo(() => { return node ? getAnnounceTextContent(node) : null; }, [node]); if (!context.viewport) return null; return (0, import_jsx_runtime6.jsxs)(import_jsx_runtime6.Fragment, { children: [ announceTextContent && (0, import_jsx_runtime6.jsx)( ToastAnnounce, { __scopeToast, role: "status", "aria-live": type === "foreground" ? "assertive" : "polite", "aria-atomic": true, children: announceTextContent } ), (0, import_jsx_runtime6.jsx)(ToastInteractiveProvider, { scope: __scopeToast, onClose: handleClose, children: ReactDOM2.createPortal( (0, import_jsx_runtime6.jsx)(Collection.ItemSlot, { scope: __scopeToast, children: (0, import_jsx_runtime6.jsx)( Root, { asChild: true, onEscapeKeyDown: composeEventHandlers(onEscapeKeyDown, () => { if (!context.isFocusedToastEscapeKeyDownRef.current) handleClose(); context.isFocusedToastEscapeKeyDownRef.current = false; }), children: (0, import_jsx_runtime6.jsx)( Primitive.li, { role: "status", "aria-live": "off", "aria-atomic": true, tabIndex: 0, "data-state": open ? "open" : "closed", "data-swipe-direction": context.swipeDirection, ...toastProps, ref: composedRefs, style: { userSelect: "none", touchAction: "none", ...props.style }, onKeyDown: composeEventHandlers(props.onKeyDown, (event) => { if (event.key !== "Escape") return; onEscapeKeyDown == null ? void 0 : onEscapeKeyDown(event.nativeEvent); if (!event.nativeEvent.defaultPrevented) { context.isFocusedToastEscapeKeyDownRef.current = true; handleClose(); } }), onPointerDown: composeEventHandlers(props.onPointerDown, (event) => { if (event.button !== 0) return; pointerStartRef.current = { x: event.clientX, y: event.clientY }; }), onPointerMove: composeEventHandlers(props.onPointerMove, (event) => { if (!pointerStartRef.current) return; const x = event.clientX - pointerStartRef.current.x; const y = event.clientY - pointerStartRef.current.y; const hasSwipeMoveStarted = Boolean(swipeDeltaRef.current); const isHorizontalSwipe = ["left", "right"].includes(context.swipeDirection); const clamp = ["left", "up"].includes(context.swipeDirection) ? Math.min : Math.max; const clampedX = isHorizontalSwipe ? clamp(0, x) : 0; const clampedY = !isHorizontalSwipe ? clamp(0, y) : 0; const moveStartBuffer = event.pointerType === "touch" ? 10 : 2; const delta = { x: clampedX, y: clampedY }; const eventDetail = { originalEvent: event, delta }; if (hasSwipeMoveStarted) { swipeDeltaRef.current = delta; handleAndDispatchCustomEvent2(TOAST_SWIPE_MOVE, onSwipeMove, eventDetail, { discrete: false }); } else if (isDeltaInDirection(delta, context.swipeDirection, moveStartBuffer)) { swipeDeltaRef.current = delta; handleAndDispatchCustomEvent2(TOAST_SWIPE_START, onSwipeStart, eventDetail, { discrete: false }); event.target.setPointerCapture(event.pointerId); } else if (Math.abs(x) > moveStartBuffer || Math.abs(y) > moveStartBuffer) { pointerStartRef.current = null; } }), onPointerUp: composeEventHandlers(props.onPointerUp, (event) => { const delta = swipeDeltaRef.current; const target = event.target; if (target.hasPointerCapture(event.pointerId)) { target.releasePointerCapture(event.pointerId); } swipeDeltaRef.current = null; pointerStartRef.current = null; if (delta) { const toast = event.currentTarget; const eventDetail = { originalEvent: event, delta }; if (isDeltaInDirection(delta, context.swipeDirection, context.swipeThreshold)) { handleAndDispatchCustomEvent2(TOAST_SWIPE_END, onSwipeEnd, eventDetail, { discrete: true }); } else { handleAndDispatchCustomEvent2( TOAST_SWIPE_CANCEL, onSwipeCancel, eventDetail, { discrete: true } ); } toast.addEventListener("click", (event2) => event2.preventDefault(), { once: true }); } }) } ) } ) }), context.viewport ) }) ] }); } ); var ToastAnnounce = (props) => { const { __scopeToast, children, ...announceProps } = props; const context = useToastProviderContext(TOAST_NAME, __scopeToast); const [renderAnnounceText, setRenderAnnounceText] = React8.useState(false); const [isAnnounced, setIsAnnounced] = React8.useState(false); useNextFrame(() => setRenderAnnounceText(true)); React8.useEffect(() => { const timer = window.setTimeout(() => setIsAnnounced(true), 1e3); return () => window.clearTimeout(timer); }, []); return isAnnounced ? null : (0, import_jsx_runtime6.jsx)(Portal, { asChild: true, children: (0, import_jsx_runtime6.jsx)(VisuallyHidden, { ...announceProps, children: renderAnnounceText && (0, import_jsx_runtime6.jsxs)(import_jsx_runtime6.Fragment, { children: [ context.label, " ", children ] }) }) }); }; var TITLE_NAME = "ToastTitle"; var ToastTitle = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, ...titleProps } = props; return (0, import_jsx_runtime6.jsx)(Primitive.div, { ...titleProps, ref: forwardedRef }); } ); ToastTitle.displayName = TITLE_NAME; var DESCRIPTION_NAME = "ToastDescription"; var ToastDescription = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, ...descriptionProps } = props; return (0, import_jsx_runtime6.jsx)(Primitive.div, { ...descriptionProps, ref: forwardedRef }); } ); ToastDescription.displayName = DESCRIPTION_NAME; var ACTION_NAME = "ToastAction"; var ToastAction = React8.forwardRef( (props, forwardedRef) => { const { altText, ...actionProps } = props; if (!altText.trim()) { console.error( `Invalid prop \`altText\` supplied to \`${ACTION_NAME}\`. Expected non-empty \`string\`.` ); return null; } return (0, import_jsx_runtime6.jsx)(ToastAnnounceExclude, { altText, asChild: true, children: (0, import_jsx_runtime6.jsx)(ToastClose, { ...actionProps, ref: forwardedRef }) }); } ); ToastAction.displayName = ACTION_NAME; var CLOSE_NAME = "ToastClose"; var ToastClose = React8.forwardRef( (props, forwardedRef) => { const { __scopeToast, ...closeProps } = props; const interactiveContext = useToastInteractiveContext(CLOSE_NAME, __scopeToast); return (0, import_jsx_runtime6.jsx)(ToastAnnounceExclude, { asChild: true, children: (0, import_jsx_runtime6.jsx)( Primitive.button, { type: "button", ...closeProps, ref: forwardedRef, onClick: composeEventHandlers(props.onClick, interactiveContext.onClose) } ) }); } ); ToastClose.displayName = CLOSE_NAME; var ToastAnnounceExclude = React8.forwardRef((props, forwardedRef) => { const { __scopeToast, altText, ...announceExcludeProps } = props; return (0, import_jsx_runtime6.jsx)( Primitive.div, { "data-radix-toast-announce-exclude": "", "data-radix-toast-announce-alt": altText || void 0, ...announceExcludeProps, ref: forwardedRef } ); }); function getAnnounceTextContent(container) { const textContent = []; const childNodes = Array.from(container.childNodes); childNodes.forEach((node) => { if (node.nodeType === node.TEXT_NODE && node.textContent) textContent.push(node.textContent); if (isHTMLElement(node)) { const isHidden = node.ariaHidden || node.hidden || node.style.display === "none"; const isExcluded = node.dataset.radixToastAnnounceExclude === ""; if (!isHidden) { if (isExcluded) { const altText = node.dataset.radixToastAnnounceAlt; if (altText) textContent.push(altText); } else { textContent.push(...getAnnounceTextContent(node)); } } } }); return textContent; } function handleAndDispatchCustomEvent2(name, handler, detail, { discrete }) { const currentTarget = detail.originalEvent.currentTarget; const event = new CustomEvent(name, { bubbles: true, cancelable: true, detail }); if (handler) currentTarget.addEventListener(name, handler, { once: true }); if (discrete) { dispatchDiscreteCustomEvent(currentTarget, event); } else { currentTarget.dispatchEvent(event); } } var isDeltaInDirection = (delta, direction, threshold = 0) => { const deltaX = Math.abs(delta.x); const deltaY = Math.abs(delta.y); const isDeltaX = deltaX > deltaY; if (direction === "left" || direction === "right") { return isDeltaX && deltaX > threshold; } else { return !isDeltaX && deltaY > threshold; } }; function useNextFrame(callback = () => { }) { const fn = useCallbackRef(callback); useLayoutEffect2(() => { let raf1 = 0; let raf2 = 0; raf1 = window.requestAnimationFrame(() => raf2 = window.requestAnimationFrame(fn)); return () => { window.cancelAnimationFrame(raf1); window.cancelAnimationFrame(raf2); }; }, [fn]); } function isHTMLElement(node) { return node.nodeType === node.ELEMENT_NODE; } function getTabbableCandidates(container) { const nodes = []; const walker = document.createTreeWalker(container, NodeFilter.SHOW_ELEMENT, { acceptNode: (node) => { const isHiddenInput = node.tagName === "INPUT" && node.type === "hidden"; if (node.disabled || node.hidden || isHiddenInput) return NodeFilter.FILTER_SKIP; return node.tabIndex >= 0 ? NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP; } }); while (walker.nextNode()) nodes.push(walker.currentNode); return nodes; } function focusFirst(candidates) { const previouslyFocusedElement = document.activeElement; return candidates.some((candidate) => { if (candidate === previouslyFocusedElement) return true; candidate.focus(); return document.activeElement !== previouslyFocusedElement; }); } var Provider = ToastProvider; var Viewport = ToastViewport; var Root2 = Toast; var Title = ToastTitle; var Description = ToastDescription; var Action = ToastAction; var Close = ToastClose; export { Action, Close, Description, Provider, Root2 as Root, Title, Toast, ToastAction, ToastClose, ToastDescription, ToastProvider, ToastTitle, ToastViewport, Viewport, createToastScope }; //# sourceMappingURL=@radix-ui_react-toast.js.map