@ultraviolet/ui 1.61.1 → 1.62.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/Modal/ModalContent.cjs +53 -0
- package/dist/components/Modal/ModalContent.d.ts +17 -0
- package/dist/components/Modal/ModalContent.js +51 -0
- package/dist/components/Modal/ModalProvider.cjs +40 -0
- package/dist/components/Modal/ModalProvider.d.ts +14 -0
- package/dist/components/Modal/ModalProvider.js +40 -0
- package/dist/components/Modal/components/Dialog.cjs +162 -0
- package/dist/components/Modal/components/Dialog.d.ts +13 -0
- package/dist/components/Modal/components/Dialog.js +160 -0
- package/dist/components/Modal/{Disclosure.d.ts → components/Disclosure.d.ts} +1 -1
- package/dist/components/Modal/index.cjs +5 -29
- package/dist/components/Modal/index.js +6 -28
- package/dist/components/Slider/components/DoubleSlider.cjs +30 -23
- package/dist/components/Slider/components/DoubleSlider.js +30 -23
- package/dist/components/Slider/components/Options.cjs +28 -5
- package/dist/components/Slider/components/Options.d.ts +11 -0
- package/dist/components/Slider/components/Options.js +25 -4
- package/dist/components/Slider/components/SingleSlider.cjs +14 -11
- package/dist/components/Slider/components/SingleSlider.js +14 -11
- package/dist/components/Slider/styles.cjs +11 -29
- package/dist/components/Slider/styles.d.ts +1 -12
- package/dist/components/Slider/styles.js +11 -29
- package/dist/components/TextArea/index.cjs +5 -5
- package/dist/components/TextArea/index.js +5 -5
- package/dist/components/UnitInput/index.cjs +5 -5
- package/dist/components/UnitInput/index.js +5 -5
- package/package.json +4 -4
- package/dist/components/Modal/Dialog.cjs +0 -144
- package/dist/components/Modal/Dialog.d.ts +0 -11
- package/dist/components/Modal/Dialog.js +0 -142
- /package/dist/components/Modal/{Disclosure.cjs → components/Disclosure.cjs} +0 -0
- /package/dist/components/Modal/{Disclosure.js → components/Disclosure.js} +0 -0
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const index = require("../Button/index.cjs");
|
|
6
|
+
const Dialog = require("./components/Dialog.cjs");
|
|
7
|
+
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
8
|
+
const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
|
|
9
|
+
const StyledContainer = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
10
|
+
target: "e14rmpyf0"
|
|
11
|
+
} : {
|
|
12
|
+
target: "e14rmpyf0",
|
|
13
|
+
label: "StyledContainer"
|
|
14
|
+
})("position:absolute;top:", ({
|
|
15
|
+
theme
|
|
16
|
+
}) => theme.space["2"], ";right:", ({
|
|
17
|
+
theme
|
|
18
|
+
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBT2tDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB0eXBlIHsgQ29tcG9uZW50UHJvcHMgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuaW1wb3J0IHR5cGUgeyBNb2RhbCB9IGZyb20gJy4nXG5cbmNvbnN0IFN0eWxlZENvbnRhaW5lciA9IHN0eWxlZC5kaXZgXG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgdG9wOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycyJ119O1xuICByaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbmBcblxudHlwZSBNb2RhbENvbnRlbnRQcm9wcyA9IENvbXBvbmVudFByb3BzPHR5cGVvZiBNb2RhbD4gJiB7XG4gIHZpc2libGU6IGJvb2xlYW5cbiAgb3BlbjogYm9vbGVhblxuICBvcGVuZWQ6IGJvb2xlYW5cbiAgcGxhY2VtZW50OiBNb2RhbFBsYWNlbWVudFxuICBmaW5hbFNpemU6IE1vZGFsU2l6ZVxuICBmaW5hbElkOiBzdHJpbmdcbiAgaGFuZGxlT3BlbjogKCkgPT4gdm9pZFxuICBoYW5kbGVUb2dnbGU6ICgpID0+IHZvaWRcbiAgaGFuZGxlQ2xvc2U6ICgpID0+IHZvaWRcbiAgZGF0YVRlc3RJZD86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTW9kYWxDb250ZW50ID0gKHtcbiAgdmlzaWJsZSxcbiAgb3BlbixcbiAgb3BlbmVkLFxuICBwbGFjZW1lbnQsXG4gIGZpbmFsU2l6ZSxcbiAgYXJpYUxhYmVsLFxuICBoaWRlT25DbGlja091dHNpZGUsXG4gIGhpZGVPbkVzYyxcbiAgcHJldmVudEJvZHlTY3JvbGwsXG4gIGhhbmRsZUNsb3NlLFxuICBjbGFzc05hbWUsXG4gIGJhY2tkcm9wQ2xhc3NOYW1lLFxuICBkYXRhVGVzdElkLFxuICBjdXN0b21EaWFsb2dTdHlsZXMsXG4gIGN1c3RvbURpYWxvZ0JhY2tkcm9wU3R5bGVzLFxuICBpc0Nsb3NhYmxlLFxuICBjaGlsZHJlbixcbiAgaGFuZGxlT3BlbixcbiAgaGFuZGxlVG9nZ2xlLFxuICBmaW5hbElkLFxufTogTW9kYWxDb250ZW50UHJvcHMpID0+XG4gIHZpc2libGUgfHwgb3BlbiB8fCBvcGVuZWQgPyAoXG4gICAgPERpYWxvZ1xuICAgICAgb3Blbj17dmlzaWJsZSB8fCBvcGVuIHx8IG9wZW5lZH1cbiAgICAgIHBsYWNlbWVudD17cGxhY2VtZW50fVxuICAgICAgc2l6ZT17ZmluYWxTaXplfVxuICAgICAgYXJpYUxhYmVsPXthcmlhTGFiZWx9XG4gICAgICBoaWRlT25DbGlja091dHNpZGU9e2hpZGVPbkNsaWNrT3V0c2lkZX1cbiAgICAgIGhpZGVPbkVzYz17aGlkZU9uRXNjfVxuICAgICAgcHJldmVudEJvZHlTY3JvbGw9e3ByZXZlbnRCb2R5U2Nyb2xsfVxuICAgICAgb25DbG9zZT17aGFuZGxlQ2xvc2V9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGJhY2tkcm9wQ2xhc3NOYW1lPXtiYWNrZHJvcENsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgaWQ9e2ZpbmFsSWR9XG4gICAgICBkaWFsb2dDc3M9e2N1c3RvbURpYWxvZ1N0eWxlc31cbiAgICAgIGJhY2tkcm9wQ3NzPXtjdXN0b21EaWFsb2dCYWNrZHJvcFN0eWxlc31cbiAgICA+XG4gICAgICA8PlxuICAgICAgICB7dHlwZW9mIGNoaWxkcmVuID09PSAnZnVuY3Rpb24nXG4gICAgICAgICAgPyBjaGlsZHJlbih7XG4gICAgICAgICAgICAgIHZpc2libGUsXG4gICAgICAgICAgICAgIG9uQ2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBvbk9wZW46IGhhbmRsZU9wZW4sXG4gICAgICAgICAgICAgIHRvZ2dsZTogaGFuZGxlVG9nZ2xlLFxuICAgICAgICAgICAgICBtb2RhbElkOiBmaW5hbElkLFxuICAgICAgICAgICAgICBoaWRlOiBoYW5kbGVDbG9zZSxcbiAgICAgICAgICAgICAgY2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBzaG93OiBoYW5kbGVPcGVuLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICA6IGNoaWxkcmVufVxuICAgICAgICA8U3R5bGVkQ29udGFpbmVyPlxuICAgICAgICAgIHtpc0Nsb3NhYmxlID8gKFxuICAgICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgICBkYXRhLXRlc3RpZD17XG4gICAgICAgICAgICAgICAgZGF0YVRlc3RJZCA/IGAke2RhdGFUZXN0SWR9LWNsb3NlLWJ1dHRvbmAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBvbkNsaWNrPXtoYW5kbGVDbG9zZX1cbiAgICAgICAgICAgICAgdmFyaWFudD1cImdob3N0XCJcbiAgICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICAgICAgaWNvbj1cImNsb3NlXCJcbiAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgICAgIDwvPlxuICAgIDwvRGlhbG9nPlxuICApIDogbnVsbFxuIl19 */"));
|
|
19
|
+
const ModalContent = ({
|
|
20
|
+
visible,
|
|
21
|
+
open,
|
|
22
|
+
opened,
|
|
23
|
+
placement,
|
|
24
|
+
finalSize,
|
|
25
|
+
ariaLabel,
|
|
26
|
+
hideOnClickOutside,
|
|
27
|
+
hideOnEsc,
|
|
28
|
+
preventBodyScroll,
|
|
29
|
+
handleClose,
|
|
30
|
+
className,
|
|
31
|
+
backdropClassName,
|
|
32
|
+
dataTestId,
|
|
33
|
+
customDialogStyles,
|
|
34
|
+
customDialogBackdropStyles,
|
|
35
|
+
isClosable,
|
|
36
|
+
children,
|
|
37
|
+
handleOpen,
|
|
38
|
+
handleToggle,
|
|
39
|
+
finalId
|
|
40
|
+
}) => visible || open || opened ? /* @__PURE__ */ jsxRuntime.jsx(Dialog.Dialog, { open: visible || open || opened, placement, size: finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, onClose: handleClose, className, backdropClassName, "data-testid": dataTestId, id: finalId, dialogCss: customDialogStyles, backdropCss: customDialogBackdropStyles, children: /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
|
|
41
|
+
typeof children === "function" ? children({
|
|
42
|
+
visible,
|
|
43
|
+
onClose: handleClose,
|
|
44
|
+
onOpen: handleOpen,
|
|
45
|
+
toggle: handleToggle,
|
|
46
|
+
modalId: finalId,
|
|
47
|
+
hide: handleClose,
|
|
48
|
+
close: handleClose,
|
|
49
|
+
show: handleOpen
|
|
50
|
+
}) : children,
|
|
51
|
+
/* @__PURE__ */ jsxRuntime.jsx(StyledContainer, { children: isClosable ? /* @__PURE__ */ jsxRuntime.jsx(index.Button, { "data-testid": dataTestId ? `${dataTestId}-close-button` : void 0, onClick: handleClose, variant: "ghost", size: "small", icon: "close", sentiment: "neutral", "aria-label": "close" }) : null })
|
|
52
|
+
] }) }) : null;
|
|
53
|
+
exports.ModalContent = ModalContent;
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import type { ComponentProps } from 'react';
|
|
2
|
+
import type { ModalPlacement, ModalSize } from './types';
|
|
3
|
+
import type { Modal } from '.';
|
|
4
|
+
type ModalContentProps = ComponentProps<typeof Modal> & {
|
|
5
|
+
visible: boolean;
|
|
6
|
+
open: boolean;
|
|
7
|
+
opened: boolean;
|
|
8
|
+
placement: ModalPlacement;
|
|
9
|
+
finalSize: ModalSize;
|
|
10
|
+
finalId: string;
|
|
11
|
+
handleOpen: () => void;
|
|
12
|
+
handleToggle: () => void;
|
|
13
|
+
handleClose: () => void;
|
|
14
|
+
dataTestId?: string;
|
|
15
|
+
};
|
|
16
|
+
export declare const ModalContent: ({ visible, open, opened, placement, finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, handleClose, className, backdropClassName, dataTestId, customDialogStyles, customDialogBackdropStyles, isClosable, children, handleOpen, handleToggle, finalId, }: ModalContentProps) => import("@emotion/react/jsx-runtime").JSX.Element | null;
|
|
17
|
+
export {};
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { jsx, jsxs, Fragment } from "@emotion/react/jsx-runtime";
|
|
2
|
+
import _styled from "@emotion/styled/base";
|
|
3
|
+
import { Button } from "../Button/index.js";
|
|
4
|
+
import { Dialog } from "./components/Dialog.js";
|
|
5
|
+
const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
6
|
+
target: "e14rmpyf0"
|
|
7
|
+
} : {
|
|
8
|
+
target: "e14rmpyf0",
|
|
9
|
+
label: "StyledContainer"
|
|
10
|
+
})("position:absolute;top:", ({
|
|
11
|
+
theme
|
|
12
|
+
}) => theme.space["2"], ";right:", ({
|
|
13
|
+
theme
|
|
14
|
+
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBT2tDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCdcbmltcG9ydCB0eXBlIHsgQ29tcG9uZW50UHJvcHMgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuaW1wb3J0IHR5cGUgeyBNb2RhbCB9IGZyb20gJy4nXG5cbmNvbnN0IFN0eWxlZENvbnRhaW5lciA9IHN0eWxlZC5kaXZgXG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgdG9wOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycyJ119O1xuICByaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbmBcblxudHlwZSBNb2RhbENvbnRlbnRQcm9wcyA9IENvbXBvbmVudFByb3BzPHR5cGVvZiBNb2RhbD4gJiB7XG4gIHZpc2libGU6IGJvb2xlYW5cbiAgb3BlbjogYm9vbGVhblxuICBvcGVuZWQ6IGJvb2xlYW5cbiAgcGxhY2VtZW50OiBNb2RhbFBsYWNlbWVudFxuICBmaW5hbFNpemU6IE1vZGFsU2l6ZVxuICBmaW5hbElkOiBzdHJpbmdcbiAgaGFuZGxlT3BlbjogKCkgPT4gdm9pZFxuICBoYW5kbGVUb2dnbGU6ICgpID0+IHZvaWRcbiAgaGFuZGxlQ2xvc2U6ICgpID0+IHZvaWRcbiAgZGF0YVRlc3RJZD86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTW9kYWxDb250ZW50ID0gKHtcbiAgdmlzaWJsZSxcbiAgb3BlbixcbiAgb3BlbmVkLFxuICBwbGFjZW1lbnQsXG4gIGZpbmFsU2l6ZSxcbiAgYXJpYUxhYmVsLFxuICBoaWRlT25DbGlja091dHNpZGUsXG4gIGhpZGVPbkVzYyxcbiAgcHJldmVudEJvZHlTY3JvbGwsXG4gIGhhbmRsZUNsb3NlLFxuICBjbGFzc05hbWUsXG4gIGJhY2tkcm9wQ2xhc3NOYW1lLFxuICBkYXRhVGVzdElkLFxuICBjdXN0b21EaWFsb2dTdHlsZXMsXG4gIGN1c3RvbURpYWxvZ0JhY2tkcm9wU3R5bGVzLFxuICBpc0Nsb3NhYmxlLFxuICBjaGlsZHJlbixcbiAgaGFuZGxlT3BlbixcbiAgaGFuZGxlVG9nZ2xlLFxuICBmaW5hbElkLFxufTogTW9kYWxDb250ZW50UHJvcHMpID0+XG4gIHZpc2libGUgfHwgb3BlbiB8fCBvcGVuZWQgPyAoXG4gICAgPERpYWxvZ1xuICAgICAgb3Blbj17dmlzaWJsZSB8fCBvcGVuIHx8IG9wZW5lZH1cbiAgICAgIHBsYWNlbWVudD17cGxhY2VtZW50fVxuICAgICAgc2l6ZT17ZmluYWxTaXplfVxuICAgICAgYXJpYUxhYmVsPXthcmlhTGFiZWx9XG4gICAgICBoaWRlT25DbGlja091dHNpZGU9e2hpZGVPbkNsaWNrT3V0c2lkZX1cbiAgICAgIGhpZGVPbkVzYz17aGlkZU9uRXNjfVxuICAgICAgcHJldmVudEJvZHlTY3JvbGw9e3ByZXZlbnRCb2R5U2Nyb2xsfVxuICAgICAgb25DbG9zZT17aGFuZGxlQ2xvc2V9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGJhY2tkcm9wQ2xhc3NOYW1lPXtiYWNrZHJvcENsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgaWQ9e2ZpbmFsSWR9XG4gICAgICBkaWFsb2dDc3M9e2N1c3RvbURpYWxvZ1N0eWxlc31cbiAgICAgIGJhY2tkcm9wQ3NzPXtjdXN0b21EaWFsb2dCYWNrZHJvcFN0eWxlc31cbiAgICA+XG4gICAgICA8PlxuICAgICAgICB7dHlwZW9mIGNoaWxkcmVuID09PSAnZnVuY3Rpb24nXG4gICAgICAgICAgPyBjaGlsZHJlbih7XG4gICAgICAgICAgICAgIHZpc2libGUsXG4gICAgICAgICAgICAgIG9uQ2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBvbk9wZW46IGhhbmRsZU9wZW4sXG4gICAgICAgICAgICAgIHRvZ2dsZTogaGFuZGxlVG9nZ2xlLFxuICAgICAgICAgICAgICBtb2RhbElkOiBmaW5hbElkLFxuICAgICAgICAgICAgICBoaWRlOiBoYW5kbGVDbG9zZSxcbiAgICAgICAgICAgICAgY2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBzaG93OiBoYW5kbGVPcGVuLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICA6IGNoaWxkcmVufVxuICAgICAgICA8U3R5bGVkQ29udGFpbmVyPlxuICAgICAgICAgIHtpc0Nsb3NhYmxlID8gKFxuICAgICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgICBkYXRhLXRlc3RpZD17XG4gICAgICAgICAgICAgICAgZGF0YVRlc3RJZCA/IGAke2RhdGFUZXN0SWR9LWNsb3NlLWJ1dHRvbmAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBvbkNsaWNrPXtoYW5kbGVDbG9zZX1cbiAgICAgICAgICAgICAgdmFyaWFudD1cImdob3N0XCJcbiAgICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICAgICAgaWNvbj1cImNsb3NlXCJcbiAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgICAgIDwvPlxuICAgIDwvRGlhbG9nPlxuICApIDogbnVsbFxuIl19 */"));
|
|
15
|
+
const ModalContent = ({
|
|
16
|
+
visible,
|
|
17
|
+
open,
|
|
18
|
+
opened,
|
|
19
|
+
placement,
|
|
20
|
+
finalSize,
|
|
21
|
+
ariaLabel,
|
|
22
|
+
hideOnClickOutside,
|
|
23
|
+
hideOnEsc,
|
|
24
|
+
preventBodyScroll,
|
|
25
|
+
handleClose,
|
|
26
|
+
className,
|
|
27
|
+
backdropClassName,
|
|
28
|
+
dataTestId,
|
|
29
|
+
customDialogStyles,
|
|
30
|
+
customDialogBackdropStyles,
|
|
31
|
+
isClosable,
|
|
32
|
+
children,
|
|
33
|
+
handleOpen,
|
|
34
|
+
handleToggle,
|
|
35
|
+
finalId
|
|
36
|
+
}) => visible || open || opened ? /* @__PURE__ */ jsx(Dialog, { open: visible || open || opened, placement, size: finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, onClose: handleClose, className, backdropClassName, "data-testid": dataTestId, id: finalId, dialogCss: customDialogStyles, backdropCss: customDialogBackdropStyles, children: /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
37
|
+
typeof children === "function" ? children({
|
|
38
|
+
visible,
|
|
39
|
+
onClose: handleClose,
|
|
40
|
+
onOpen: handleOpen,
|
|
41
|
+
toggle: handleToggle,
|
|
42
|
+
modalId: finalId,
|
|
43
|
+
hide: handleClose,
|
|
44
|
+
close: handleClose,
|
|
45
|
+
show: handleOpen
|
|
46
|
+
}) : children,
|
|
47
|
+
/* @__PURE__ */ jsx(StyledContainer, { children: isClosable ? /* @__PURE__ */ jsx(Button, { "data-testid": dataTestId ? `${dataTestId}-close-button` : void 0, onClick: handleClose, variant: "ghost", size: "small", icon: "close", sentiment: "neutral", "aria-label": "close" }) : null })
|
|
48
|
+
] }) }) : null;
|
|
49
|
+
export {
|
|
50
|
+
ModalContent
|
|
51
|
+
};
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const React = require("react");
|
|
5
|
+
const ModalContext = React.createContext(void 0);
|
|
6
|
+
const useModal = () => {
|
|
7
|
+
const context = React.useContext(ModalContext);
|
|
8
|
+
if (!context) {
|
|
9
|
+
throw new Error("useModal must be used within a ModalProvider");
|
|
10
|
+
}
|
|
11
|
+
return context;
|
|
12
|
+
};
|
|
13
|
+
const ModalProvider = ({
|
|
14
|
+
children
|
|
15
|
+
}) => {
|
|
16
|
+
const [openedModals, setOpenedModals] = React.useState([]);
|
|
17
|
+
const [previsousOpenedModales, setPreviousOpenedModales] = React.useState([]);
|
|
18
|
+
const registerModal = React.useCallback((id) => {
|
|
19
|
+
setOpenedModals((prev) => {
|
|
20
|
+
setPreviousOpenedModales(prev);
|
|
21
|
+
return [...prev, id];
|
|
22
|
+
});
|
|
23
|
+
}, []);
|
|
24
|
+
const unregisterModal = React.useCallback((id) => {
|
|
25
|
+
setOpenedModals((prev) => {
|
|
26
|
+
setPreviousOpenedModales(prev);
|
|
27
|
+
return prev.filter((modalId) => modalId !== id);
|
|
28
|
+
});
|
|
29
|
+
}, []);
|
|
30
|
+
const value = React.useMemo(() => ({
|
|
31
|
+
openedModals,
|
|
32
|
+
registerModal,
|
|
33
|
+
unregisterModal,
|
|
34
|
+
previsousOpenedModales
|
|
35
|
+
}), [openedModals, registerModal, unregisterModal, previsousOpenedModales]);
|
|
36
|
+
return /* @__PURE__ */ jsxRuntime.jsx(ModalContext.Provider, { value, children });
|
|
37
|
+
};
|
|
38
|
+
exports.ModalContext = ModalContext;
|
|
39
|
+
exports.ModalProvider = ModalProvider;
|
|
40
|
+
exports.useModal = useModal;
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { ReactNode } from 'react';
|
|
2
|
+
type ModalContextValues = {
|
|
3
|
+
openedModals: string[];
|
|
4
|
+
registerModal: (id: string) => void;
|
|
5
|
+
unregisterModal: (id: string) => void;
|
|
6
|
+
previsousOpenedModales: string[];
|
|
7
|
+
};
|
|
8
|
+
export declare const ModalContext: import("react").Context<ModalContextValues | undefined>;
|
|
9
|
+
export declare const useModal: () => ModalContextValues;
|
|
10
|
+
type ModalProviderProps = {
|
|
11
|
+
children: ReactNode;
|
|
12
|
+
};
|
|
13
|
+
export declare const ModalProvider: ({ children }: ModalProviderProps) => import("@emotion/react/jsx-runtime").JSX.Element;
|
|
14
|
+
export {};
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { jsx } from "@emotion/react/jsx-runtime";
|
|
2
|
+
import { createContext, useState, useCallback, useMemo, useContext } from "react";
|
|
3
|
+
const ModalContext = createContext(void 0);
|
|
4
|
+
const useModal = () => {
|
|
5
|
+
const context = useContext(ModalContext);
|
|
6
|
+
if (!context) {
|
|
7
|
+
throw new Error("useModal must be used within a ModalProvider");
|
|
8
|
+
}
|
|
9
|
+
return context;
|
|
10
|
+
};
|
|
11
|
+
const ModalProvider = ({
|
|
12
|
+
children
|
|
13
|
+
}) => {
|
|
14
|
+
const [openedModals, setOpenedModals] = useState([]);
|
|
15
|
+
const [previsousOpenedModales, setPreviousOpenedModales] = useState([]);
|
|
16
|
+
const registerModal = useCallback((id) => {
|
|
17
|
+
setOpenedModals((prev) => {
|
|
18
|
+
setPreviousOpenedModales(prev);
|
|
19
|
+
return [...prev, id];
|
|
20
|
+
});
|
|
21
|
+
}, []);
|
|
22
|
+
const unregisterModal = useCallback((id) => {
|
|
23
|
+
setOpenedModals((prev) => {
|
|
24
|
+
setPreviousOpenedModales(prev);
|
|
25
|
+
return prev.filter((modalId) => modalId !== id);
|
|
26
|
+
});
|
|
27
|
+
}, []);
|
|
28
|
+
const value = useMemo(() => ({
|
|
29
|
+
openedModals,
|
|
30
|
+
registerModal,
|
|
31
|
+
unregisterModal,
|
|
32
|
+
previsousOpenedModales
|
|
33
|
+
}), [openedModals, registerModal, unregisterModal, previsousOpenedModales]);
|
|
34
|
+
return /* @__PURE__ */ jsx(ModalContext.Provider, { value, children });
|
|
35
|
+
};
|
|
36
|
+
export {
|
|
37
|
+
ModalContext,
|
|
38
|
+
ModalProvider,
|
|
39
|
+
useModal
|
|
40
|
+
};
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
|
+
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const React = require("react");
|
|
6
|
+
const reactDom = require("react-dom");
|
|
7
|
+
const animations = require("../../../utils/animations.cjs");
|
|
8
|
+
const ModalProvider = require("../ModalProvider.cjs");
|
|
9
|
+
const constants = require("../constants.cjs");
|
|
10
|
+
const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
|
|
11
|
+
const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
|
|
12
|
+
const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
13
|
+
target: "e1fims041"
|
|
14
|
+
} : {
|
|
15
|
+
target: "e1fims041",
|
|
16
|
+
label: "StyledBackdrop"
|
|
17
|
+
})("position:fixed;top:0;right:0;height:0;width:0;overflow:hidden;background-color:", ({
|
|
18
|
+
theme
|
|
19
|
+
}) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
|
|
20
|
+
theme
|
|
21
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size }) =>\n    position > 0\n      ? `\n    width: ${MODAL_WIDTH[size] - position * 50}px !important;\n    transform: translate3d(0, -${position * 24}px, 0);\n  `\n      : undefined}\n`\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal(id)\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const position = [...openedModals].reverse().indexOf(id) // reverse method mutate array so we need to create a new array\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        data-animation={\n          openedModals.length > 1 &&\n          position === 0 &&\n          previsousOpenedModales.length < openedModals.length\n        }\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
22
|
+
const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
|
|
23
|
+
shouldForwardProp: (prop) => !["position", "size", "openedModals"].includes(prop),
|
|
24
|
+
target: "e1fims040"
|
|
25
|
+
} : {
|
|
26
|
+
shouldForwardProp: (prop) => !["position", "size", "openedModals"].includes(prop),
|
|
27
|
+
target: "e1fims040",
|
|
28
|
+
label: "StyledDialog"
|
|
29
|
+
})("background-color:", ({
|
|
30
|
+
theme
|
|
31
|
+
}) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
|
|
32
|
+
theme
|
|
33
|
+
}) => theme.radii.default, ";border:0;padding:", ({
|
|
34
|
+
theme
|
|
35
|
+
}) => theme.space["3"], ";width:", constants.MODAL_WIDTH.medium, "px;box-shadow:", ({
|
|
36
|
+
theme
|
|
37
|
+
}) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(constants.MODAL_WIDTH).map(([size, value]) => `
|
|
38
|
+
&[data-size="${size}"] {
|
|
39
|
+
width: ${value}px;
|
|
40
|
+
}
|
|
41
|
+
`), " ", Object.entries(constants.MODAL_PLACEMENT).map(([placement, value]) => `
|
|
42
|
+
&[data-placement="${placement}"] {
|
|
43
|
+
${value}
|
|
44
|
+
}
|
|
45
|
+
`), " &[data-animation='true']{animation:", animations.slideFromBottom, " 0.3s ease-in-out;}transition:width 0.3s ease-in-out,transform 0.3s ease-in-out;", ({
|
|
46
|
+
position,
|
|
47
|
+
size
|
|
48
|
+
}) => position > 0 ? `
|
|
49
|
+
width: ${constants.MODAL_WIDTH[size] - position * 50}px !important;
|
|
50
|
+
transform: translate3d(0, -${position * 24}px, 0);
|
|
51
|
+
` : void 0, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx"],"names":[],"mappings":"AA6CqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals'].includes(prop),\n})<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size }) =>\n    position > 0\n      ? `\n    width: ${MODAL_WIDTH[size] - position * 50}px !important;\n    transform: translate3d(0, -${position * 24}px, 0);\n  `\n      : undefined}\n`\n\nexport const Dialog = ({\n  children,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n  const {\n    registerModal,\n    unregisterModal,\n    openedModals,\n    previsousOpenedModales,\n  } = useModal()\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal(id)\n\n    return () => {\n      unregisterModal(id)\n    }\n  }, [id, registerModal, unregisterModal])\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    document.body.appendChild(element)\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [])\n\n  // Save the reassignment of eventHandler in the useEffect below\n  useEffect(() => {\n    onCloseRef.current = onClose\n  }, [onClose])\n\n  // On open focus the modal\n  useEffect(() => {\n    dialogRef.current?.focus()\n  }, [])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  // We need to reverse the array as the last opened modal should be the first to be with normal size\n  // while the first opened modal should shrink\n  const position = [...openedModals].reverse().indexOf(id) // reverse method mutate array so we need to create a new array\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      css={backdropCss}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n    >\n      <StyledDialog\n        css={dialogCss}\n        onKeyUp={handleKeyUp}\n        onKeyDown={handleFocusTrap}\n        className={className}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n        data-placement={placement}\n        data-size={size}\n        open\n        onClick={stopClick}\n        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        data-animation={\n          openedModals.length > 1 &&\n          position === 0 &&\n          previsousOpenedModales.length < openedModals.length\n        }\n        size={size}\n      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
52
|
+
const Dialog = ({
|
|
53
|
+
children,
|
|
54
|
+
placement,
|
|
55
|
+
onClose,
|
|
56
|
+
hideOnClickOutside,
|
|
57
|
+
size,
|
|
58
|
+
id,
|
|
59
|
+
ariaLabel,
|
|
60
|
+
className,
|
|
61
|
+
"data-testid": dataTestId,
|
|
62
|
+
preventBodyScroll,
|
|
63
|
+
hideOnEsc,
|
|
64
|
+
backdropClassName,
|
|
65
|
+
dialogCss,
|
|
66
|
+
backdropCss
|
|
67
|
+
}) => {
|
|
68
|
+
const containerRef = React.useRef(document.createElement("div"));
|
|
69
|
+
const dialogRef = React.useRef(null);
|
|
70
|
+
const onCloseRef = React.useRef(onClose);
|
|
71
|
+
const {
|
|
72
|
+
registerModal,
|
|
73
|
+
unregisterModal,
|
|
74
|
+
openedModals,
|
|
75
|
+
previsousOpenedModales
|
|
76
|
+
} = ModalProvider.useModal();
|
|
77
|
+
React.useEffect(() => {
|
|
78
|
+
registerModal(id);
|
|
79
|
+
return () => {
|
|
80
|
+
unregisterModal(id);
|
|
81
|
+
};
|
|
82
|
+
}, [id, registerModal, unregisterModal]);
|
|
83
|
+
React.useEffect(() => {
|
|
84
|
+
const element = containerRef.current;
|
|
85
|
+
document.body.appendChild(element);
|
|
86
|
+
return () => {
|
|
87
|
+
if (document.body.contains(element)) {
|
|
88
|
+
document.body.removeChild(element);
|
|
89
|
+
}
|
|
90
|
+
};
|
|
91
|
+
}, []);
|
|
92
|
+
React.useEffect(() => {
|
|
93
|
+
onCloseRef.current = onClose;
|
|
94
|
+
}, [onClose]);
|
|
95
|
+
React.useEffect(() => {
|
|
96
|
+
dialogRef.current?.focus();
|
|
97
|
+
}, []);
|
|
98
|
+
React.useEffect(() => {
|
|
99
|
+
const previousOverflow = document.body.style.overflow;
|
|
100
|
+
if (preventBodyScroll) {
|
|
101
|
+
document.body.style.overflow = "hidden";
|
|
102
|
+
}
|
|
103
|
+
return () => {
|
|
104
|
+
document.body.style.overflow = previousOverflow;
|
|
105
|
+
};
|
|
106
|
+
}, [preventBodyScroll]);
|
|
107
|
+
const stopFocus = React.useCallback((event) => {
|
|
108
|
+
event.stopPropagation();
|
|
109
|
+
}, []);
|
|
110
|
+
const stopClick = React.useCallback((event) => {
|
|
111
|
+
event.stopPropagation();
|
|
112
|
+
}, []);
|
|
113
|
+
const handleKeyUp = React.useCallback((event) => {
|
|
114
|
+
event.stopPropagation();
|
|
115
|
+
if (event.key === "Escape" && hideOnEsc) {
|
|
116
|
+
event.preventDefault();
|
|
117
|
+
onCloseRef.current();
|
|
118
|
+
}
|
|
119
|
+
}, [hideOnEsc]);
|
|
120
|
+
const handleClose = React.useCallback((event) => {
|
|
121
|
+
event.stopPropagation();
|
|
122
|
+
if (hideOnClickOutside) {
|
|
123
|
+
onCloseRef.current();
|
|
124
|
+
} else {
|
|
125
|
+
dialogRef.current?.focus();
|
|
126
|
+
}
|
|
127
|
+
}, [hideOnClickOutside]);
|
|
128
|
+
const handleFocusTrap = React.useCallback((event) => {
|
|
129
|
+
event.stopPropagation();
|
|
130
|
+
if (event.key === "Escape") {
|
|
131
|
+
event.preventDefault();
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
const isTabPressed = event.key === "Tab";
|
|
135
|
+
if (!isTabPressed) {
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
const focusableEls = dialogRef.current?.querySelectorAll("a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])") ?? [];
|
|
139
|
+
if (focusableEls.length === 0) {
|
|
140
|
+
event.preventDefault();
|
|
141
|
+
}
|
|
142
|
+
const firstFocusableEl = focusableEls[0];
|
|
143
|
+
const lastFocusableEl = focusableEls[focusableEls.length - 1];
|
|
144
|
+
if (event.shiftKey) {
|
|
145
|
+
if (document.activeElement === firstFocusableEl || document.activeElement === dialogRef.current) {
|
|
146
|
+
lastFocusableEl.focus();
|
|
147
|
+
event.preventDefault();
|
|
148
|
+
}
|
|
149
|
+
} else if (document.activeElement === lastFocusableEl || document.activeElement === dialogRef.current) {
|
|
150
|
+
firstFocusableEl.focus();
|
|
151
|
+
event.preventDefault();
|
|
152
|
+
}
|
|
153
|
+
}, []);
|
|
154
|
+
const stopCancel = (event) => {
|
|
155
|
+
event.preventDefault();
|
|
156
|
+
event.stopPropagation();
|
|
157
|
+
};
|
|
158
|
+
const position = [...openedModals].reverse().indexOf(id);
|
|
159
|
+
return reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, children: /* @__PURE__ */ jsxRuntime.jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true, onClick: stopClick, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, position, "data-animation": openedModals.length > 1 && position === 0 && previsousOpenedModales.length < openedModals.length, size, children }) }), containerRef.current);
|
|
160
|
+
};
|
|
161
|
+
exports.Dialog = Dialog;
|
|
162
|
+
exports.StyledDialog = StyledDialog;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { DialogProps, ModalPlacement, ModalSize } from '../types';
|
|
2
|
+
type StyledDialogProps = {
|
|
3
|
+
'data-size': ModalSize;
|
|
4
|
+
'data-placement': ModalPlacement;
|
|
5
|
+
position: number;
|
|
6
|
+
size: ModalSize;
|
|
7
|
+
};
|
|
8
|
+
export declare const StyledDialog: import("@emotion/styled").StyledComponent<{
|
|
9
|
+
theme?: import("@emotion/react").Theme;
|
|
10
|
+
as?: import("react").ElementType;
|
|
11
|
+
} & StyledDialogProps, import("react").DetailedHTMLProps<import("react").DialogHTMLAttributes<HTMLDialogElement>, HTMLDialogElement>, {}>;
|
|
12
|
+
export declare const Dialog: ({ children, placement, onClose, hideOnClickOutside, size, id, ariaLabel, className, "data-testid": dataTestId, preventBodyScroll, hideOnEsc, backdropClassName, dialogCss, backdropCss, }: DialogProps) => import("react").ReactPortal;
|
|
13
|
+
export {};
|