@ultraviolet/ui 2.0.0-beta.14 → 2.0.0-beta.15
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/LineChart/CustomLegend.cjs +9 -9
- package/dist/components/LineChart/CustomLegend.js +9 -9
- package/dist/components/Link/index.cjs +12 -29
- package/dist/components/Link/index.d.ts +1 -11
- package/dist/components/Link/index.js +12 -29
- package/dist/components/List/Cell.cjs +2 -15
- package/dist/components/List/Cell.d.ts +0 -6
- package/dist/components/List/Cell.js +2 -15
- package/dist/components/List/Row.cjs +7 -7
- package/dist/components/List/Row.js +7 -7
- package/dist/components/List/index.d.ts +0 -1
- package/dist/components/Menu/MenuContent.cjs +8 -12
- package/dist/components/Menu/MenuContent.d.ts +0 -3
- package/dist/components/Menu/MenuContent.js +8 -12
- package/dist/components/Menu/constants.cjs +0 -1
- package/dist/components/Menu/constants.js +0 -1
- package/dist/components/Menu/index.d.ts +0 -2
- package/dist/components/Menu/types.d.ts +0 -9
- package/dist/components/Modal/ModalContent.cjs +2 -8
- package/dist/components/Modal/ModalContent.d.ts +1 -2
- package/dist/components/Modal/ModalContent.js +2 -8
- package/dist/components/Modal/components/Dialog.cjs +6 -8
- package/dist/components/Modal/components/Dialog.d.ts +1 -1
- package/dist/components/Modal/components/Dialog.js +6 -8
- package/dist/components/Modal/components/Disclosure.cjs +0 -3
- package/dist/components/Modal/components/Disclosure.js +0 -3
- package/dist/components/Modal/index.cjs +2 -7
- package/dist/components/Modal/index.d.ts +1 -18
- package/dist/components/Modal/index.js +2 -7
- package/dist/components/Modal/types.d.ts +0 -15
- package/dist/components/RadioGroup/index.cjs +2 -3
- package/dist/components/RadioGroup/index.d.ts +2 -7
- package/dist/components/RadioGroup/index.js +2 -3
- package/dist/components/Separator/index.cjs +8 -9
- package/dist/components/Separator/index.d.ts +1 -5
- package/dist/components/Separator/index.js +8 -9
- package/dist/components/Table/HeaderCell.cjs +4 -4
- package/dist/components/Table/HeaderCell.js +4 -4
- package/dist/components/Text/index.cjs +2 -4
- package/dist/components/Text/index.d.ts +2 -6
- package/dist/components/Text/index.js +2 -4
- package/package.json +3 -3
|
@@ -13,9 +13,7 @@ export declare const Menu: import("react").ForwardRefExoticComponent<{
|
|
|
13
13
|
visible?: boolean;
|
|
14
14
|
'data-testid'?: string;
|
|
15
15
|
maxHeight?: string;
|
|
16
|
-
maxWidth?: string;
|
|
17
16
|
portalTarget?: HTMLElement;
|
|
18
|
-
size?: keyof typeof import("./constants").SIZES;
|
|
19
17
|
triggerMethod?: "click" | "hover";
|
|
20
18
|
searchable?: boolean;
|
|
21
19
|
hideOnClickItem?: boolean;
|
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
import type { ButtonHTMLAttributes, ComponentProps, ReactElement, ReactNode, Ref } from 'react';
|
|
2
2
|
import type { Popup } from '../Popup';
|
|
3
|
-
import type { SIZES } from './constants';
|
|
4
3
|
type ChildMenuProps = {
|
|
5
4
|
toggle: () => void;
|
|
6
5
|
};
|
|
@@ -20,19 +19,11 @@ export type MenuProps = {
|
|
|
20
19
|
visible?: boolean;
|
|
21
20
|
'data-testid'?: string;
|
|
22
21
|
maxHeight?: string;
|
|
23
|
-
/**
|
|
24
|
-
* @deprecated: use `size` instead
|
|
25
|
-
*/
|
|
26
|
-
maxWidth?: string;
|
|
27
22
|
/**
|
|
28
23
|
* By default, the portal target is children container or document.body if children is a function. You can override this
|
|
29
24
|
* behavior by setting a portalTarget prop.
|
|
30
25
|
*/
|
|
31
26
|
portalTarget?: HTMLElement;
|
|
32
|
-
/**
|
|
33
|
-
* @deprecated The size of the menu is automatic now to fit the content. Use this prop to set a fixed size.
|
|
34
|
-
*/
|
|
35
|
-
size?: keyof typeof SIZES;
|
|
36
27
|
/**
|
|
37
28
|
* The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.
|
|
38
29
|
* If set to `hover`, the menu will open when the user hovers over the disclosure.
|
|
@@ -17,11 +17,10 @@ const StyledContainer = /* @__PURE__ */ _styled__default.default("div", process.
|
|
|
17
17
|
theme
|
|
18
18
|
}) => theme.space["2"], ";right:", ({
|
|
19
19
|
theme
|
|
20
|
-
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBVWtDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBDbG9zZUljb24gfSBmcm9tICdAdWx0cmF2aW9sZXQvaWNvbnMnXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IE1vZGFsIH0gZnJvbSAnLidcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5gXG5cbnR5cGUgTW9kYWxDb250ZW50UHJvcHMgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgTW9kYWw+
|
|
20
|
+
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBVWtDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBDbG9zZUljb24gfSBmcm9tICdAdWx0cmF2aW9sZXQvaWNvbnMnXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IE1vZGFsIH0gZnJvbSAnLidcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5gXG5cbnR5cGUgTW9kYWxDb250ZW50UHJvcHMgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgTW9kYWw+ICYge1xuICB2aXNpYmxlOiBib29sZWFuXG4gIG9wZW46IGJvb2xlYW5cbiAgcGxhY2VtZW50OiBNb2RhbFBsYWNlbWVudFxuICBmaW5hbFNpemU6IE1vZGFsU2l6ZVxuICBmaW5hbElkOiBzdHJpbmdcbiAgaGFuZGxlT3BlbjogKCkgPT4gdm9pZFxuICBoYW5kbGVUb2dnbGU6ICgpID0+IHZvaWRcbiAgaGFuZGxlQ2xvc2U6ICgpID0+IHZvaWRcbiAgZGF0YVRlc3RJZD86IHN0cmluZ1xuICBpbWFnZT86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTW9kYWxDb250ZW50ID0gKHtcbiAgdmlzaWJsZSxcbiAgb3BlbixcbiAgcGxhY2VtZW50LFxuICBmaW5hbFNpemUsXG4gIGFyaWFMYWJlbCxcbiAgaGlkZU9uQ2xpY2tPdXRzaWRlLFxuICBoaWRlT25Fc2MsXG4gIHByZXZlbnRCb2R5U2Nyb2xsLFxuICBoYW5kbGVDbG9zZSxcbiAgY2xhc3NOYW1lLFxuICBiYWNrZHJvcENsYXNzTmFtZSxcbiAgZGF0YVRlc3RJZCxcbiAgaXNDbG9zYWJsZSxcbiAgY2hpbGRyZW4sXG4gIGhhbmRsZU9wZW4sXG4gIGhhbmRsZVRvZ2dsZSxcbiAgZmluYWxJZCxcbiAgaW1hZ2UsXG59OiBNb2RhbENvbnRlbnRQcm9wcykgPT5cbiAgdmlzaWJsZSB8fCBvcGVuID8gKFxuICAgIDxEaWFsb2dcbiAgICAgIG9wZW49e3Zpc2libGUgfHwgb3Blbn1cbiAgICAgIHBsYWNlbWVudD17cGxhY2VtZW50fVxuICAgICAgc2l6ZT17ZmluYWxTaXplfVxuICAgICAgYXJpYUxhYmVsPXthcmlhTGFiZWx9XG4gICAgICBoaWRlT25DbGlja091dHNpZGU9e2hpZGVPbkNsaWNrT3V0c2lkZX1cbiAgICAgIGhpZGVPbkVzYz17aGlkZU9uRXNjfVxuICAgICAgcHJldmVudEJvZHlTY3JvbGw9e3ByZXZlbnRCb2R5U2Nyb2xsfVxuICAgICAgb25DbG9zZT17aGFuZGxlQ2xvc2V9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGJhY2tkcm9wQ2xhc3NOYW1lPXtiYWNrZHJvcENsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgaWQ9e2ZpbmFsSWR9XG4gICAgICBpbWFnZT17aW1hZ2V9XG4gICAgPlxuICAgICAgPD5cbiAgICAgICAge3R5cGVvZiBjaGlsZHJlbiA9PT0gJ2Z1bmN0aW9uJ1xuICAgICAgICAgID8gY2hpbGRyZW4oe1xuICAgICAgICAgICAgICB2aXNpYmxlLFxuICAgICAgICAgICAgICB0b2dnbGU6IGhhbmRsZVRvZ2dsZSxcbiAgICAgICAgICAgICAgbW9kYWxJZDogZmluYWxJZCxcbiAgICAgICAgICAgICAgY2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBzaG93OiBoYW5kbGVPcGVuLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICA6IGNoaWxkcmVufVxuICAgICAgICA8U3R5bGVkQ29udGFpbmVyPlxuICAgICAgICAgIHtpc0Nsb3NhYmxlID8gKFxuICAgICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgICBkYXRhLXRlc3RpZD17XG4gICAgICAgICAgICAgICAgZGF0YVRlc3RJZCA/IGAke2RhdGFUZXN0SWR9LWNsb3NlLWJ1dHRvbmAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBvbkNsaWNrPXtoYW5kbGVDbG9zZX1cbiAgICAgICAgICAgICAgdmFyaWFudD1cImdob3N0XCJcbiAgICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxDbG9zZUljb24gLz5cbiAgICAgICAgICAgIDwvQnV0dG9uPlxuICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgICAgIDwvPlxuICAgIDwvRGlhbG9nPlxuICApIDogbnVsbFxuIl19 */"));
|
|
21
21
|
const ModalContent = ({
|
|
22
22
|
visible,
|
|
23
23
|
open,
|
|
24
|
-
opened,
|
|
25
24
|
placement,
|
|
26
25
|
finalSize,
|
|
27
26
|
ariaLabel,
|
|
@@ -32,22 +31,17 @@ const ModalContent = ({
|
|
|
32
31
|
className,
|
|
33
32
|
backdropClassName,
|
|
34
33
|
dataTestId,
|
|
35
|
-
customDialogStyles,
|
|
36
|
-
customDialogBackdropStyles,
|
|
37
34
|
isClosable,
|
|
38
35
|
children,
|
|
39
36
|
handleOpen,
|
|
40
37
|
handleToggle,
|
|
41
38
|
finalId,
|
|
42
39
|
image
|
|
43
|
-
}) => visible || open
|
|
40
|
+
}) => visible || open ? /* @__PURE__ */ jsxRuntime.jsx(Dialog.Dialog, { open: visible || open, placement, size: finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, onClose: handleClose, className, backdropClassName, "data-testid": dataTestId, id: finalId, image, children: /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
|
|
44
41
|
typeof children === "function" ? children({
|
|
45
42
|
visible,
|
|
46
|
-
onClose: handleClose,
|
|
47
|
-
onOpen: handleOpen,
|
|
48
43
|
toggle: handleToggle,
|
|
49
44
|
modalId: finalId,
|
|
50
|
-
hide: handleClose,
|
|
51
45
|
close: handleClose,
|
|
52
46
|
show: handleOpen
|
|
53
47
|
}) : children,
|
|
@@ -4,7 +4,6 @@ import type { ModalPlacement, ModalSize } from './types';
|
|
|
4
4
|
type ModalContentProps = ComponentProps<typeof Modal> & {
|
|
5
5
|
visible: boolean;
|
|
6
6
|
open: boolean;
|
|
7
|
-
opened: boolean;
|
|
8
7
|
placement: ModalPlacement;
|
|
9
8
|
finalSize: ModalSize;
|
|
10
9
|
finalId: string;
|
|
@@ -14,5 +13,5 @@ type ModalContentProps = ComponentProps<typeof Modal> & {
|
|
|
14
13
|
dataTestId?: string;
|
|
15
14
|
image?: string;
|
|
16
15
|
};
|
|
17
|
-
export declare const ModalContent: ({ visible, open,
|
|
16
|
+
export declare const ModalContent: ({ visible, open, placement, finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, handleClose, className, backdropClassName, dataTestId, isClosable, children, handleOpen, handleToggle, finalId, image, }: ModalContentProps) => import("@emotion/react/jsx-runtime").JSX.Element | null;
|
|
18
17
|
export {};
|
|
@@ -13,11 +13,10 @@ const StyledContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ===
|
|
|
13
13
|
theme
|
|
14
14
|
}) => theme.space["2"], ";right:", ({
|
|
15
15
|
theme
|
|
16
|
-
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBVWtDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBDbG9zZUljb24gfSBmcm9tICdAdWx0cmF2aW9sZXQvaWNvbnMnXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IE1vZGFsIH0gZnJvbSAnLidcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5gXG5cbnR5cGUgTW9kYWxDb250ZW50UHJvcHMgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgTW9kYWw+
|
|
16
|
+
}) => theme.space["2"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBVWtDIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL01vZGFsQ29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBDbG9zZUljb24gfSBmcm9tICdAdWx0cmF2aW9sZXQvaWNvbnMnXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IE1vZGFsIH0gZnJvbSAnLidcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4vY29tcG9uZW50cy9EaWFsb2cnXG5pbXBvcnQgdHlwZSB7IE1vZGFsUGxhY2VtZW50LCBNb2RhbFNpemUgfSBmcm9tICcuL3R5cGVzJ1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5gXG5cbnR5cGUgTW9kYWxDb250ZW50UHJvcHMgPSBDb21wb25lbnRQcm9wczx0eXBlb2YgTW9kYWw+ICYge1xuICB2aXNpYmxlOiBib29sZWFuXG4gIG9wZW46IGJvb2xlYW5cbiAgcGxhY2VtZW50OiBNb2RhbFBsYWNlbWVudFxuICBmaW5hbFNpemU6IE1vZGFsU2l6ZVxuICBmaW5hbElkOiBzdHJpbmdcbiAgaGFuZGxlT3BlbjogKCkgPT4gdm9pZFxuICBoYW5kbGVUb2dnbGU6ICgpID0+IHZvaWRcbiAgaGFuZGxlQ2xvc2U6ICgpID0+IHZvaWRcbiAgZGF0YVRlc3RJZD86IHN0cmluZ1xuICBpbWFnZT86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTW9kYWxDb250ZW50ID0gKHtcbiAgdmlzaWJsZSxcbiAgb3BlbixcbiAgcGxhY2VtZW50LFxuICBmaW5hbFNpemUsXG4gIGFyaWFMYWJlbCxcbiAgaGlkZU9uQ2xpY2tPdXRzaWRlLFxuICBoaWRlT25Fc2MsXG4gIHByZXZlbnRCb2R5U2Nyb2xsLFxuICBoYW5kbGVDbG9zZSxcbiAgY2xhc3NOYW1lLFxuICBiYWNrZHJvcENsYXNzTmFtZSxcbiAgZGF0YVRlc3RJZCxcbiAgaXNDbG9zYWJsZSxcbiAgY2hpbGRyZW4sXG4gIGhhbmRsZU9wZW4sXG4gIGhhbmRsZVRvZ2dsZSxcbiAgZmluYWxJZCxcbiAgaW1hZ2UsXG59OiBNb2RhbENvbnRlbnRQcm9wcykgPT5cbiAgdmlzaWJsZSB8fCBvcGVuID8gKFxuICAgIDxEaWFsb2dcbiAgICAgIG9wZW49e3Zpc2libGUgfHwgb3Blbn1cbiAgICAgIHBsYWNlbWVudD17cGxhY2VtZW50fVxuICAgICAgc2l6ZT17ZmluYWxTaXplfVxuICAgICAgYXJpYUxhYmVsPXthcmlhTGFiZWx9XG4gICAgICBoaWRlT25DbGlja091dHNpZGU9e2hpZGVPbkNsaWNrT3V0c2lkZX1cbiAgICAgIGhpZGVPbkVzYz17aGlkZU9uRXNjfVxuICAgICAgcHJldmVudEJvZHlTY3JvbGw9e3ByZXZlbnRCb2R5U2Nyb2xsfVxuICAgICAgb25DbG9zZT17aGFuZGxlQ2xvc2V9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgIGJhY2tkcm9wQ2xhc3NOYW1lPXtiYWNrZHJvcENsYXNzTmFtZX1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgaWQ9e2ZpbmFsSWR9XG4gICAgICBpbWFnZT17aW1hZ2V9XG4gICAgPlxuICAgICAgPD5cbiAgICAgICAge3R5cGVvZiBjaGlsZHJlbiA9PT0gJ2Z1bmN0aW9uJ1xuICAgICAgICAgID8gY2hpbGRyZW4oe1xuICAgICAgICAgICAgICB2aXNpYmxlLFxuICAgICAgICAgICAgICB0b2dnbGU6IGhhbmRsZVRvZ2dsZSxcbiAgICAgICAgICAgICAgbW9kYWxJZDogZmluYWxJZCxcbiAgICAgICAgICAgICAgY2xvc2U6IGhhbmRsZUNsb3NlLFxuICAgICAgICAgICAgICBzaG93OiBoYW5kbGVPcGVuLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICA6IGNoaWxkcmVufVxuICAgICAgICA8U3R5bGVkQ29udGFpbmVyPlxuICAgICAgICAgIHtpc0Nsb3NhYmxlID8gKFxuICAgICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgICBkYXRhLXRlc3RpZD17XG4gICAgICAgICAgICAgICAgZGF0YVRlc3RJZCA/IGAke2RhdGFUZXN0SWR9LWNsb3NlLWJ1dHRvbmAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBvbkNsaWNrPXtoYW5kbGVDbG9zZX1cbiAgICAgICAgICAgICAgdmFyaWFudD1cImdob3N0XCJcbiAgICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICAgICAgc2VudGltZW50PVwibmV1dHJhbFwiXG4gICAgICAgICAgICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxDbG9zZUljb24gLz5cbiAgICAgICAgICAgIDwvQnV0dG9uPlxuICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgICAgIDwvPlxuICAgIDwvRGlhbG9nPlxuICApIDogbnVsbFxuIl19 */"));
|
|
17
17
|
const ModalContent = ({
|
|
18
18
|
visible,
|
|
19
19
|
open,
|
|
20
|
-
opened,
|
|
21
20
|
placement,
|
|
22
21
|
finalSize,
|
|
23
22
|
ariaLabel,
|
|
@@ -28,22 +27,17 @@ const ModalContent = ({
|
|
|
28
27
|
className,
|
|
29
28
|
backdropClassName,
|
|
30
29
|
dataTestId,
|
|
31
|
-
customDialogStyles,
|
|
32
|
-
customDialogBackdropStyles,
|
|
33
30
|
isClosable,
|
|
34
31
|
children,
|
|
35
32
|
handleOpen,
|
|
36
33
|
handleToggle,
|
|
37
34
|
finalId,
|
|
38
35
|
image
|
|
39
|
-
}) => visible || open
|
|
36
|
+
}) => visible || open ? /* @__PURE__ */ jsx(Dialog, { open: visible || open, placement, size: finalSize, ariaLabel, hideOnClickOutside, hideOnEsc, preventBodyScroll, onClose: handleClose, className, backdropClassName, "data-testid": dataTestId, id: finalId, image, children: /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
40
37
|
typeof children === "function" ? children({
|
|
41
38
|
visible,
|
|
42
|
-
onClose: handleClose,
|
|
43
|
-
onOpen: handleOpen,
|
|
44
39
|
toggle: handleToggle,
|
|
45
40
|
modalId: finalId,
|
|
46
|
-
hide: handleClose,
|
|
47
41
|
close: handleClose,
|
|
48
42
|
show: handleOpen
|
|
49
43
|
}) : children,
|
|
@@ -21,7 +21,7 @@ const StyledDiv = /* @__PURE__ */ _styled__default.default(index.Stack, process.
|
|
|
21
21
|
label: "StyledDiv"
|
|
22
22
|
})("width:100%;height:15rem;background-color:", ({
|
|
23
23
|
theme
|
|
24
|
-
}) => theme.colors.primary.background, ";overflow:hidden;" + (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":"AAiB+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\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      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
24
|
+
}) => theme.colors.primary.background, ";overflow:hidden;" + (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":"AAiB+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
25
25
|
const StyledImg = /* @__PURE__ */ _styled__default.default("img", process.env.NODE_ENV === "production" ? {
|
|
26
26
|
target: "e1fims043"
|
|
27
27
|
} : {
|
|
@@ -32,7 +32,7 @@ const StyledImg = /* @__PURE__ */ _styled__default.default("img", process.env.NO
|
|
|
32
32
|
styles: "margin-inline:auto;object-fit:cover;height:100%;width:100%"
|
|
33
33
|
} : {
|
|
34
34
|
name: "1dpiy6s",
|
|
35
|
-
styles: "margin-inline:auto;object-fit:cover;height:100%;width:100%/*# 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":"AAwB4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\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      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */",
|
|
35
|
+
styles: "margin-inline:auto;object-fit:cover;height:100%;width:100%/*# 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":"AAwB4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */",
|
|
36
36
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
37
37
|
});
|
|
38
38
|
const StyledStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -42,7 +42,7 @@ const StyledStack = /* @__PURE__ */ _styled__default.default(index.Stack, proces
|
|
|
42
42
|
label: "StyledStack"
|
|
43
43
|
})("padding:", ({
|
|
44
44
|
theme
|
|
45
|
-
}) => theme.space["3"], ";" + (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":"AA+BiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\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      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
45
|
+
}) => theme.space["3"], ";" + (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":"AA+BiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
46
46
|
const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
47
47
|
target: "e1fims041"
|
|
48
48
|
} : {
|
|
@@ -52,7 +52,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
|
|
|
52
52
|
theme
|
|
53
53
|
}) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
|
|
54
54
|
theme
|
|
55
|
-
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}" + (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":"AAmC2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\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      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
55
|
+
}) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}" + (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":"AAmC2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
56
56
|
const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
|
|
57
57
|
shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
|
|
58
58
|
target: "e1fims040"
|
|
@@ -83,7 +83,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
83
83
|
}) => position > 0 ? `
|
|
84
84
|
width: calc(${constants.MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
|
|
85
85
|
transform: translate3d(0, -${top}px, 0);
|
|
86
|
-
` : 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":"AAyEqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\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      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
86
|
+
` : 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":"AAyEqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { Stack } from '../../Stack'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledDiv = styled(Stack)`\n  width: 100%;\n  height: 15rem;\n  background-color: ${({ theme }) => theme.colors.primary.background};\n  overflow: hidden;\n`\n\nconst StyledImg = styled.img`\n  margin-inline: auto;\n  object-fit: cover;\n  height: 100%;\n  width: 100%;\n`\n\nconst StyledStack = styled(Stack)`\n  padding: ${({ theme }) => theme.space['3']};\n `\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  opacity: 0;\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  &[data-visible='true'] {\n    opacity: 1;\n  }\n\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n  position: number\n  size: ModalSize\n  top?: number\n}\n\nexport const StyledDialog = styled('dialog', {\n  shouldForwardProp: prop =>\n    !['position', 'size', 'openedModals', 'top'].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}rem;\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n  \n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}rem;\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-has-image='true'] {\n    padding: 0;\n  }\n\n  &[data-animation='true'] {\n    animation: ${slideFromBottom} 0.3s ease-in-out forwards;\n  }\n\n  transition: width 0.3s ease-in-out, transform 0.3s ease-in-out;\n\n  ${({ position, size, top }) =>\n    position > 0\n      ? `\n    width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;\n    transform: translate3d(0, -${top}px, 0);\n  `\n      : undefined}\n`\n\n// Prevent default behaviour on Escape\nconst stopCancel: ReactEventHandler = event => {\n  event.preventDefault()\n  event.stopPropagation()\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  image,\n}: DialogProps) => {\n  const [isVisible, setIsVisible] = useState(false)\n\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  useEffect(() => {\n    setIsVisible(true)\n  }, [])\n\n  // register/unregister the modal to handle nested modals\n  useEffect(() => {\n    registerModal({ id, ref: dialogRef })\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  // 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 realPosition = [...openedModals].findIndex(object => object.id === id)\n  const position = [...openedModals]\n    .reverse()\n    .findIndex(object => object.id === id) // reverse method mutate array so we need to create a new array\n  const modalAbove = openedModals[realPosition + 1]\n  const currentModalHeight = dialogRef.current?.offsetHeight\n  let top = 0\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      // if the user actually click outside of modal\n      if (\n        hideOnClickOutside &&\n        dialogRef.current &&\n        !dialogRef.current.contains(event.target as Node) &&\n        position === 0\n      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside, position],\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  if (\n    modalAbove?.ref &&\n    typeof modalAbove.ref === 'object' &&\n    'current' in modalAbove.ref &&\n    currentModalHeight\n  ) {\n    top =\n      (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 -\n      currentModalHeight / 2 +\n      20\n  }\n\n  const animation =\n    openedModals.length > 1 &&\n    position === 0 &&\n    previsousOpenedModales.length < openedModals.length\n\n  return createPortal(\n    <StyledBackdrop\n      data-open\n      onClick={handleClose}\n      className={backdropClassName}\n      data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n      onFocus={stopFocus}\n      data-animation={animation}\n      data-visible={isVisible}\n      id=\"backdrop-modal\"\n    >\n      <StyledDialog\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        onCancel={stopCancel}\n        onClose={stopCancel}\n        aria-modal\n        ref={dialogRef}\n        tabIndex={0}\n        position={position}\n        top={Math.max(top, 0)}\n        data-animation={animation}\n        size={size}\n        data-has-image={!!image}\n      >\n        {image ? (\n          <>\n            <StyledDiv alignItems=\"end\" justifyContent=\"center\">\n              <StyledImg src={image} alt=\"illustration\" />\n            </StyledDiv>\n            <StyledStack gap={5}>{children}</StyledStack>\n          </>\n        ) : (\n          children\n        )}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
|
|
87
87
|
const stopCancel = (event) => {
|
|
88
88
|
event.preventDefault();
|
|
89
89
|
event.stopPropagation();
|
|
@@ -101,8 +101,6 @@ const Dialog = ({
|
|
|
101
101
|
preventBodyScroll,
|
|
102
102
|
hideOnEsc,
|
|
103
103
|
backdropClassName,
|
|
104
|
-
dialogCss,
|
|
105
|
-
backdropCss,
|
|
106
104
|
image
|
|
107
105
|
}) => {
|
|
108
106
|
const [isVisible, setIsVisible] = react.useState(false);
|
|
@@ -201,7 +199,7 @@ const Dialog = ({
|
|
|
201
199
|
top = (modalAbove?.ref?.current?.offsetHeight ?? 0) / 2 - currentModalHeight / 2 + 20;
|
|
202
200
|
}
|
|
203
201
|
const animation = openedModals.length > 1 && position === 0 && previsousOpenedModales.length < openedModals.length;
|
|
204
|
-
return reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName,
|
|
202
|
+
return reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": true, onClick: handleClose, className: backdropClassName, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, "data-animation": animation, "data-visible": isVisible, id: "backdrop-modal", children: /* @__PURE__ */ jsxRuntime.jsx(StyledDialog, { onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open: true, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, position, top: Math.max(top, 0), "data-animation": animation, size, "data-has-image": !!image, children: image ? /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
|
|
205
203
|
/* @__PURE__ */ jsxRuntime.jsx(StyledDiv, { alignItems: "end", justifyContent: "center", children: /* @__PURE__ */ jsxRuntime.jsx(StyledImg, { src: image, alt: "illustration" }) }),
|
|
206
204
|
/* @__PURE__ */ jsxRuntime.jsx(StyledStack, { gap: 5, children })
|
|
207
205
|
] }) : children }) }), containerRef.current);
|
|
@@ -10,5 +10,5 @@ export declare const StyledDialog: import("@emotion/styled").StyledComponent<{
|
|
|
10
10
|
theme?: import("@emotion/react").Theme;
|
|
11
11
|
as?: React.ElementType;
|
|
12
12
|
} & StyledDialogProps, import("react").DetailedHTMLProps<import("react").DialogHTMLAttributes<HTMLDialogElement>, HTMLDialogElement>, {}>;
|
|
13
|
-
export declare const Dialog: ({ children, placement, onClose, hideOnClickOutside, size, id, ariaLabel, className, "data-testid": dataTestId, preventBodyScroll, hideOnEsc, backdropClassName,
|
|
13
|
+
export declare const Dialog: ({ children, placement, onClose, hideOnClickOutside, size, id, ariaLabel, className, "data-testid": dataTestId, preventBodyScroll, hideOnEsc, backdropClassName, image, }: DialogProps) => import("react").ReactPortal;
|
|
14
14
|
export {};
|