- ✅ Ticket 1.1: Estructura Clean Architecture en backend - ✅ Ticket 1.2: Schemas Zod compartidos - ✅ Ticket 1.3: Refactorización drugs.ts (1362 → 8 archivos modulares) - ✅ Ticket 1.4: Refactorización procedures.ts (3583 → 6 archivos modulares) - ✅ Ticket 1.5: Eliminación de duplicidades (~50 líneas) Cambios principales: - Creada estructura Clean Architecture en backend/src/ - Schemas Zod compartidos en backend/src/shared/schemas/ - Refactorización modular de drugs y procedures - Utilidades genéricas en src/utils/ (filter, validation) - Eliminados scripts obsoletos y documentación antigua - Corregidos errores: QueryClient, import test-error-handling - Build verificado y funcionando correctamente
832 lines
33 KiB
JavaScript
832 lines
33 KiB
JavaScript
"use client";
|
|
import {
|
|
Branch,
|
|
Portal,
|
|
Presence,
|
|
Primitive,
|
|
Root,
|
|
VisuallyHidden,
|
|
composeEventHandlers,
|
|
createContextScope,
|
|
dispatchDiscreteCustomEvent,
|
|
useControllableState,
|
|
useLayoutEffect2
|
|
} from "./chunk-QHJT3T76.js";
|
|
import {
|
|
useCallbackRef
|
|
} from "./chunk-IFCS6WH6.js";
|
|
import {
|
|
require_react_dom
|
|
} from "./chunk-PBX5ABKV.js";
|
|
import {
|
|
composeRefs,
|
|
useComposedRefs
|
|
} from "./chunk-OHANU4DK.js";
|
|
import {
|
|
require_jsx_runtime
|
|
} from "./chunk-4UTF2CDO.js";
|
|
import {
|
|
require_react
|
|
} from "./chunk-BXEBRY3I.js";
|
|
import {
|
|
__toESM
|
|
} from "./chunk-V4OQ3NZ2.js";
|
|
|
|
// node_modules/@radix-ui/react-toast/dist/index.mjs
|
|
var React3 = __toESM(require_react(), 1);
|
|
var ReactDOM = __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-slot/dist/index.mjs
|
|
var React = __toESM(require_react(), 1);
|
|
var import_jsx_runtime = __toESM(require_jsx_runtime(), 1);
|
|
function createSlot(ownerName) {
|
|
const SlotClone = createSlotClone(ownerName);
|
|
const Slot2 = React.forwardRef((props, forwardedRef) => {
|
|
const { children, ...slotProps } = props;
|
|
const childrenArray = React.Children.toArray(children);
|
|
const slottable = childrenArray.find(isSlottable);
|
|
if (slottable) {
|
|
const newElement = slottable.props.children;
|
|
const newChildren = childrenArray.map((child) => {
|
|
if (child === slottable) {
|
|
if (React.Children.count(newElement) > 1) return React.Children.only(null);
|
|
return React.isValidElement(newElement) ? newElement.props.children : null;
|
|
} else {
|
|
return child;
|
|
}
|
|
});
|
|
return (0, import_jsx_runtime.jsx)(SlotClone, { ...slotProps, ref: forwardedRef, children: React.isValidElement(newElement) ? React.cloneElement(newElement, void 0, newChildren) : null });
|
|
}
|
|
return (0, import_jsx_runtime.jsx)(SlotClone, { ...slotProps, ref: forwardedRef, children });
|
|
});
|
|
Slot2.displayName = `${ownerName}.Slot`;
|
|
return Slot2;
|
|
}
|
|
var Slot = createSlot("Slot");
|
|
function createSlotClone(ownerName) {
|
|
const SlotClone = React.forwardRef((props, forwardedRef) => {
|
|
const { children, ...slotProps } = props;
|
|
if (React.isValidElement(children)) {
|
|
const childrenRef = getElementRef(children);
|
|
const props2 = mergeProps(slotProps, children.props);
|
|
if (children.type !== React.Fragment) {
|
|
props2.ref = forwardedRef ? composeRefs(forwardedRef, childrenRef) : childrenRef;
|
|
}
|
|
return React.cloneElement(children, props2);
|
|
}
|
|
return React.Children.count(children) > 1 ? React.Children.only(null) : null;
|
|
});
|
|
SlotClone.displayName = `${ownerName}.SlotClone`;
|
|
return SlotClone;
|
|
}
|
|
var SLOTTABLE_IDENTIFIER = /* @__PURE__ */ Symbol("radix.slottable");
|
|
function createSlottable(ownerName) {
|
|
const Slottable2 = ({ children }) => {
|
|
return (0, import_jsx_runtime.jsx)(import_jsx_runtime.Fragment, { children });
|
|
};
|
|
Slottable2.displayName = `${ownerName}.Slottable`;
|
|
Slottable2.__radixId = SLOTTABLE_IDENTIFIER;
|
|
return Slottable2;
|
|
}
|
|
var Slottable = createSlottable("Slottable");
|
|
function isSlottable(child) {
|
|
return React.isValidElement(child) && typeof child.type === "function" && "__radixId" in child.type && child.type.__radixId === SLOTTABLE_IDENTIFIER;
|
|
}
|
|
function mergeProps(slotProps, childProps) {
|
|
const overrideProps = { ...childProps };
|
|
for (const propName in childProps) {
|
|
const slotPropValue = slotProps[propName];
|
|
const childPropValue = childProps[propName];
|
|
const isHandler = /^on[A-Z]/.test(propName);
|
|
if (isHandler) {
|
|
if (slotPropValue && childPropValue) {
|
|
overrideProps[propName] = (...args) => {
|
|
const result = childPropValue(...args);
|
|
slotPropValue(...args);
|
|
return result;
|
|
};
|
|
} else if (slotPropValue) {
|
|
overrideProps[propName] = slotPropValue;
|
|
}
|
|
} else if (propName === "style") {
|
|
overrideProps[propName] = { ...slotPropValue, ...childPropValue };
|
|
} else if (propName === "className") {
|
|
overrideProps[propName] = [slotPropValue, childPropValue].filter(Boolean).join(" ");
|
|
}
|
|
}
|
|
return { ...slotProps, ...overrideProps };
|
|
}
|
|
function getElementRef(element) {
|
|
let getter = Object.getOwnPropertyDescriptor(element.props, "ref")?.get;
|
|
let mayWarn = getter && "isReactWarning" in getter && getter.isReactWarning;
|
|
if (mayWarn) {
|
|
return element.ref;
|
|
}
|
|
getter = Object.getOwnPropertyDescriptor(element, "ref")?.get;
|
|
mayWarn = getter && "isReactWarning" in getter && getter.isReactWarning;
|
|
if (mayWarn) {
|
|
return element.props.ref;
|
|
}
|
|
return element.props.ref || element.ref;
|
|
}
|
|
|
|
// node_modules/@radix-ui/react-collection/dist/index.mjs
|
|
var import_jsx_runtime2 = __toESM(require_jsx_runtime(), 1);
|
|
var import_react2 = __toESM(require_react(), 1);
|
|
var import_jsx_runtime3 = __toESM(require_jsx_runtime(), 1);
|
|
function createCollection(name) {
|
|
const PROVIDER_NAME2 = name + "CollectionProvider";
|
|
const [createCollectionContext, createCollectionScope2] = createContextScope(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 CollectionSlotImpl = createSlot(COLLECTION_SLOT_NAME);
|
|
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)(CollectionSlotImpl, { ref: composedRefs, children });
|
|
}
|
|
);
|
|
CollectionSlot.displayName = COLLECTION_SLOT_NAME;
|
|
const ITEM_SLOT_NAME = name + "CollectionItemSlot";
|
|
const ITEM_DATA_ATTR = "data-radix-collection-item";
|
|
const CollectionItemSlotImpl = createSlot(ITEM_SLOT_NAME);
|
|
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)(CollectionItemSlotImpl, { ...{ [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-toast/dist/index.mjs
|
|
var import_jsx_runtime4 = __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] = React3.useState(null);
|
|
const [toastCount, setToastCount] = React3.useState(0);
|
|
const isFocusedToastEscapeKeyDownRef = React3.useRef(false);
|
|
const isClosePausedRef = React3.useRef(false);
|
|
if (!label.trim()) {
|
|
console.error(
|
|
`Invalid prop \`label\` supplied to \`${PROVIDER_NAME}\`. Expected non-empty \`string\`.`
|
|
);
|
|
}
|
|
return (0, import_jsx_runtime4.jsx)(Collection.Provider, { scope: __scopeToast, children: (0, import_jsx_runtime4.jsx)(
|
|
ToastProviderProvider,
|
|
{
|
|
scope: __scopeToast,
|
|
label,
|
|
duration,
|
|
swipeDirection,
|
|
swipeThreshold,
|
|
toastCount,
|
|
viewport,
|
|
onViewportChange: setViewport,
|
|
onToastAdd: React3.useCallback(() => setToastCount((prevCount) => prevCount + 1), []),
|
|
onToastRemove: React3.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 = React3.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 = React3.useRef(null);
|
|
const headFocusProxyRef = React3.useRef(null);
|
|
const tailFocusProxyRef = React3.useRef(null);
|
|
const ref = React3.useRef(null);
|
|
const composedRefs = useComposedRefs(forwardedRef, ref, context.onViewportChange);
|
|
const hotkeyLabel = hotkey.join("+").replace(/Key/g, "").replace(/Digit/g, "");
|
|
const hasToasts = context.toastCount > 0;
|
|
React3.useEffect(() => {
|
|
const handleKeyDown = (event) => {
|
|
const isHotkeyPressed = hotkey.length !== 0 && hotkey.every((key) => event[key] || event.code === key);
|
|
if (isHotkeyPressed) ref.current?.focus();
|
|
};
|
|
document.addEventListener("keydown", handleKeyDown);
|
|
return () => document.removeEventListener("keydown", handleKeyDown);
|
|
}, [hotkey]);
|
|
React3.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 = React3.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]
|
|
);
|
|
React3.useEffect(() => {
|
|
const viewport = ref.current;
|
|
if (viewport) {
|
|
const handleKeyDown = (event) => {
|
|
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) {
|
|
headFocusProxyRef.current?.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 ? headFocusProxyRef.current?.focus() : tailFocusProxyRef.current?.focus();
|
|
}
|
|
}
|
|
};
|
|
viewport.addEventListener("keydown", handleKeyDown);
|
|
return () => viewport.removeEventListener("keydown", handleKeyDown);
|
|
}
|
|
}, [getItems, getSortedTabbableCandidates]);
|
|
return (0, import_jsx_runtime4.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_runtime4.jsx)(
|
|
FocusProxy,
|
|
{
|
|
ref: headFocusProxyRef,
|
|
onFocusFromOutsideViewport: () => {
|
|
const tabbableCandidates = getSortedTabbableCandidates({
|
|
tabbingDirection: "forwards"
|
|
});
|
|
focusFirst(tabbableCandidates);
|
|
}
|
|
}
|
|
),
|
|
(0, import_jsx_runtime4.jsx)(Collection.Slot, { scope: __scopeToast, children: (0, import_jsx_runtime4.jsx)(Primitive.ol, { tabIndex: -1, ...viewportProps, ref: composedRefs }) }),
|
|
hasToasts && (0, import_jsx_runtime4.jsx)(
|
|
FocusProxy,
|
|
{
|
|
ref: tailFocusProxyRef,
|
|
onFocusFromOutsideViewport: () => {
|
|
const tabbableCandidates = getSortedTabbableCandidates({
|
|
tabbingDirection: "backwards"
|
|
});
|
|
focusFirst(tabbableCandidates);
|
|
}
|
|
}
|
|
)
|
|
]
|
|
}
|
|
);
|
|
}
|
|
);
|
|
ToastViewport.displayName = VIEWPORT_NAME;
|
|
var FOCUS_PROXY_NAME = "ToastFocusProxy";
|
|
var FocusProxy = React3.forwardRef(
|
|
(props, forwardedRef) => {
|
|
const { __scopeToast, onFocusFromOutsideViewport, ...proxyProps } = props;
|
|
const context = useToastProviderContext(FOCUS_PROXY_NAME, __scopeToast);
|
|
return (0, import_jsx_runtime4.jsx)(
|
|
VisuallyHidden,
|
|
{
|
|
tabIndex: 0,
|
|
...proxyProps,
|
|
ref: forwardedRef,
|
|
style: { position: "fixed" },
|
|
onFocus: (event) => {
|
|
const prevFocusedElement = event.relatedTarget;
|
|
const isFocusFromOutsideViewport = !context.viewport?.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 = React3.forwardRef(
|
|
(props, forwardedRef) => {
|
|
const { forceMount, open: openProp, defaultOpen, onOpenChange, ...toastProps } = props;
|
|
const [open, setOpen] = useControllableState({
|
|
prop: openProp,
|
|
defaultProp: defaultOpen ?? true,
|
|
onChange: onOpenChange,
|
|
caller: TOAST_NAME
|
|
});
|
|
return (0, import_jsx_runtime4.jsx)(Presence, { present: forceMount || open, children: (0, import_jsx_runtime4.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 = React3.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] = React3.useState(null);
|
|
const composedRefs = useComposedRefs(forwardedRef, (node2) => setNode(node2));
|
|
const pointerStartRef = React3.useRef(null);
|
|
const swipeDeltaRef = React3.useRef(null);
|
|
const duration = durationProp || context.duration;
|
|
const closeTimerStartTimeRef = React3.useRef(0);
|
|
const closeTimerRemainingTimeRef = React3.useRef(duration);
|
|
const closeTimerRef = React3.useRef(0);
|
|
const { onToastAdd, onToastRemove } = context;
|
|
const handleClose = useCallbackRef(() => {
|
|
const isFocusInToast = node?.contains(document.activeElement);
|
|
if (isFocusInToast) context.viewport?.focus();
|
|
onClose();
|
|
});
|
|
const startTimer = React3.useCallback(
|
|
(duration2) => {
|
|
if (!duration2 || duration2 === Infinity) return;
|
|
window.clearTimeout(closeTimerRef.current);
|
|
closeTimerStartTimeRef.current = (/* @__PURE__ */ new Date()).getTime();
|
|
closeTimerRef.current = window.setTimeout(handleClose, duration2);
|
|
},
|
|
[handleClose]
|
|
);
|
|
React3.useEffect(() => {
|
|
const viewport = context.viewport;
|
|
if (viewport) {
|
|
const handleResume = () => {
|
|
startTimer(closeTimerRemainingTimeRef.current);
|
|
onResume?.();
|
|
};
|
|
const handlePause = () => {
|
|
const elapsedTime = (/* @__PURE__ */ new Date()).getTime() - closeTimerStartTimeRef.current;
|
|
closeTimerRemainingTimeRef.current = closeTimerRemainingTimeRef.current - elapsedTime;
|
|
window.clearTimeout(closeTimerRef.current);
|
|
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]);
|
|
React3.useEffect(() => {
|
|
if (open && !context.isClosePausedRef.current) startTimer(duration);
|
|
}, [open, duration, context.isClosePausedRef, startTimer]);
|
|
React3.useEffect(() => {
|
|
onToastAdd();
|
|
return () => onToastRemove();
|
|
}, [onToastAdd, onToastRemove]);
|
|
const announceTextContent = React3.useMemo(() => {
|
|
return node ? getAnnounceTextContent(node) : null;
|
|
}, [node]);
|
|
if (!context.viewport) return null;
|
|
return (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
|
|
announceTextContent && (0, import_jsx_runtime4.jsx)(
|
|
ToastAnnounce,
|
|
{
|
|
__scopeToast,
|
|
role: "status",
|
|
"aria-live": type === "foreground" ? "assertive" : "polite",
|
|
children: announceTextContent
|
|
}
|
|
),
|
|
(0, import_jsx_runtime4.jsx)(ToastInteractiveProvider, { scope: __scopeToast, onClose: handleClose, children: ReactDOM.createPortal(
|
|
(0, import_jsx_runtime4.jsx)(Collection.ItemSlot, { scope: __scopeToast, children: (0, import_jsx_runtime4.jsx)(
|
|
Root,
|
|
{
|
|
asChild: true,
|
|
onEscapeKeyDown: composeEventHandlers(onEscapeKeyDown, () => {
|
|
if (!context.isFocusedToastEscapeKeyDownRef.current) handleClose();
|
|
context.isFocusedToastEscapeKeyDownRef.current = false;
|
|
}),
|
|
children: (0, import_jsx_runtime4.jsx)(
|
|
Primitive.li,
|
|
{
|
|
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?.(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;
|
|
handleAndDispatchCustomEvent(TOAST_SWIPE_MOVE, onSwipeMove, eventDetail, {
|
|
discrete: false
|
|
});
|
|
} else if (isDeltaInDirection(delta, context.swipeDirection, moveStartBuffer)) {
|
|
swipeDeltaRef.current = delta;
|
|
handleAndDispatchCustomEvent(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)) {
|
|
handleAndDispatchCustomEvent(TOAST_SWIPE_END, onSwipeEnd, eventDetail, {
|
|
discrete: true
|
|
});
|
|
} else {
|
|
handleAndDispatchCustomEvent(
|
|
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] = React3.useState(false);
|
|
const [isAnnounced, setIsAnnounced] = React3.useState(false);
|
|
useNextFrame(() => setRenderAnnounceText(true));
|
|
React3.useEffect(() => {
|
|
const timer = window.setTimeout(() => setIsAnnounced(true), 1e3);
|
|
return () => window.clearTimeout(timer);
|
|
}, []);
|
|
return isAnnounced ? null : (0, import_jsx_runtime4.jsx)(Portal, { asChild: true, children: (0, import_jsx_runtime4.jsx)(VisuallyHidden, { ...announceProps, children: renderAnnounceText && (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
|
|
context.label,
|
|
" ",
|
|
children
|
|
] }) }) });
|
|
};
|
|
var TITLE_NAME = "ToastTitle";
|
|
var ToastTitle = React3.forwardRef(
|
|
(props, forwardedRef) => {
|
|
const { __scopeToast, ...titleProps } = props;
|
|
return (0, import_jsx_runtime4.jsx)(Primitive.div, { ...titleProps, ref: forwardedRef });
|
|
}
|
|
);
|
|
ToastTitle.displayName = TITLE_NAME;
|
|
var DESCRIPTION_NAME = "ToastDescription";
|
|
var ToastDescription = React3.forwardRef(
|
|
(props, forwardedRef) => {
|
|
const { __scopeToast, ...descriptionProps } = props;
|
|
return (0, import_jsx_runtime4.jsx)(Primitive.div, { ...descriptionProps, ref: forwardedRef });
|
|
}
|
|
);
|
|
ToastDescription.displayName = DESCRIPTION_NAME;
|
|
var ACTION_NAME = "ToastAction";
|
|
var ToastAction = React3.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_runtime4.jsx)(ToastAnnounceExclude, { altText, asChild: true, children: (0, import_jsx_runtime4.jsx)(ToastClose, { ...actionProps, ref: forwardedRef }) });
|
|
}
|
|
);
|
|
ToastAction.displayName = ACTION_NAME;
|
|
var CLOSE_NAME = "ToastClose";
|
|
var ToastClose = React3.forwardRef(
|
|
(props, forwardedRef) => {
|
|
const { __scopeToast, ...closeProps } = props;
|
|
const interactiveContext = useToastInteractiveContext(CLOSE_NAME, __scopeToast);
|
|
return (0, import_jsx_runtime4.jsx)(ToastAnnounceExclude, { asChild: true, children: (0, import_jsx_runtime4.jsx)(
|
|
Primitive.button,
|
|
{
|
|
type: "button",
|
|
...closeProps,
|
|
ref: forwardedRef,
|
|
onClick: composeEventHandlers(props.onClick, interactiveContext.onClose)
|
|
}
|
|
) });
|
|
}
|
|
);
|
|
ToastClose.displayName = CLOSE_NAME;
|
|
var ToastAnnounceExclude = React3.forwardRef((props, forwardedRef) => {
|
|
const { __scopeToast, altText, ...announceExcludeProps } = props;
|
|
return (0, import_jsx_runtime4.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 handleAndDispatchCustomEvent(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
|