Spaces:
Runtime error
Runtime error
| import type { Action } from "svelte/action"; | |
| export const clickOutside: Action<HTMLElement, () => void> = (node, callback) => { | |
| let _callback = callback; | |
| function update(newCallback: () => void) { | |
| _callback = newCallback; | |
| } | |
| function handleClick(event: MouseEvent) { | |
| if (window.getSelection()?.toString()) { | |
| // Don't close if text is selected | |
| return; | |
| } | |
| // For dialog elements, check if click was on the backdrop | |
| if (node instanceof HTMLDialogElement) { | |
| const rect = node.getBoundingClientRect(); | |
| const isInDialog = | |
| event.clientX >= rect.left && | |
| event.clientX <= rect.right && | |
| event.clientY >= rect.top && | |
| event.clientY <= rect.bottom; | |
| if (!isInDialog) { | |
| _callback(); | |
| return; | |
| } | |
| } | |
| // For non-dialog elements, use the standard contains check | |
| if (!node.contains(event.target as Node) && !event.defaultPrevented) { | |
| _callback(); | |
| } | |
| } | |
| // For dialogs, listen on the element itself | |
| if (node instanceof HTMLDialogElement) { | |
| node.addEventListener("click", handleClick); | |
| } else { | |
| // For other elements, listen on the document | |
| document.addEventListener("click", handleClick, true); | |
| } | |
| return { | |
| update, | |
| destroy() { | |
| if (node instanceof HTMLDialogElement) { | |
| node.removeEventListener("click", handleClick); | |
| } else { | |
| document.removeEventListener("click", handleClick, true); | |
| } | |
| }, | |
| }; | |
| }; | |