Skip to content

Commit 82266a8

Browse files
author
Fabien MARIE-LOUISE
committed
feat(interactions): add createOverlay
1 parent 024cd65 commit 82266a8

File tree

5 files changed

+445
-0
lines changed

5 files changed

+445
-0
lines changed

packages/overlays/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@
4040
},
4141
"dependencies": {
4242
"@solid-aria/button": "workspace:^",
43+
"@solid-aria/interactions": "workspace:^",
4344
"@solid-aria/types": "workspace:^",
4445
"@solid-aria/utils": "workspace:^"
4546
},
+177
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,177 @@
1+
import { createFocusWithin, createInteractOutside } from "@solid-aria/interactions";
2+
import { DOMElements } from "@solid-aria/types";
3+
import { access, MaybeAccessor } from "@solid-primitives/utils";
4+
import { Accessor, createEffect, createMemo, JSX, onCleanup } from "solid-js";
5+
6+
interface CreateOverlayProps {
7+
/**
8+
* Whether the overlay is currently open.
9+
*/
10+
isOpen?: MaybeAccessor<boolean | undefined>;
11+
12+
/**
13+
* Whether to close the overlay when the user interacts outside it.
14+
* @default false
15+
*/
16+
isDismissable?: MaybeAccessor<boolean | undefined>;
17+
18+
/**
19+
* Whether the overlay should close when focus is lost or moves outside it.
20+
*/
21+
shouldCloseOnBlur?: MaybeAccessor<boolean | undefined>;
22+
23+
/**
24+
* Whether pressing the escape key to close the overlay should be disabled.
25+
* @default false
26+
*/
27+
isKeyboardDismissDisabled?: MaybeAccessor<boolean | undefined>;
28+
29+
/**
30+
* Handler that is called when the overlay should close.
31+
*/
32+
onClose?: () => void;
33+
34+
/**
35+
* When user interacts with the argument element outside of the overlay ref,
36+
* return true if onClose should be called. This gives you a chance to filter
37+
* out interaction with elements that should not dismiss the overlay.
38+
* By default, onClose will always be called on interaction outside the overlay ref.
39+
*/
40+
shouldCloseOnInteractOutside?: (element: HTMLElement) => boolean;
41+
}
42+
43+
interface OverlayAria<
44+
OverlayElementType extends DOMElements,
45+
UnderlayElementType extends DOMElements
46+
> {
47+
/**
48+
* Props to apply to the overlay container element.
49+
*/
50+
overlayProps: Accessor<JSX.IntrinsicElements[OverlayElementType]>;
51+
52+
/**
53+
* Props to apply to the underlay element, if any.
54+
*/
55+
underlayProps: Accessor<JSX.IntrinsicElements[UnderlayElementType]>;
56+
}
57+
58+
const visibleOverlays: Array<Accessor<HTMLElement | undefined>> = [];
59+
60+
/**
61+
* Provides the behavior for overlays such as dialogs, popovers, and menus.
62+
* Hides the overlay when the user interacts outside it, when the Escape key is pressed,
63+
* or optionally, on blur. Only the top-most overlay will close at once.
64+
*/
65+
export function createOverlay<
66+
OverlayElementType extends DOMElements = "div",
67+
UnderlayElementType extends DOMElements = "div",
68+
RefElement extends HTMLElement = HTMLDivElement
69+
>(
70+
props: CreateOverlayProps,
71+
ref: Accessor<RefElement | undefined>
72+
): OverlayAria<OverlayElementType, UnderlayElementType> {
73+
// Add the overlay ref to the stack of visible overlays on mount, and remove on unmount.
74+
createEffect(() => {
75+
if (access(props.isOpen)) {
76+
visibleOverlays.push(ref);
77+
}
78+
79+
onCleanup(() => {
80+
const index = visibleOverlays.indexOf(ref);
81+
82+
if (index >= 0) {
83+
visibleOverlays.splice(index, 1);
84+
}
85+
});
86+
});
87+
88+
// Only hide the overlay when it is the topmost visible overlay in the stack.
89+
const onHide = () => {
90+
if (visibleOverlays[visibleOverlays.length - 1]() === ref()) {
91+
props.onClose?.();
92+
}
93+
};
94+
95+
const onInteractOutsideStart = (e: Event) => {
96+
if (
97+
!props.shouldCloseOnInteractOutside ||
98+
props.shouldCloseOnInteractOutside(e.target as HTMLElement)
99+
) {
100+
if (visibleOverlays[visibleOverlays.length - 1]() === ref()) {
101+
e.stopPropagation();
102+
e.preventDefault();
103+
}
104+
}
105+
};
106+
107+
const onInteractOutside = (e: Event) => {
108+
if (!access(props.isDismissable)) {
109+
return;
110+
}
111+
112+
if (
113+
!props.shouldCloseOnInteractOutside ||
114+
props.shouldCloseOnInteractOutside(e.target as HTMLElement)
115+
) {
116+
if (visibleOverlays[visibleOverlays.length - 1]() === ref()) {
117+
e.stopPropagation();
118+
e.preventDefault();
119+
}
120+
onHide();
121+
}
122+
};
123+
124+
// Handle the escape key
125+
const onKeyDown = (e: KeyboardEvent) => {
126+
if (e.key === "Escape" && !access(props.isKeyboardDismissDisabled)) {
127+
e.stopPropagation();
128+
e.preventDefault();
129+
onHide();
130+
}
131+
};
132+
133+
// Handle clicking outside the overlay to close it
134+
createInteractOutside(
135+
{
136+
onInteractOutside,
137+
onInteractOutsideStart
138+
},
139+
ref
140+
);
141+
142+
const { focusWithinProps } = createFocusWithin({
143+
isDisabled: () => !access(props.shouldCloseOnBlur),
144+
onFocusOut: e => {
145+
if (
146+
!props.shouldCloseOnInteractOutside ||
147+
props.shouldCloseOnInteractOutside(e.relatedTarget as HTMLElement)
148+
) {
149+
props.onClose?.();
150+
}
151+
}
152+
});
153+
154+
const onPointerDownUnderlay = (e: Event) => {
155+
// fixes a firefox issue that starts text selection https://bugzilla.mozilla.org/show_bug.cgi?id=1675846
156+
if (e.target === e.currentTarget) {
157+
e.preventDefault();
158+
}
159+
};
160+
161+
const overlayProps = createMemo(
162+
() =>
163+
({
164+
onKeyDown,
165+
...focusWithinProps()
166+
} as JSX.IntrinsicElements[OverlayElementType])
167+
);
168+
169+
const underlayProps = createMemo(
170+
() =>
171+
({
172+
onPointerDown: onPointerDownUnderlay
173+
} as JSX.IntrinsicElements[UnderlayElementType])
174+
);
175+
176+
return { overlayProps, underlayProps };
177+
}

packages/overlays/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
export * from "./createModal";
2+
export * from "./createOverlay";
23
export * from "./createOverlayTrigger";
34
export * from "./createOverlayTriggerState";
45
export * from "./createPreventScroll";

packages/overlays/src/test-utils.ts

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
/**
2+
* Enables reading pageX/pageY from fireEvent.mouse*(..., {pageX: ..., pageY: ...}).
3+
*/
4+
export function installMouseEvent() {
5+
beforeAll(() => {
6+
const oldMouseEvent = MouseEvent;
7+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
8+
// @ts-ignore
9+
global.MouseEvent = class FakeMouseEvent extends MouseEvent {
10+
_init: { pageX: number; pageY: number };
11+
constructor(name: any, init: any) {
12+
super(name, init);
13+
this._init = init;
14+
}
15+
get pageX() {
16+
return this._init.pageX;
17+
}
18+
get pageY() {
19+
return this._init.pageY;
20+
}
21+
};
22+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
23+
// @ts-ignore
24+
global.MouseEvent.oldMouseEvent = oldMouseEvent;
25+
});
26+
afterAll(() => {
27+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
28+
// @ts-ignore
29+
global.MouseEvent = global.MouseEvent.oldMouseEvent;
30+
});
31+
}
32+
33+
export function installPointerEvent() {
34+
beforeAll(() => {
35+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
36+
// @ts-ignore
37+
global.PointerEvent = class FakePointerEvent extends MouseEvent {
38+
_init: {
39+
pageX: number;
40+
pageY: number;
41+
pointerType: string;
42+
pointerId: number;
43+
width: number;
44+
height: number;
45+
};
46+
constructor(name: any, init: any) {
47+
super(name, init);
48+
this._init = init;
49+
}
50+
get pointerType() {
51+
return this._init.pointerType;
52+
}
53+
get pointerId() {
54+
return this._init.pointerId;
55+
}
56+
get pageX() {
57+
return this._init.pageX;
58+
}
59+
get pageY() {
60+
return this._init.pageY;
61+
}
62+
get width() {
63+
return this._init.width;
64+
}
65+
get height() {
66+
return this._init.height;
67+
}
68+
};
69+
});
70+
71+
afterAll(() => {
72+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
73+
// @ts-ignore
74+
delete global.PointerEvent;
75+
});
76+
}

0 commit comments

Comments
 (0)