@ultraviolet/ui 1.61.1 → 1.63.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (50) hide show
  1. package/dist/components/Badge/index.cjs +9 -5
  2. package/dist/components/Badge/index.d.ts +5 -0
  3. package/dist/components/Badge/index.js +10 -6
  4. package/dist/components/ExpandableCard/components/Title.cjs +11 -0
  5. package/dist/components/ExpandableCard/components/Title.d.ts +7 -0
  6. package/dist/components/ExpandableCard/components/Title.js +11 -0
  7. package/dist/components/ExpandableCard/index.cjs +87 -0
  8. package/dist/components/ExpandableCard/index.d.ts +22 -0
  9. package/dist/components/ExpandableCard/index.js +85 -0
  10. package/dist/components/Modal/ModalContent.cjs +53 -0
  11. package/dist/components/Modal/ModalContent.d.ts +17 -0
  12. package/dist/components/Modal/ModalContent.js +51 -0
  13. package/dist/components/Modal/ModalProvider.cjs +40 -0
  14. package/dist/components/Modal/ModalProvider.d.ts +14 -0
  15. package/dist/components/Modal/ModalProvider.js +40 -0
  16. package/dist/components/Modal/components/Dialog.cjs +162 -0
  17. package/dist/components/Modal/components/Dialog.d.ts +13 -0
  18. package/dist/components/Modal/components/Dialog.js +160 -0
  19. package/dist/components/Modal/{Disclosure.d.ts → components/Disclosure.d.ts} +1 -1
  20. package/dist/components/Modal/index.cjs +5 -29
  21. package/dist/components/Modal/index.js +6 -28
  22. package/dist/components/Slider/components/DoubleSlider.cjs +53 -63
  23. package/dist/components/Slider/components/DoubleSlider.d.ts +1 -1
  24. package/dist/components/Slider/components/DoubleSlider.js +55 -65
  25. package/dist/components/Slider/components/Options.cjs +29 -7
  26. package/dist/components/Slider/components/Options.d.ts +23 -2
  27. package/dist/components/Slider/components/Options.js +26 -6
  28. package/dist/components/Slider/components/SingleSlider.cjs +32 -42
  29. package/dist/components/Slider/components/SingleSlider.d.ts +1 -1
  30. package/dist/components/Slider/components/SingleSlider.js +33 -43
  31. package/dist/components/Slider/index.cjs +4 -5
  32. package/dist/components/Slider/index.d.ts +1 -1
  33. package/dist/components/Slider/index.js +4 -5
  34. package/dist/components/Slider/styles.cjs +15 -52
  35. package/dist/components/Slider/styles.d.ts +1 -31
  36. package/dist/components/Slider/styles.js +13 -50
  37. package/dist/components/Slider/types.d.ts +2 -18
  38. package/dist/components/TextArea/index.cjs +5 -5
  39. package/dist/components/TextArea/index.js +5 -5
  40. package/dist/components/UnitInput/index.cjs +5 -5
  41. package/dist/components/UnitInput/index.js +5 -5
  42. package/dist/components/index.d.ts +1 -0
  43. package/dist/index.cjs +110 -108
  44. package/dist/index.js +2 -0
  45. package/package.json +4 -4
  46. package/dist/components/Modal/Dialog.cjs +0 -144
  47. package/dist/components/Modal/Dialog.d.ts +0 -11
  48. package/dist/components/Modal/Dialog.js +0 -142
  49. /package/dist/components/Modal/{Disclosure.cjs → components/Disclosure.cjs} +0 -0
  50. /package/dist/components/Modal/{Disclosure.js → components/Disclosure.js} +0 -0
@@ -1,144 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
- const jsxRuntime = require("@emotion/react/jsx-runtime");
4
- const _styled = require("@emotion/styled/base");
5
- const React = require("react");
6
- const reactDom = require("react-dom");
7
- const constants = require("./constants.cjs");
8
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
9
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
10
- const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
11
- target: "e1cqen9h1"
12
- } : {
13
- target: "e1cqen9h1",
14
- label: "StyledBackdrop"
15
- })("position:fixed;top:0;right:0;height:0;width:0;overflow:hidden;background-color:", ({
16
- theme
17
- }) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
18
- theme
19
- }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx"],"names":[],"mappings":"AAY2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from './constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from './types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\n        >\n          <StyledDialog\n            css={dialogCss}\n            onKeyUp={handleKeyUp}\n            onKeyDown={handleFocusTrap}\n            className={className}\n            id={id}\n            data-testid={dataTestId}\n            aria-label={ariaLabel}\n            data-placement={placement}\n            data-size={size}\n            open={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
20
- const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
21
- target: "e1cqen9h0"
22
- } : {
23
- target: "e1cqen9h0",
24
- label: "StyledDialog"
25
- })("background-color:", ({
26
- theme
27
- }) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
28
- theme
29
- }) => theme.radii.default, ";border:0;padding:", ({
30
- theme
31
- }) => theme.space["3"], ";width:", constants.MODAL_WIDTH.medium, "px;box-shadow:", ({
32
- theme
33
- }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(constants.MODAL_WIDTH).map(([size, value]) => `
34
- &[data-size="${size}"] {
35
- width: ${value}px;
36
- }
37
- `), " ", Object.entries(constants.MODAL_PLACEMENT).map(([placement, value]) => `
38
- &[data-placement="${placement}"] {
39
- ${value}
40
- }
41
- `), ";" + (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/Dialog.tsx"],"names":[],"mappings":"AAsC4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from './constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from './types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\n        >\n          <StyledDialog\n            css={dialogCss}\n            onKeyUp={handleKeyUp}\n            onKeyDown={handleFocusTrap}\n            className={className}\n            id={id}\n            data-testid={dataTestId}\n            aria-label={ariaLabel}\n            data-placement={placement}\n            data-size={size}\n            open={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
42
- const Dialog = ({
43
- children,
44
- open,
45
- placement,
46
- onClose,
47
- hideOnClickOutside,
48
- size,
49
- id,
50
- ariaLabel,
51
- className,
52
- "data-testid": dataTestId,
53
- preventBodyScroll,
54
- hideOnEsc,
55
- backdropClassName,
56
- dialogCss,
57
- backdropCss
58
- }) => {
59
- const containerRef = React.useRef(document.createElement("div"));
60
- const dialogRef = React.useRef(null);
61
- const onCloseRef = React.useRef(onClose);
62
- React.useEffect(() => {
63
- const element = containerRef.current;
64
- if (open) {
65
- document.body.appendChild(element);
66
- }
67
- return () => {
68
- if (document.body.contains(element)) {
69
- document.body.removeChild(element);
70
- }
71
- };
72
- }, [open]);
73
- React.useEffect(() => {
74
- onCloseRef.current = onClose;
75
- }, [onClose]);
76
- React.useEffect(() => {
77
- if (open) {
78
- dialogRef.current?.focus();
79
- }
80
- }, [open]);
81
- React.useEffect(() => {
82
- const previousOverflow = document.body.style.overflow;
83
- if (open && preventBodyScroll) {
84
- document.body.style.overflow = "hidden";
85
- }
86
- return () => {
87
- document.body.style.overflow = previousOverflow;
88
- };
89
- }, [preventBodyScroll, open]);
90
- const stopFocus = React.useCallback((event) => {
91
- event.stopPropagation();
92
- }, []);
93
- const stopClick = React.useCallback((event) => {
94
- event.stopPropagation();
95
- }, []);
96
- const handleKeyUp = React.useCallback((event) => {
97
- event.stopPropagation();
98
- if (event.key === "Escape" && hideOnEsc) {
99
- event.preventDefault();
100
- onCloseRef.current();
101
- }
102
- }, [hideOnEsc]);
103
- const handleClose = React.useCallback((event) => {
104
- event.stopPropagation();
105
- if (hideOnClickOutside) {
106
- onCloseRef.current();
107
- } else {
108
- dialogRef.current?.focus();
109
- }
110
- }, [hideOnClickOutside]);
111
- const handleFocusTrap = React.useCallback((event) => {
112
- event.stopPropagation();
113
- if (event.key === "Escape") {
114
- event.preventDefault();
115
- return;
116
- }
117
- const isTabPressed = event.key === "Tab";
118
- if (!isTabPressed) {
119
- return;
120
- }
121
- const focusableEls = dialogRef.current?.querySelectorAll("a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])") ?? [];
122
- if (focusableEls.length === 0) {
123
- event.preventDefault();
124
- }
125
- const firstFocusableEl = focusableEls[0];
126
- const lastFocusableEl = focusableEls[focusableEls.length - 1];
127
- if (event.shiftKey) {
128
- if (document.activeElement === firstFocusableEl || document.activeElement === dialogRef.current) {
129
- lastFocusableEl.focus();
130
- event.preventDefault();
131
- }
132
- } else if (document.activeElement === lastFocusableEl || document.activeElement === dialogRef.current) {
133
- firstFocusableEl.focus();
134
- event.preventDefault();
135
- }
136
- }, []);
137
- const stopCancel = (event) => {
138
- event.preventDefault();
139
- event.stopPropagation();
140
- };
141
- return open ? reactDom.createPortal(/* @__PURE__ */ jsxRuntime.jsx(StyledBackdrop, { "data-open": open, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, children: /* @__PURE__ */ jsxRuntime.jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open, onClick: stopClick, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, children: open ? children : null }) }), containerRef.current) : null;
142
- };
143
- exports.Dialog = Dialog;
144
- exports.StyledDialog = StyledDialog;
@@ -1,11 +0,0 @@
1
- import type { DialogProps, ModalPlacement, ModalSize } from './types';
2
- type StyledDialogProps = {
3
- 'data-size': ModalSize;
4
- 'data-placement': ModalPlacement;
5
- };
6
- export declare const StyledDialog: import("@emotion/styled").StyledComponent<{
7
- theme?: import("@emotion/react").Theme;
8
- as?: React.ElementType;
9
- } & StyledDialogProps, import("react").DetailedHTMLProps<import("react").DialogHTMLAttributes<HTMLDialogElement>, HTMLDialogElement>, {}>;
10
- export declare const Dialog: ({ children, open, placement, onClose, hideOnClickOutside, size, id, ariaLabel, className, "data-testid": dataTestId, preventBodyScroll, hideOnEsc, backdropClassName, dialogCss, backdropCss, }: DialogProps) => import("react").ReactPortal | null;
11
- export {};
@@ -1,142 +0,0 @@
1
- import { jsx } from "@emotion/react/jsx-runtime";
2
- import _styled from "@emotion/styled/base";
3
- import { useRef, useEffect, useCallback } from "react";
4
- import { createPortal } from "react-dom";
5
- import { MODAL_WIDTH, MODAL_PLACEMENT } from "./constants.js";
6
- const StyledBackdrop = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
7
- target: "e1cqen9h1"
8
- } : {
9
- target: "e1cqen9h1",
10
- label: "StyledBackdrop"
11
- })("position:fixed;top:0;right:0;height:0;width:0;overflow:hidden;background-color:", ({
12
- theme
13
- }) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
14
- theme
15
- }) => theme.space["2"], ";overflow:auto;display:flex;bottom:0;left:0;height:100%;width:100%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx"],"names":[],"mappings":"AAY2D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from './constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from './types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\n        >\n          <StyledDialog\n            css={dialogCss}\n            onKeyUp={handleKeyUp}\n            onKeyDown={handleFocusTrap}\n            className={className}\n            id={id}\n            data-testid={dataTestId}\n            aria-label={ariaLabel}\n            data-placement={placement}\n            data-size={size}\n            open={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
16
- const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV === "production" ? {
17
- target: "e1cqen9h0"
18
- } : {
19
- target: "e1cqen9h0",
20
- label: "StyledDialog"
21
- })("background-color:", ({
22
- theme
23
- }) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
24
- theme
25
- }) => theme.radii.default, ";border:0;padding:", ({
26
- theme
27
- }) => theme.space["3"], ";width:", MODAL_WIDTH.medium, "px;box-shadow:", ({
28
- theme
29
- }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(MODAL_WIDTH).map(([size, value]) => `
30
- &[data-size="${size}"] {
31
- width: ${value}px;
32
- }
33
- `), " ", Object.entries(MODAL_PLACEMENT).map(([placement, value]) => `
34
- &[data-placement="${placement}"] {
35
- ${value}
36
- }
37
- `), ";" + (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/Dialog.tsx"],"names":[],"mappings":"AAsC4D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Modal/Dialog.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  FocusEventHandler,\n  KeyboardEventHandler,\n  MouseEventHandler,\n  ReactEventHandler,\n} from 'react'\nimport { useCallback, useEffect, useRef } from 'react'\nimport { createPortal } from 'react-dom'\nimport { MODAL_PLACEMENT, MODAL_WIDTH } from './constants'\nimport type { DialogProps, ModalPlacement, ModalSize } from './types'\n\nconst StyledBackdrop = styled.div<{ 'data-open': boolean }>`\n  position: fixed;\n  top: 0;\n  right: 0;\n  height: 0;\n  width: 0;\n  overflow: hidden;\n  background-color: ${({ theme }) => theme.colors.overlay};\n  z-index: 1;\n\n  &[data-open='true'] {\n    padding: ${({ theme }) => theme.space['2']};\n    overflow: auto;\n    display: flex;\n    bottom: 0;\n    left: 0;\n    height: 100%;\n    width: 100%;\n  }\n`\n\ntype StyledDialogProps = {\n  'data-size': ModalSize\n  'data-placement': ModalPlacement\n}\n\nexport const StyledDialog = styled.dialog<StyledDialogProps>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.overlay};\n  position: relative;\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 0;\n  padding: ${({ theme }) => theme.space['3']};\n  width: ${MODAL_WIDTH.medium}px;\n  box-shadow: ${({ theme }) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`};\n\n\n  ${Object.entries(MODAL_WIDTH).map(\n    ([size, value]) => `\n      &[data-size=\"${size}\"] {\n        width: ${value}px;\n      }\n      `,\n  )}\n\n  ${Object.entries(MODAL_PLACEMENT).map(\n    ([placement, value]) => `\n        &[data-placement=\"${placement}\"] {\n          ${value}\n        }\n        `,\n  )}\n`\n\nexport const Dialog = ({\n  children,\n  open,\n  placement,\n  onClose,\n  hideOnClickOutside,\n  size,\n  id,\n  ariaLabel,\n  className,\n  'data-testid': dataTestId,\n  preventBodyScroll,\n  hideOnEsc,\n  backdropClassName,\n  dialogCss,\n  backdropCss,\n}: DialogProps) => {\n  const containerRef = useRef(document.createElement('div'))\n  const dialogRef = useRef<HTMLDialogElement>(null)\n  const onCloseRef = useRef(onClose)\n\n  // Portal to put the modal in\n  useEffect(() => {\n    const element = containerRef.current\n    if (open) {\n      document.body.appendChild(element)\n    }\n\n    return () => {\n      if (document.body.contains(element)) {\n        document.body.removeChild(element)\n      }\n    }\n  }, [open])\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    if (open) {\n      dialogRef.current?.focus()\n    }\n  }, [open])\n\n  // Handle body scroll\n  useEffect(() => {\n    const previousOverflow = document.body.style.overflow\n\n    if (open && preventBodyScroll) {\n      document.body.style.overflow = 'hidden'\n    }\n\n    return () => {\n      document.body.style.overflow = previousOverflow\n    }\n  }, [preventBodyScroll, open])\n\n  // Stop focus to prevent unexpected body loose focus\n  const stopFocus: FocusEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // Stop click to prevent unexpected dialog close\n  const stopClick: MouseEventHandler = useCallback(event => {\n    event.stopPropagation()\n  }, [])\n\n  // handle key up : used when having inputs in modals - useful for hideOnEsc\n  const handleKeyUp: KeyboardEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (event.key === 'Escape' && hideOnEsc) {\n        event.preventDefault()\n        onCloseRef.current()\n      }\n    },\n    [hideOnEsc],\n  )\n\n  const handleClose: MouseEventHandler = useCallback(\n    event => {\n      event.stopPropagation()\n      if (hideOnClickOutside) {\n        onCloseRef.current()\n      } else {\n        // Because overlay is not focusable we can't handle hideOnEsc properly\n        dialogRef.current?.focus()\n      }\n    },\n    [hideOnClickOutside],\n  )\n\n  // Enable focus trap inside the modal\n  const handleFocusTrap: KeyboardEventHandler = useCallback(event => {\n    event.stopPropagation()\n    if (event.key === 'Escape') {\n      event.preventDefault()\n\n      return\n    }\n    const isTabPressed = event.key === 'Tab'\n\n    if (!isTabPressed) {\n      return\n    }\n\n    const focusableEls =\n      dialogRef.current?.querySelectorAll(\n        'a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])',\n      ) ?? []\n\n    // Handle case when no interactive element are within the modal (including close icon)\n    if (focusableEls.length === 0) {\n      event.preventDefault()\n    }\n\n    const firstFocusableEl = focusableEls[0] as HTMLElement\n    const lastFocusableEl = focusableEls[focusableEls.length - 1] as HTMLElement\n\n    if (event.shiftKey) {\n      if (\n        document.activeElement === firstFocusableEl ||\n        document.activeElement === dialogRef.current\n      ) {\n        lastFocusableEl.focus()\n        event.preventDefault()\n      }\n    } else if (\n      document.activeElement === lastFocusableEl ||\n      document.activeElement === dialogRef.current\n    ) {\n      firstFocusableEl.focus()\n      event.preventDefault()\n    }\n  }, [])\n\n  // Prevent default behaviour on Escape\n  const stopCancel: ReactEventHandler = event => {\n    event.preventDefault()\n    event.stopPropagation()\n  }\n\n  return open\n    ? createPortal(\n        <StyledBackdrop\n          data-open={open}\n          onClick={handleClose}\n          className={backdropClassName}\n          css={backdropCss}\n          data-testid={dataTestId ? `${dataTestId}-backdrop` : undefined}\n          onFocus={stopFocus}\n        >\n          <StyledDialog\n            css={dialogCss}\n            onKeyUp={handleKeyUp}\n            onKeyDown={handleFocusTrap}\n            className={className}\n            id={id}\n            data-testid={dataTestId}\n            aria-label={ariaLabel}\n            data-placement={placement}\n            data-size={size}\n            open={open}\n            onClick={stopClick}\n            onCancel={stopCancel}\n            onClose={stopCancel}\n            aria-modal\n            ref={dialogRef}\n            tabIndex={0}\n          >\n            {open ? children : null}\n          </StyledDialog>\n        </StyledBackdrop>,\n        containerRef.current,\n      )\n    : null\n}\n"]} */"));
38
- const Dialog = ({
39
- children,
40
- open,
41
- placement,
42
- onClose,
43
- hideOnClickOutside,
44
- size,
45
- id,
46
- ariaLabel,
47
- className,
48
- "data-testid": dataTestId,
49
- preventBodyScroll,
50
- hideOnEsc,
51
- backdropClassName,
52
- dialogCss,
53
- backdropCss
54
- }) => {
55
- const containerRef = useRef(document.createElement("div"));
56
- const dialogRef = useRef(null);
57
- const onCloseRef = useRef(onClose);
58
- useEffect(() => {
59
- const element = containerRef.current;
60
- if (open) {
61
- document.body.appendChild(element);
62
- }
63
- return () => {
64
- if (document.body.contains(element)) {
65
- document.body.removeChild(element);
66
- }
67
- };
68
- }, [open]);
69
- useEffect(() => {
70
- onCloseRef.current = onClose;
71
- }, [onClose]);
72
- useEffect(() => {
73
- if (open) {
74
- dialogRef.current?.focus();
75
- }
76
- }, [open]);
77
- useEffect(() => {
78
- const previousOverflow = document.body.style.overflow;
79
- if (open && preventBodyScroll) {
80
- document.body.style.overflow = "hidden";
81
- }
82
- return () => {
83
- document.body.style.overflow = previousOverflow;
84
- };
85
- }, [preventBodyScroll, open]);
86
- const stopFocus = useCallback((event) => {
87
- event.stopPropagation();
88
- }, []);
89
- const stopClick = useCallback((event) => {
90
- event.stopPropagation();
91
- }, []);
92
- const handleKeyUp = useCallback((event) => {
93
- event.stopPropagation();
94
- if (event.key === "Escape" && hideOnEsc) {
95
- event.preventDefault();
96
- onCloseRef.current();
97
- }
98
- }, [hideOnEsc]);
99
- const handleClose = useCallback((event) => {
100
- event.stopPropagation();
101
- if (hideOnClickOutside) {
102
- onCloseRef.current();
103
- } else {
104
- dialogRef.current?.focus();
105
- }
106
- }, [hideOnClickOutside]);
107
- const handleFocusTrap = useCallback((event) => {
108
- event.stopPropagation();
109
- if (event.key === "Escape") {
110
- event.preventDefault();
111
- return;
112
- }
113
- const isTabPressed = event.key === "Tab";
114
- if (!isTabPressed) {
115
- return;
116
- }
117
- const focusableEls = dialogRef.current?.querySelectorAll("a[href]:not([disabled]), button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled])") ?? [];
118
- if (focusableEls.length === 0) {
119
- event.preventDefault();
120
- }
121
- const firstFocusableEl = focusableEls[0];
122
- const lastFocusableEl = focusableEls[focusableEls.length - 1];
123
- if (event.shiftKey) {
124
- if (document.activeElement === firstFocusableEl || document.activeElement === dialogRef.current) {
125
- lastFocusableEl.focus();
126
- event.preventDefault();
127
- }
128
- } else if (document.activeElement === lastFocusableEl || document.activeElement === dialogRef.current) {
129
- firstFocusableEl.focus();
130
- event.preventDefault();
131
- }
132
- }, []);
133
- const stopCancel = (event) => {
134
- event.preventDefault();
135
- event.stopPropagation();
136
- };
137
- return open ? createPortal(/* @__PURE__ */ jsx(StyledBackdrop, { "data-open": open, onClick: handleClose, className: backdropClassName, css: backdropCss, "data-testid": dataTestId ? `${dataTestId}-backdrop` : void 0, onFocus: stopFocus, children: /* @__PURE__ */ jsx(StyledDialog, { css: dialogCss, onKeyUp: handleKeyUp, onKeyDown: handleFocusTrap, className, id, "data-testid": dataTestId, "aria-label": ariaLabel, "data-placement": placement, "data-size": size, open, onClick: stopClick, onCancel: stopCancel, onClose: stopCancel, "aria-modal": true, ref: dialogRef, tabIndex: 0, children: open ? children : null }) }), containerRef.current) : null;
138
- };
139
- export {
140
- Dialog,
141
- StyledDialog
142
- };