@ultraviolet/ui 1.82.3 → 1.83.1

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.
Files changed (32) hide show
  1. package/dist/components/MenuV2/MenuContent.cjs +128 -0
  2. package/dist/components/MenuV2/MenuContent.d.ts +54 -0
  3. package/dist/components/MenuV2/MenuContent.js +126 -0
  4. package/dist/components/MenuV2/MenuProvider.cjs +30 -0
  5. package/dist/components/MenuV2/MenuProvider.d.ts +14 -0
  6. package/dist/components/MenuV2/MenuProvider.js +30 -0
  7. package/dist/components/MenuV2/components/Group.cjs +28 -0
  8. package/dist/components/MenuV2/components/Group.d.ts +8 -0
  9. package/dist/components/MenuV2/components/Group.js +26 -0
  10. package/dist/components/MenuV2/components/Item.cjs +133 -0
  11. package/dist/components/MenuV2/components/Item.js +132 -0
  12. package/dist/components/MenuV2/constants.cjs +8 -0
  13. package/dist/components/MenuV2/constants.d.ts +5 -0
  14. package/dist/components/MenuV2/constants.js +8 -0
  15. package/dist/components/MenuV2/helpers.cjs +26 -0
  16. package/dist/components/MenuV2/helpers.d.ts +5 -0
  17. package/dist/components/MenuV2/helpers.js +26 -0
  18. package/dist/components/MenuV2/index.cjs +10 -84
  19. package/dist/components/MenuV2/index.d.ts +17 -42
  20. package/dist/components/MenuV2/index.js +11 -83
  21. package/dist/components/MenuV2/types.d.ts +48 -0
  22. package/dist/components/Modal/components/Dialog.cjs +2 -4
  23. package/dist/components/Modal/components/Dialog.js +2 -4
  24. package/dist/components/Tabs/TabMenuItem.cjs +3 -3
  25. package/dist/components/Tabs/TabMenuItem.js +3 -3
  26. package/package.json +5 -5
  27. package/dist/components/MenuV2/Group.cjs +0 -23
  28. package/dist/components/MenuV2/Group.d.ts +0 -7
  29. package/dist/components/MenuV2/Group.js +0 -21
  30. package/dist/components/MenuV2/Item.cjs +0 -115
  31. package/dist/components/MenuV2/Item.js +0 -114
  32. /package/dist/components/MenuV2/{Item.d.ts → components/Item.d.ts} +0 -0
@@ -1,63 +1,38 @@
1
- import type { ButtonHTMLAttributes, ReactElement, ReactNode, Ref } from 'react';
2
- declare const SIZES: {
3
- small: string;
4
- medium: string;
5
- large: string;
6
- };
7
- export type DisclosureProps = {
8
- visible: boolean;
9
- };
10
- type DisclosureElement = ((disclosure: DisclosureProps) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>) | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {
11
- ref?: Ref<HTMLButtonElement>;
12
- });
13
- type ChildMenuProps = {
14
- toggle: () => void;
15
- };
16
- /**
17
- * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.
18
- * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a
19
- * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.
20
- * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.
21
- */
1
+ import type { Ref } from 'react';
22
2
  export declare const MenuV2: import("react").ForwardRefExoticComponent<{
23
3
  id?: string;
24
4
  ariaLabel?: string;
25
- children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode);
5
+ children?: import("react").ReactNode | (({ toggle }: {
6
+ toggle: () => void;
7
+ }) => import("react").ReactNode);
26
8
  className?: string;
27
- disclosure: DisclosureElement;
9
+ disclosure: (import("react").ReactElement<import("react").ButtonHTMLAttributes<HTMLButtonElement>, string | import("react").JSXElementConstructor<any>> & {
10
+ ref?: Ref<HTMLButtonElement>;
11
+ }) | ((disclosure: import("./types").DisclosureProps) => import("react").ReactElement<import("react").ButtonHTMLAttributes<HTMLButtonElement>>);
28
12
  hasArrow?: boolean;
29
13
  visible?: boolean;
30
14
  'data-testid'?: string;
31
15
  maxHeight?: string;
32
- /**
33
- * @deprecated: use `size` instead
34
- */
35
16
  maxWidth?: string;
36
- /**
37
- * By default, the portal target is children container or document.body if children is a function. You can override this
38
- * behavior by setting a portalTarget prop.
39
- */
40
17
  portalTarget?: HTMLElement;
41
- size?: keyof typeof SIZES;
42
- /**
43
- * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.
44
- * If set to `hover`, the menu will open when the user hovers over the disclosure.
45
- */
18
+ size?: keyof typeof import("./constants").SIZES;
46
19
  triggerMethod?: "click" | "hover";
20
+ searchable?: boolean;
21
+ hideOnClickItem?: boolean;
47
22
  } & Pick<{
48
23
  id?: string;
49
- children: ReactNode | ((renderProps: {
24
+ children: import("react").ReactNode | ((renderProps: {
50
25
  className?: string;
51
26
  onBlur: () => void;
52
27
  onFocus: () => void;
53
28
  onPointerEnter: () => void;
54
29
  onPointerLeave: () => void;
55
30
  ref: import("react").RefObject<HTMLDivElement | null>;
56
- }) => ReactNode);
31
+ }) => import("react").ReactNode);
57
32
  maxWidth?: number | string;
58
33
  placement?: import("../Popup/helpers").PopupPlacement;
59
34
  align?: import("../Popup/helpers").PopupAlign;
60
- text?: ReactNode;
35
+ text?: import("react").ReactNode;
61
36
  className?: string;
62
37
  containerFullWidth?: boolean;
63
38
  visible?: boolean;
@@ -81,16 +56,16 @@ export declare const MenuV2: import("react").ForwardRefExoticComponent<{
81
56
  disabled?: boolean | undefined;
82
57
  tooltip?: string | undefined;
83
58
  className?: string | undefined;
84
- children: ReactNode;
59
+ children: import("react").ReactNode;
85
60
  onClick?: import("react").MouseEventHandler<HTMLElement> | undefined;
86
61
  borderless?: boolean;
87
62
  sentiment?: "primary" | "neutral" | "danger";
88
63
  active?: boolean;
89
64
  'data-testid'?: string;
90
65
  } & import("react").RefAttributes<HTMLElement>>;
91
- Group: ({ label, children }: {
66
+ Group: ({ label, children, labelDescription }: {
92
67
  label: string;
93
- children: ReactNode;
68
+ children: import("react").ReactNode;
69
+ labelDescription?: import("react").ReactNode;
94
70
  }) => import("@emotion/react/jsx-runtime").JSX.Element;
95
71
  };
96
- export {};
@@ -1,88 +1,16 @@
1
1
  import { jsx } from "@emotion/react/jsx-runtime";
2
- import _styled from "@emotion/styled/base";
3
- import { forwardRef, useState, useRef, useId, isValidElement, useImperativeHandle, cloneElement } from "react";
4
- import { Popup } from "../Popup/index.js";
5
- import { Stack } from "../Stack/index.js";
6
- import { Group } from "./Group.js";
7
- import Item from "./Item.js";
8
- const SIZES = {
9
- small: "180px",
10
- medium: "280px",
11
- large: "380px"
12
- };
13
- const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "production" ? {
14
- shouldForwardProp: (prop) => !["size"].includes(prop),
15
- target: "e1jn11gg1"
16
- } : {
17
- shouldForwardProp: (prop) => !["size"].includes(prop),
18
- target: "e1jn11gg1",
19
- label: "StyledPopup"
20
- })("background-color:", ({
21
- theme
22
- }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
23
- theme
24
- }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
25
- theme
26
- }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
27
- size
28
- }) => SIZES[size], ";max-width:none;padding:", ({
29
- theme
30
- }) => `${theme.space["0.25"]} 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/MenuV2/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
31
- const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
32
- target: "e1jn11gg0"
33
- } : {
34
- target: "e1jn11gg0",
35
- label: "MenuList"
36
- })("overflow-y:auto;overflow-x:hidden;&:after,&:before{border:solid transparent;border-width:9px;content:' ';height:0;width:0;position:absolute;pointer-events:none;}&:after{border-color:transparent;}&:before{border-color:transparent;}background-color:", ({
37
- theme
38
- }) => theme.colors.other.elevation.background.raised, ";color:", ({
39
- theme
40
- }) => theme.colors.neutral.text, ";border-radius:", ({
41
- theme
42
- }) => theme.radii.default, ";position:relative;" + (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/MenuV2/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
43
- const FwdMenu = forwardRef(({
44
- id,
45
- ariaLabel = "Menu",
2
+ import { forwardRef } from "react";
3
+ import { Menu } from "./MenuContent.js";
4
+ import { MenuProvider } from "./MenuProvider.js";
5
+ import { Group } from "./components/Group.js";
6
+ import Item from "./components/Item.js";
7
+ const Component = forwardRef(({
46
8
  children,
47
- disclosure,
48
- hasArrow = false,
49
- placement = "bottom",
50
- visible = false,
51
- className,
52
- "data-testid": dataTestId,
53
- maxHeight,
54
- maxWidth,
55
- portalTarget = document.body,
56
- size = "small",
57
- triggerMethod = "click",
58
- dynamicDomRendering,
59
- align
60
- }, ref) => {
61
- const [isVisible, setIsVisible] = useState(visible);
62
- const popupRef = useRef(null);
63
- const disclosureRef = useRef(null);
64
- const tempId = useId();
65
- const finalId = `menu-${id ?? tempId}`;
66
- const target = isValidElement(disclosure) ? disclosure : disclosure({
67
- visible: isVisible
68
- });
69
- const innerRef = useRef(target);
70
- useImperativeHandle(ref, () => innerRef.current);
71
- const finalDisclosure = cloneElement(target, {
72
- onClick: (event) => {
73
- target.props.onClick?.(event);
74
- setIsVisible(!isVisible);
75
- },
76
- "aria-haspopup": "dialog",
77
- "aria-expanded": isVisible,
78
- // @ts-expect-error not sure how to fix this
79
- ref: disclosureRef
80
- });
81
- return /* @__PURE__ */ jsx(StyledPopup, { debounceDelay: triggerMethod === "hover" ? 250 : 0, hideOnClickOutside: true, "aria-label": ariaLabel, className, visible: triggerMethod === "click" ? isVisible : void 0, placement, hasArrow, "data-has-arrow": hasArrow, role: "dialog", id: finalId, ref: popupRef, onClose: () => setIsVisible(false), tabIndex: -1, maxHeight: maxHeight ?? "480px", maxWidth, size, text: /* @__PURE__ */ jsx(MenuList, { "data-testid": dataTestId, className, role: "menu", children: typeof children === "function" ? children({
82
- toggle: () => setIsVisible(!isVisible)
83
- }) : children }), portalTarget, dynamicDomRendering, align, children: finalDisclosure });
84
- });
85
- const MenuV2 = Object.assign(FwdMenu, {
9
+ visible,
10
+ hideOnClickItem,
11
+ ...props
12
+ }, ref) => /* @__PURE__ */ jsx(MenuProvider, { visible, hideOnClickItem, children: /* @__PURE__ */ jsx(Menu, { ...props, ref, children }) }));
13
+ const MenuV2 = Object.assign(Component, {
86
14
  Item,
87
15
  Group
88
16
  });
@@ -0,0 +1,48 @@
1
+ import type { ButtonHTMLAttributes, ComponentProps, ReactElement, ReactNode, Ref } from 'react';
2
+ import type { Popup } from '../Popup';
3
+ import type { SIZES } from './constants';
4
+ type ChildMenuProps = {
5
+ toggle: () => void;
6
+ };
7
+ type DisclosureElement = ((disclosure: DisclosureProps) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>) | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {
8
+ ref?: Ref<HTMLButtonElement>;
9
+ });
10
+ export type DisclosureProps = {
11
+ visible: boolean;
12
+ };
13
+ export type MenuProps = {
14
+ id?: string;
15
+ ariaLabel?: string;
16
+ children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode);
17
+ className?: string;
18
+ disclosure: DisclosureElement;
19
+ hasArrow?: boolean;
20
+ visible?: boolean;
21
+ 'data-testid'?: string;
22
+ maxHeight?: string;
23
+ /**
24
+ * @deprecated: use `size` instead
25
+ */
26
+ maxWidth?: string;
27
+ /**
28
+ * By default, the portal target is children container or document.body if children is a function. You can override this
29
+ * behavior by setting a portalTarget prop.
30
+ */
31
+ portalTarget?: HTMLElement;
32
+ size?: keyof typeof SIZES;
33
+ /**
34
+ * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.
35
+ * If set to `hover`, the menu will open when the user hovers over the disclosure.
36
+ */
37
+ triggerMethod?: 'click' | 'hover';
38
+ /**
39
+ * If set to true, the menu will be searchable. This will add a search input at the top of the menu.
40
+ * This doesn't work when children is a function.
41
+ */
42
+ searchable?: boolean;
43
+ /**
44
+ * When set to true the menu will automatically close when a `MenuV2.Item` is clicked.
45
+ */
46
+ hideOnClickItem?: boolean;
47
+ } & Pick<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering' | 'align'>;
48
+ export {};
@@ -18,7 +18,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
18
18
  theme
19
19
  }) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
20
20
  theme
21
- }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{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":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
21
+ }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{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":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
22
22
  const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
23
23
  shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
24
24
  target: "e1fims040"
@@ -49,7 +49,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
49
49
  }) => position > 0 ? `
50
50
  width: calc(${constants.MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
51
51
  transform: translate3d(0, -${top}px, 0);
52
- ` : 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":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
52
+ ` : 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":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
53
53
  const stopCancel = (event) => {
54
54
  event.preventDefault();
55
55
  event.stopPropagation();
@@ -130,8 +130,6 @@ const Dialog = ({
130
130
  event.stopPropagation();
131
131
  if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target)) {
132
132
  onCloseRef.current();
133
- } else {
134
- dialogRef.current?.focus();
135
133
  }
136
134
  }, [hideOnClickOutside]);
137
135
  const handleFocusTrap = React.useCallback((event) => {
@@ -14,7 +14,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "
14
14
  theme
15
15
  }) => theme.colors.overlay, ";z-index:1;opacity:0;&[data-open='true']{padding:", ({
16
16
  theme
17
- }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{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":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
17
+ }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}&[data-visible='true']{opacity:1;}&[data-animation='true']{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":"AAc2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
18
18
  const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV === "production" ? {
19
19
  shouldForwardProp: (prop) => !["position", "size", "openedModals", "top"].includes(prop),
20
20
  target: "e1fims040"
@@ -45,7 +45,7 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
45
45
  }) => position > 0 ? `
46
46
  width: calc(${MODAL_WIDTH[size]}rem - ${position * 50}px) !important;
47
47
  transform: translate3d(0, -${top}px, 0);
48
- ` : 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":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
48
+ ` : 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":"AAuDqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/components/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef, useState } from 'react'\nimport { createPortal } from 'react-dom'\nimport { slideFromBottom } from '../../../utils/animations'\nimport { useModal } from '../ModalProvider'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from '../constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from '../types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n  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  &[data-animation='true'] {\n    overflow: hidden;\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  ${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-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}: 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  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n\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      ) {\n        onCloseRef.current()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // 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  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      >\n        {children}\n      </StyledDialog>\n    </StyledBackdrop>,\n    containerRef.current,\n  )\n}\n"]} */"));
49
49
  const stopCancel = (event) => {
50
50
  event.preventDefault();
51
51
  event.stopPropagation();
@@ -126,8 +126,6 @@ const Dialog = ({
126
126
  event.stopPropagation();
127
127
  if (hideOnClickOutside && dialogRef.current && !dialogRef.current.contains(event.target)) {
128
128
  onCloseRef.current();
129
- } else {
130
- dialogRef.current?.focus();
131
129
  }
132
130
  }, [hideOnClickOutside]);
133
131
  const handleFocusTrap = useCallback((event) => {
@@ -3,18 +3,18 @@ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
3
  const jsxRuntime = require("@emotion/react/jsx-runtime");
4
4
  const _styled = require("@emotion/styled/base");
5
5
  const React = require("react");
6
- const Item = require("../MenuV2/Item.cjs");
6
+ const index = require("../MenuV2/index.cjs");
7
7
  const TabsContext = require("./TabsContext.cjs");
8
8
  const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
9
9
  const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
10
- const StyledMenuItem = /* @__PURE__ */ _styled__default.default(Item, process.env.NODE_ENV === "production" ? {
10
+ const StyledMenuItem = /* @__PURE__ */ _styled__default.default(index.MenuV2.Item, process.env.NODE_ENV === "production" ? {
11
11
  target: "eeet93f0"
12
12
  } : {
13
13
  target: "eeet93f0",
14
14
  label: "StyledMenuItem"
15
15
  })("&[aria-selected='true']{color:", ({
16
16
  theme
17
- }) => theme.colors.primary.text, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYnMvVGFiTWVudUl0ZW0udHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQU1tQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9UYWJzL1RhYk1lbnVJdGVtLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IEl0ZW0gZnJvbSAnLi4vTWVudVYyL0l0ZW0nXG5pbXBvcnQgeyB1c2VUYWJzQ29udGV4dCB9IGZyb20gJy4vVGFic0NvbnRleHQnXG5cbmNvbnN0IFN0eWxlZE1lbnVJdGVtID0gc3R5bGVkKEl0ZW0pYFxuICAmW2FyaWEtc2VsZWN0ZWQ9J3RydWUnXSB7XG4gICAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLnByaW1hcnkudGV4dH07XG4gIH1cbmBcblxudHlwZSBUYWJNZW51SXRlbVByb3BzID0ge1xuICB2YWx1ZT86IHN0cmluZyB8IG51bWJlclxufSAmIENvbXBvbmVudFByb3BzPHR5cGVvZiBTdHlsZWRNZW51SXRlbT5cblxuZXhwb3J0IGNvbnN0IFRhYk1lbnVJdGVtID0gKHtcbiAgdmFsdWUsXG4gIGNoaWxkcmVuLFxuICBvbkNsaWNrLFxuICAuLi5wcm9wc1xufTogVGFiTWVudUl0ZW1Qcm9wcykgPT4ge1xuICBjb25zdCB7IHNlbGVjdGVkLCBvbkNoYW5nZSB9ID0gdXNlVGFic0NvbnRleHQoKVxuXG4gIGNvbnN0IGlzU2VsZWN0ZWQgPSB1c2VNZW1vKFxuICAgICgpID0+IHZhbHVlICE9PSB1bmRlZmluZWQgJiYgc2VsZWN0ZWQgPT09IHZhbHVlLFxuICAgIFt2YWx1ZSwgc2VsZWN0ZWRdLFxuICApXG5cbiAgcmV0dXJuIChcbiAgICA8U3R5bGVkTWVudUl0ZW1cbiAgICAgIG9uQ2xpY2s9e2V2ZW50ID0+IHtcbiAgICAgICAgaWYgKHZhbHVlICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICBvbkNoYW5nZSh2YWx1ZSlcbiAgICAgICAgfVxuICAgICAgICBvbkNsaWNrPy4oZXZlbnQpXG4gICAgICB9fVxuICAgICAgYXJpYS1zZWxlY3RlZD17aXNTZWxlY3RlZH1cbiAgICAgIHsuLi5wcm9wc31cbiAgICA+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9TdHlsZWRNZW51SXRlbT5cbiAgKVxufVxuIl19 */"));
17
+ }) => theme.colors.primary.text, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYnMvVGFiTWVudUl0ZW0udHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQU0wQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9UYWJzL1RhYk1lbnVJdGVtLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgTWVudVYyIH0gZnJvbSAnLi4vTWVudVYyJ1xuaW1wb3J0IHsgdXNlVGFic0NvbnRleHQgfSBmcm9tICcuL1RhYnNDb250ZXh0J1xuXG5jb25zdCBTdHlsZWRNZW51SXRlbSA9IHN0eWxlZChNZW51VjIuSXRlbSlgXG4gICZbYXJpYS1zZWxlY3RlZD0ndHJ1ZSddIHtcbiAgICBjb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMucHJpbWFyeS50ZXh0fTtcbiAgfVxuYFxuXG50eXBlIFRhYk1lbnVJdGVtUHJvcHMgPSB7XG4gIHZhbHVlPzogc3RyaW5nIHwgbnVtYmVyXG59ICYgQ29tcG9uZW50UHJvcHM8dHlwZW9mIFN0eWxlZE1lbnVJdGVtPlxuXG5leHBvcnQgY29uc3QgVGFiTWVudUl0ZW0gPSAoe1xuICB2YWx1ZSxcbiAgY2hpbGRyZW4sXG4gIG9uQ2xpY2ssXG4gIC4uLnByb3BzXG59OiBUYWJNZW51SXRlbVByb3BzKSA9PiB7XG4gIGNvbnN0IHsgc2VsZWN0ZWQsIG9uQ2hhbmdlIH0gPSB1c2VUYWJzQ29udGV4dCgpXG5cbiAgY29uc3QgaXNTZWxlY3RlZCA9IHVzZU1lbW8oXG4gICAgKCkgPT4gdmFsdWUgIT09IHVuZGVmaW5lZCAmJiBzZWxlY3RlZCA9PT0gdmFsdWUsXG4gICAgW3ZhbHVlLCBzZWxlY3RlZF0sXG4gIClcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRNZW51SXRlbVxuICAgICAgb25DbGljaz17ZXZlbnQgPT4ge1xuICAgICAgICBpZiAodmFsdWUgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgIG9uQ2hhbmdlKHZhbHVlKVxuICAgICAgICB9XG4gICAgICAgIG9uQ2xpY2s/LihldmVudClcbiAgICAgIH19XG4gICAgICBhcmlhLXNlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgey4uLnByb3BzfVxuICAgID5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L1N0eWxlZE1lbnVJdGVtPlxuICApXG59XG4iXX0= */"));
18
18
  const TabMenuItem = ({
19
19
  value,
20
20
  children,
@@ -1,16 +1,16 @@
1
1
  import { jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { useMemo } from "react";
4
- import Item from "../MenuV2/Item.js";
4
+ import { MenuV2 } from "../MenuV2/index.js";
5
5
  import { useTabsContext } from "./TabsContext.js";
6
- const StyledMenuItem = /* @__PURE__ */ _styled(Item, process.env.NODE_ENV === "production" ? {
6
+ const StyledMenuItem = /* @__PURE__ */ _styled(MenuV2.Item, process.env.NODE_ENV === "production" ? {
7
7
  target: "eeet93f0"
8
8
  } : {
9
9
  target: "eeet93f0",
10
10
  label: "StyledMenuItem"
11
11
  })("&[aria-selected='true']{color:", ({
12
12
  theme
13
- }) => theme.colors.primary.text, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYnMvVGFiTWVudUl0ZW0udHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQU1tQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9UYWJzL1RhYk1lbnVJdGVtLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IEl0ZW0gZnJvbSAnLi4vTWVudVYyL0l0ZW0nXG5pbXBvcnQgeyB1c2VUYWJzQ29udGV4dCB9IGZyb20gJy4vVGFic0NvbnRleHQnXG5cbmNvbnN0IFN0eWxlZE1lbnVJdGVtID0gc3R5bGVkKEl0ZW0pYFxuICAmW2FyaWEtc2VsZWN0ZWQ9J3RydWUnXSB7XG4gICAgY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLnByaW1hcnkudGV4dH07XG4gIH1cbmBcblxudHlwZSBUYWJNZW51SXRlbVByb3BzID0ge1xuICB2YWx1ZT86IHN0cmluZyB8IG51bWJlclxufSAmIENvbXBvbmVudFByb3BzPHR5cGVvZiBTdHlsZWRNZW51SXRlbT5cblxuZXhwb3J0IGNvbnN0IFRhYk1lbnVJdGVtID0gKHtcbiAgdmFsdWUsXG4gIGNoaWxkcmVuLFxuICBvbkNsaWNrLFxuICAuLi5wcm9wc1xufTogVGFiTWVudUl0ZW1Qcm9wcykgPT4ge1xuICBjb25zdCB7IHNlbGVjdGVkLCBvbkNoYW5nZSB9ID0gdXNlVGFic0NvbnRleHQoKVxuXG4gIGNvbnN0IGlzU2VsZWN0ZWQgPSB1c2VNZW1vKFxuICAgICgpID0+IHZhbHVlICE9PSB1bmRlZmluZWQgJiYgc2VsZWN0ZWQgPT09IHZhbHVlLFxuICAgIFt2YWx1ZSwgc2VsZWN0ZWRdLFxuICApXG5cbiAgcmV0dXJuIChcbiAgICA8U3R5bGVkTWVudUl0ZW1cbiAgICAgIG9uQ2xpY2s9e2V2ZW50ID0+IHtcbiAgICAgICAgaWYgKHZhbHVlICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICBvbkNoYW5nZSh2YWx1ZSlcbiAgICAgICAgfVxuICAgICAgICBvbkNsaWNrPy4oZXZlbnQpXG4gICAgICB9fVxuICAgICAgYXJpYS1zZWxlY3RlZD17aXNTZWxlY3RlZH1cbiAgICAgIHsuLi5wcm9wc31cbiAgICA+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9TdHlsZWRNZW51SXRlbT5cbiAgKVxufVxuIl19 */"));
13
+ }) => theme.colors.primary.text, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYnMvVGFiTWVudUl0ZW0udHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQU0wQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9UYWJzL1RhYk1lbnVJdGVtLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTWVtbyB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgTWVudVYyIH0gZnJvbSAnLi4vTWVudVYyJ1xuaW1wb3J0IHsgdXNlVGFic0NvbnRleHQgfSBmcm9tICcuL1RhYnNDb250ZXh0J1xuXG5jb25zdCBTdHlsZWRNZW51SXRlbSA9IHN0eWxlZChNZW51VjIuSXRlbSlgXG4gICZbYXJpYS1zZWxlY3RlZD0ndHJ1ZSddIHtcbiAgICBjb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMucHJpbWFyeS50ZXh0fTtcbiAgfVxuYFxuXG50eXBlIFRhYk1lbnVJdGVtUHJvcHMgPSB7XG4gIHZhbHVlPzogc3RyaW5nIHwgbnVtYmVyXG59ICYgQ29tcG9uZW50UHJvcHM8dHlwZW9mIFN0eWxlZE1lbnVJdGVtPlxuXG5leHBvcnQgY29uc3QgVGFiTWVudUl0ZW0gPSAoe1xuICB2YWx1ZSxcbiAgY2hpbGRyZW4sXG4gIG9uQ2xpY2ssXG4gIC4uLnByb3BzXG59OiBUYWJNZW51SXRlbVByb3BzKSA9PiB7XG4gIGNvbnN0IHsgc2VsZWN0ZWQsIG9uQ2hhbmdlIH0gPSB1c2VUYWJzQ29udGV4dCgpXG5cbiAgY29uc3QgaXNTZWxlY3RlZCA9IHVzZU1lbW8oXG4gICAgKCkgPT4gdmFsdWUgIT09IHVuZGVmaW5lZCAmJiBzZWxlY3RlZCA9PT0gdmFsdWUsXG4gICAgW3ZhbHVlLCBzZWxlY3RlZF0sXG4gIClcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRNZW51SXRlbVxuICAgICAgb25DbGljaz17ZXZlbnQgPT4ge1xuICAgICAgICBpZiAodmFsdWUgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgIG9uQ2hhbmdlKHZhbHVlKVxuICAgICAgICB9XG4gICAgICAgIG9uQ2xpY2s/LihldmVudClcbiAgICAgIH19XG4gICAgICBhcmlhLXNlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgey4uLnByb3BzfVxuICAgID5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L1N0eWxlZE1lbnVJdGVtPlxuICApXG59XG4iXX0= */"));
14
14
  const TabMenuItem = ({
15
15
  value,
16
16
  children,