-
Notifications
You must be signed in to change notification settings - Fork 277
/
Copy pathDismissableLayer.tsx
99 lines (86 loc) · 2.63 KB
/
DismissableLayer.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
import type React from 'react';
import { useEffect, useRef } from 'react';
type DismissableLayerProps = {
children?: React.ReactNode;
disableEscapeKey?: boolean;
disableOutsideClick?: boolean;
onDismiss?: () => void;
/**
* Reference to the trigger element that should not dismiss the layer when clicked.
*/
triggerRef?: React.RefObject<HTMLElement>;
/**
* When `true`, prevents trigger click events from bubbling up.
* Useful for bottom sheets to prevent unwanted side effects.
*/
preventTriggerBubbling?: boolean;
};
// DismissableLayer handles dismissal using outside clicks and escape key events
export function DismissableLayer({
children,
disableEscapeKey = false,
disableOutsideClick = false,
onDismiss,
triggerRef,
preventTriggerBubbling = false,
}: DismissableLayerProps) {
const layerRef = useRef<HTMLDivElement>(null);
const handleTriggerClick = (event: PointerEvent) => {
// If preventTriggerBubbling is enabled, stop the click from propagating
// This prevents unwanted side effects when the layer is used in nested contexts
if (preventTriggerBubbling) {
event.preventDefault();
event.stopPropagation();
}
};
const isClickInsideLayer = (target: Node) => {
// Check if click occurred inside the dismissable layer itself
// If so, do nothing and let the click event proceed normally
return layerRef.current?.contains(target);
};
useEffect(() => {
if (disableOutsideClick && disableEscapeKey) {
return;
}
const handlePointerDownCapture = (event: PointerEvent) => {
if (disableOutsideClick) {
return;
}
const target = event.target as Node;
// Check if click occurred on the trigger element (e.g., a button that opened this layer)
if (triggerRef?.current?.contains(target)) {
handleTriggerClick(event);
return;
}
if (!isClickInsideLayer(target)) {
onDismiss?.();
}
};
const handleKeyDown = (event: KeyboardEvent) => {
if (!disableEscapeKey && event.key === 'Escape') {
onDismiss?.();
}
};
document.addEventListener('pointerdown', handlePointerDownCapture, true);
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener(
'pointerdown',
handlePointerDownCapture,
true,
);
document.removeEventListener('keydown', handleKeyDown);
};
}, [
disableOutsideClick,
disableEscapeKey,
onDismiss,
triggerRef,
preventTriggerBubbling,
]);
return (
<div ref={layerRef} data-testid="ockDismissableLayer">
{children}
</div>
);
}