@ultraviolet/ui 1.59.1 → 1.60.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/ActionBar/index.cjs +2 -2
- package/dist/components/ActionBar/index.js +2 -2
- package/dist/components/DateInput/index.cjs +6 -4
- package/dist/components/DateInput/index.js +6 -4
- package/dist/components/MenuV2/index.cjs +6 -6
- package/dist/components/MenuV2/index.js +6 -6
- package/dist/components/Modal/Dialog.cjs +4 -4
- package/dist/components/Modal/Dialog.js +4 -4
- package/dist/components/Notification/index.cjs +1 -1
- package/dist/components/Notification/index.js +1 -1
- package/dist/components/SearchInput/index.cjs +21 -4
- package/dist/components/SearchInput/index.d.ts +2 -32
- package/dist/components/SearchInput/index.js +21 -4
- package/dist/components/SearchInput/types.d.ts +1 -1
- package/dist/components/SelectInputV2/Dropdown.cjs +12 -12
- package/dist/components/SelectInputV2/Dropdown.js +12 -12
- package/package.json +1 -1
|
@@ -19,15 +19,15 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
|
|
|
19
19
|
label: "StyledPopup"
|
|
20
20
|
})("background-color:", ({
|
|
21
21
|
theme
|
|
22
|
-
}) => theme.colors.
|
|
22
|
+
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
23
23
|
theme
|
|
24
|
-
}) => theme.shadows.
|
|
24
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
|
|
25
25
|
theme
|
|
26
|
-
}) => theme.colors.
|
|
26
|
+
}) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}width:", ({
|
|
27
27
|
size
|
|
28
28
|
}) => SIZES[size], ";max-width:none;padding:", ({
|
|
29
29
|
theme
|
|
30
|
-
}) => `${theme.space["0.25"]} 0`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBeUNnQyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
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<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\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,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\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      >\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
31
|
const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
32
32
|
target: "e1jn11gg0"
|
|
33
33
|
} : {
|
|
@@ -35,11 +35,11 @@ const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
|
|
|
35
35
|
label: "MenuList"
|
|
36
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
37
|
theme
|
|
38
|
-
}) => theme.colors.
|
|
38
|
+
}) => theme.colors.other.elevation.background.raised, ";color:", ({
|
|
39
39
|
theme
|
|
40
40
|
}) => theme.colors.neutral.text, ";border-radius:", ({
|
|
41
41
|
theme
|
|
42
|
-
}) => theme.radii.default, ";position:relative;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnVWMi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEQ4QiIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9NZW51VjIvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEJ1dHRvbkhUTUxBdHRyaWJ1dGVzLFxuICBDb21wb25lbnRQcm9wcyxcbiAgTW91c2VFdmVudCxcbiAgUmVhY3RFbGVtZW50LFxuICBSZWFjdE5vZGUsXG4gIFJlZixcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQge1xuICBjbG9uZUVsZW1lbnQsXG4gIGZvcndhcmRSZWYsXG4gIGlzVmFsaWRFbGVtZW50LFxuICB1c2VJZCxcbiAgdXNlSW1wZXJhdGl2ZUhhbmRsZSxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbn0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBQb3B1cCB9IGZyb20gJy4uL1BvcHVwJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IEdyb3VwIH0gZnJvbSAnLi9Hcm91cCdcbmltcG9ydCBJdGVtIGZyb20gJy4vSXRlbSdcblxuY29uc3QgU0laRVMgPSB7XG4gIHNtYWxsOiAnMTgwcHgnLFxuICBtZWRpdW06ICcyODBweCcsXG4gIGxhcmdlOiAnMzgwcHgnLFxufVxuXG5leHBvcnQgdHlwZSBEaXNjbG9zdXJlUHJvcHMgPSB7IHZpc2libGU6IGJvb2xlYW4gfVxuXG50eXBlIERpc2Nsb3N1cmVFbGVtZW50ID1cbiAgfCAoKFxuICAgICAgZGlzY2xvc3VyZTogRGlzY2xvc3VyZVByb3BzLFxuICAgICkgPT4gUmVhY3RFbGVtZW50PEJ1dHRvbkhUTUxBdHRyaWJ1dGVzPEhUTUxCdXR0b25FbGVtZW50Pj4pXG4gIHwgKFJlYWN0RWxlbWVudDxCdXR0b25IVE1MQXR0cmlidXRlczxIVE1MQnV0dG9uRWxlbWVudD4+ICYge1xuICAgICAgcmVmPzogUmVmPEhUTUxCdXR0b25FbGVtZW50PlxuICAgIH0pXG5cbmNvbnN0IFN0eWxlZFBvcHVwID0gc3R5bGVkKFBvcHVwLCB7XG4gIHNob3VsZEZvcndhcmRQcm9wOiBwcm9wID0+ICFbJ3NpemUnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPHsgc2l6ZToga2V5b2YgdHlwZW9mIFNJWkVTIH0+YFxuICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+
|
|
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<ComponentProps<typeof Popup>, 'placement' | 'dynamicDomRendering'>\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,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\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      >\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
43
|
const FwdMenu = forwardRef(({
|
|
44
44
|
id,
|
|
45
45
|
ariaLabel = "Menu",
|
|
@@ -16,7 +16,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled__default.default("div", process.e
|
|
|
16
16
|
theme
|
|
17
17
|
}) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
|
|
18
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,
|
|
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
20
|
const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.env.NODE_ENV === "production" ? {
|
|
21
21
|
target: "e1cqen9h0"
|
|
22
22
|
} : {
|
|
@@ -24,13 +24,13 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
24
24
|
label: "StyledDialog"
|
|
25
25
|
})("background-color:", ({
|
|
26
26
|
theme
|
|
27
|
-
}) => theme.colors.
|
|
27
|
+
}) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
|
|
28
28
|
theme
|
|
29
29
|
}) => theme.radii.default, ";border:0;padding:", ({
|
|
30
30
|
theme
|
|
31
31
|
}) => theme.space["3"], ";width:", constants.MODAL_WIDTH.medium, "px;box-shadow:", ({
|
|
32
32
|
theme
|
|
33
|
-
}) => theme.shadows.
|
|
33
|
+
}) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(constants.MODAL_WIDTH).map(([size, value]) => `
|
|
34
34
|
&[data-size="${size}"] {
|
|
35
35
|
width: ${value}px;
|
|
36
36
|
}
|
|
@@ -38,7 +38,7 @@ const StyledDialog = /* @__PURE__ */ _styled__default.default("dialog", process.
|
|
|
38
38
|
&[data-placement="${placement}"] {
|
|
39
39
|
${value}
|
|
40
40
|
}
|
|
41
|
-
`), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL0RpYWxvZy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBc0M0RCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Nb2RhbC9EaWFsb2cudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEZvY3VzRXZlbnRIYW5kbGVyLFxuICBLZXlib2FyZEV2ZW50SGFuZGxlcixcbiAgTW91c2VFdmVudEhhbmRsZXIsXG4gIFJlYWN0RXZlbnRIYW5kbGVyLFxufSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VFZmZlY3QsIHVzZVJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgY3JlYXRlUG9ydGFsIH0gZnJvbSAncmVhY3QtZG9tJ1xuaW1wb3J0IHsgTU9EQUxfUExBQ0VNRU5ULCBNT0RBTF9XSURUSCB9IGZyb20gJy4vY29uc3RhbnRzJ1xuaW1wb3J0IHR5cGUgeyBEaWFsb2dQcm9wcywgTW9kYWxQbGFjZW1lbnQsIE1vZGFsU2l6ZSB9IGZyb20gJy4vdHlwZXMnXG5cbmNvbnN0IFN0eWxlZEJhY2tkcm9wID0gc3R5bGVkLmRpdjx7ICdkYXRhLW9wZW4nOiBib29sZWFuIH0+
|
|
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
42
|
const Dialog = ({
|
|
43
43
|
children,
|
|
44
44
|
open,
|
|
@@ -12,7 +12,7 @@ const StyledBackdrop = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "
|
|
|
12
12
|
theme
|
|
13
13
|
}) => theme.colors.overlay, ";z-index:1;&[data-open='true']{padding:", ({
|
|
14
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,
|
|
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
16
|
const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV === "production" ? {
|
|
17
17
|
target: "e1cqen9h0"
|
|
18
18
|
} : {
|
|
@@ -20,13 +20,13 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
|
|
|
20
20
|
label: "StyledDialog"
|
|
21
21
|
})("background-color:", ({
|
|
22
22
|
theme
|
|
23
|
-
}) => theme.colors.
|
|
23
|
+
}) => theme.colors.other.elevation.background.overlay, ";position:relative;border-radius:", ({
|
|
24
24
|
theme
|
|
25
25
|
}) => theme.radii.default, ";border:0;padding:", ({
|
|
26
26
|
theme
|
|
27
27
|
}) => theme.space["3"], ";width:", MODAL_WIDTH.medium, "px;box-shadow:", ({
|
|
28
28
|
theme
|
|
29
|
-
}) => theme.shadows.
|
|
29
|
+
}) => `${theme.shadows.overlay[0]}, ${theme.shadows.overlay[1]}`, ";", Object.entries(MODAL_WIDTH).map(([size, value]) => `
|
|
30
30
|
&[data-size="${size}"] {
|
|
31
31
|
width: ${value}px;
|
|
32
32
|
}
|
|
@@ -34,7 +34,7 @@ const StyledDialog = /* @__PURE__ */ _styled("dialog", process.env.NODE_ENV ===
|
|
|
34
34
|
&[data-placement="${placement}"] {
|
|
35
35
|
${value}
|
|
36
36
|
}
|
|
37
|
-
`), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01vZGFsL0RpYWxvZy50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBc0M0RCIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Nb2RhbC9EaWFsb2cudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7XG4gIEZvY3VzRXZlbnRIYW5kbGVyLFxuICBLZXlib2FyZEV2ZW50SGFuZGxlcixcbiAgTW91c2VFdmVudEhhbmRsZXIsXG4gIFJlYWN0RXZlbnRIYW5kbGVyLFxufSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUNhbGxiYWNrLCB1c2VFZmZlY3QsIHVzZVJlZiB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgY3JlYXRlUG9ydGFsIH0gZnJvbSAncmVhY3QtZG9tJ1xuaW1wb3J0IHsgTU9EQUxfUExBQ0VNRU5ULCBNT0RBTF9XSURUSCB9IGZyb20gJy4vY29uc3RhbnRzJ1xuaW1wb3J0IHR5cGUgeyBEaWFsb2dQcm9wcywgTW9kYWxQbGFjZW1lbnQsIE1vZGFsU2l6ZSB9IGZyb20gJy4vdHlwZXMnXG5cbmNvbnN0IFN0eWxlZEJhY2tkcm9wID0gc3R5bGVkLmRpdjx7ICdkYXRhLW9wZW4nOiBib29sZWFuIH0+
|
|
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
38
|
const Dialog = ({
|
|
39
39
|
children,
|
|
40
40
|
open,
|
|
@@ -11,7 +11,7 @@ const PREFIX = ".Toastify";
|
|
|
11
11
|
const styles = {
|
|
12
12
|
toast: ({
|
|
13
13
|
theme
|
|
14
|
-
}) => /* @__PURE__ */ react.css("border-radius:", theme.radii.default, ";&", PREFIX, "__toast{background-color:", theme.colors.
|
|
14
|
+
}) => /* @__PURE__ */ react.css("border-radius:", theme.radii.default, ";&", PREFIX, "__toast{background-color:", theme.colors.other.elevation.background.raised, ";color:", theme.colors.neutral.text, ";padding:", theme.space["2"], ";box-shadow:", theme.shadows.raised[0], ",", theme.shadows.raised[1], ";}&", PREFIX, "__toast-container{width:312px;}", PREFIX, "__toast-body{margin:0;padding:0;display:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:toast;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL05vdGlmaWNhdGlvbi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZ0N3QyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Ob3RpZmljYXRpb24vaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgQ2xhc3NOYW1lcywgR2xvYmFsLCBjc3MsIHVzZVRoZW1lIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgVG9hc3RDb250YWluZXIgYXMgQmFzZVRvYXN0Q29udGFpbmVyLFxuICBTbGlkZSxcbiAgdG9hc3QgYXMgYmFzZVRvYXN0LFxufSBmcm9tICdyZWFjdC10b2FzdGlmeSdcbmltcG9ydCB0eXBlIHtcbiAgVGhlbWUgYXMgVGhlbWVUb2FzdGlmeSxcbiAgVG9hc3RPcHRpb25zLFxuICBUeXBlT3B0aW9ucyxcbn0gZnJvbSAncmVhY3QtdG9hc3RpZnknXG5pbXBvcnQgeyBCdXR0b24gfSBmcm9tICcuLi9CdXR0b24nXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9yZWFjdC10b2FzdGlmeS5jc3M/aW5saW5lJ1xuXG5jb25zdCBQUkVGSVggPSAnLlRvYXN0aWZ5J1xuXG50eXBlIFN0eWxlc1Byb3BzID0ge1xuICB0aGVtZTogVGhlbWVcbn1cblxudHlwZSBDbG9zZUJ1dHRvblByb3BzID0ge1xuICBjbG9zZVRvYXN0OiAoZXZlbnQ6IFJlYWN0Lk1vdXNlRXZlbnQ8SFRNTEVsZW1lbnQ+KSA9PiB2b2lkXG4gIHR5cGU6IFR5cGVPcHRpb25zXG4gIGFyaWFMYWJlbD86IHN0cmluZ1xuICB0aGVtZTogVGhlbWVUb2FzdGlmeVxufVxuXG5jb25zdCBzdHlsZXMgPSB7XG4gIHRvYXN0OiAoeyB0aGVtZSB9OiBTdHlsZXNQcm9wcykgPT4gY3NzYFxuICAgIGJvcmRlci1yYWRpdXM6ICR7dGhlbWUucmFkaWkuZGVmYXVsdH07XG5cbiAgICAmJHtQUkVGSVh9X190b2FzdCB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAke3RoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gICAgICBwYWRkaW5nOiAke3RoZW1lLnNwYWNlWycyJ119O1xuICAgICAgYm94LXNoYWRvdzogJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFswXX0sICR7dGhlbWUuc2hhZG93cy5yYWlzZWRbMV19O1xuICAgIH1cblxuICAgICYke1BSRUZJWH1fX3RvYXN0LWNvbnRhaW5lciB7XG4gICAgICB3aWR0aDogMzEycHg7XG4gICAgfVxuXG4gICAgJHtQUkVGSVh9X190b2FzdC1ib2R5IHtcbiAgICAgIG1hcmdpbjogMDtcbiAgICAgIHBhZGRpbmc6IDA7XG4gICAgICBkaXNwbGF5OiBub25lO1xuICAgIH1cbiAgYCxcbn1cblxuY29uc3QgY2xvc2VCdXR0b24gPSAocHJvcHM6IENsb3NlQnV0dG9uUHJvcHMpID0+IChcbiAgPEJ1dHRvblxuICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgaWNvbj1cImNsb3NlXCJcbiAgICBzZW50aW1lbnQ9XCJuZXV0cmFsXCJcbiAgICB2YXJpYW50PVwiZ2hvc3RcIlxuICAgIG9uQ2xpY2s9e3Byb3BzLmNsb3NlVG9hc3R9XG4gICAgc2l6ZT1cInhzbWFsbFwiXG4gIC8+XG4pXG5cbmV4cG9ydCBjb25zdCBub3RpZmljYXRpb24gPSAoXG4gIGNoaWxkcmVuOiAoKHByb3BzOiBDbG9zZUJ1dHRvblByb3BzKSA9PiBSZWFjdE5vZGUpIHwgUmVhY3ROb2RlLFxuICB0aXRsZTogc3RyaW5nLFxuICBpY29uPzogUmVhY3ROb2RlLFxuICBpc0Nsb3NhYmxlPzogYm9vbGVhbixcbiAgY29udGFpbmVySWQ/OiBzdHJpbmcsXG4gIG9wdGlvbnM/OiBUb2FzdE9wdGlvbnMsXG4pID0+XG4gIGJhc2VUb2FzdCgnJywge1xuICAgIC4uLm9wdGlvbnMsXG4gICAgY2xvc2VCdXR0b246IHByb3BzID0+IChcbiAgICAgIDxTdGFjayBkaXJlY3Rpb249XCJyb3dcIiBnYXA9ezJ9PlxuICAgICAgICA8ZGl2PntpY29ufTwvZGl2PlxuICAgICAgICA8U3RhY2sgZGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgPFRleHQgYXM9XCJoM1wiIHZhcmlhbnQ9XCJib2R5U21hbGxTdHJvbmdlclwiPlxuICAgICAgICAgICAge3RpdGxlfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICB7dHlwZW9mIGNoaWxkcmVuID09PSAnZnVuY3Rpb24nID8gY2hpbGRyZW4ocHJvcHMpIDogY2hpbGRyZW59XG4gICAgICAgIDwvU3RhY2s+XG4gICAgICAgIHtpc0Nsb3NhYmxlID8gY2xvc2VCdXR0b24ocHJvcHMpIDogbnVsbH1cbiAgICAgIDwvU3RhY2s+XG4gICAgKSxcbiAgICBjb250YWluZXJJZDogY29udGFpbmVySWQgPz8gJ25vdGlmaWNhdGlvbicsXG4gIH0pXG5cbnR5cGUgTm90aWZpY2F0aW9uQ29udGFpbmVyUHJvcHMgPSB7XG4gIC8qKlxuICAgKiBEZWxheSAoaW4gbXMpIGJlZm9yZSB0aGUgbm90aWZpY2F0aW9uIGF1dG9jbG9zZXMuIFRvIGRpc2FibGUgYXV0b2Nsb3NlLCBzZXQgdG8gZmFsc2VcbiAgICovXG4gIGF1dG9DbG9zZT86IGZhbHNlIHwgbnVtYmVyXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIGRpc3BsYXkgdGhlIG5ld2VzdCB0b2FzdCBvbiB0b3AuXG4gICAqIGBEZWZhdWx0OiBmYWxzZWBcbiAgICovXG4gIG5ld2VzdE9uVG9wPzogYm9vbGVhblxuICAvKipcbiAgICogTGltaXQgdGhlIG51bWJlciBvZiB0b2FzdCBkaXNwbGF5ZWQgYXQgdGhlIHNhbWUgdGltZVxuICAgKi9cbiAgbGltaXQ/OiBudW1iZXJcbiAgLyoqXG4gICAqIFBvc2l0aW9uIG9uIHRoZSBub3RpZmljYXRpb24gY29udGFpbmVyXG4gICAqL1xuICBwb3NpdGlvbj86IFRvYXN0T3B0aW9uc1sncG9zaXRpb24nXVxuICAnZGF0YS10ZXN0aWQnPzogc3RyaW5nXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICAvKipcbiAgICogR2l2ZSBhIHBlcnNvbmFsaXplZCBjb250YWluZXJJZCBpbiBjYXNlIHRoZXJlIGFyZSBtdWx0aXBsZSBub3RpZmljYXRpb25zIHdpdGggZGlmZmVyZW50IHN0eWxlZCB0byBkaXNwbGF5XG4gICAqL1xuICBjb250YWluZXJJZD86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTm90aWZpY2F0aW9uQ29udGFpbmVyID0gKHtcbiAgbmV3ZXN0T25Ub3AsXG4gIGxpbWl0LFxuICBhdXRvQ2xvc2UgPSBmYWxzZSxcbiAgcG9zaXRpb24gPSAndG9wLXJpZ2h0JyxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbiAgY2xhc3NOYW1lLFxuICBjb250YWluZXJJZCA9ICdub3RpZmljYXRpb24nLFxufTogTm90aWZpY2F0aW9uQ29udGFpbmVyUHJvcHMpID0+IHtcbiAgY29uc3QgdGhlbWUgPSB1c2VUaGVtZSgpXG5cbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgPEdsb2JhbCBzdHlsZXM9e3N0eWxlfSAvPlxuICAgICAgPENsYXNzTmFtZXM+XG4gICAgICAgIHsoeyBjc3M6IGxvY2FsQ3NzIH0pID0+IChcbiAgICAgICAgICA8QmFzZVRvYXN0Q29udGFpbmVyXG4gICAgICAgICAgICBkYXRhLXRlc3RpZD17ZGF0YVRlc3RJZH1cbiAgICAgICAgICAgIHRvYXN0Q2xhc3NOYW1lPXtsb2NhbENzcyhzdHlsZXMudG9hc3QoeyB0aGVtZSB9KSl9XG4gICAgICAgICAgICBpY29uPXtmYWxzZX1cbiAgICAgICAgICAgIGF1dG9DbG9zZT17YXV0b0Nsb3NlfVxuICAgICAgICAgICAgbmV3ZXN0T25Ub3A9e25ld2VzdE9uVG9wfVxuICAgICAgICAgICAgbGltaXQ9e2xpbWl0fVxuICAgICAgICAgICAgcG9zaXRpb249e3Bvc2l0aW9ufVxuICAgICAgICAgICAgaGlkZVByb2dyZXNzQmFyXG4gICAgICAgICAgICBkcmFnZ2FibGU9e2ZhbHNlfVxuICAgICAgICAgICAgdHJhbnNpdGlvbj17U2xpZGV9XG4gICAgICAgICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgICAgICAgIGNvbnRhaW5lcklkPXtjb250YWluZXJJZH1cbiAgICAgICAgICAvPlxuICAgICAgICApfVxuICAgICAgPC9DbGFzc05hbWVzPlxuICAgIDwvPlxuICApXG59XG4iXX0= */")
|
|
15
15
|
};
|
|
16
16
|
const closeButton = (props) => /* @__PURE__ */ jsxRuntime.jsx(index$2.Button, { "aria-label": "close", icon: "close", sentiment: "neutral", variant: "ghost", onClick: props.closeToast, size: "xsmall" });
|
|
17
17
|
const notification = (children, title, icon, isClosable, containerId, options) => reactToastify.toast("", {
|
|
@@ -9,7 +9,7 @@ const PREFIX = ".Toastify";
|
|
|
9
9
|
const styles = {
|
|
10
10
|
toast: ({
|
|
11
11
|
theme
|
|
12
|
-
}) => /* @__PURE__ */ css("border-radius:", theme.radii.default, ";&", PREFIX, "__toast{background-color:", theme.colors.
|
|
12
|
+
}) => /* @__PURE__ */ css("border-radius:", theme.radii.default, ";&", PREFIX, "__toast{background-color:", theme.colors.other.elevation.background.raised, ";color:", theme.colors.neutral.text, ";padding:", theme.space["2"], ";box-shadow:", theme.shadows.raised[0], ",", theme.shadows.raised[1], ";}&", PREFIX, "__toast-container{width:312px;}", PREFIX, "__toast-body{margin:0;padding:0;display:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:toast;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL05vdGlmaWNhdGlvbi9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZ0N3QyIsImZpbGUiOiIvaG9tZS9ydW5uZXIvd29yay91bHRyYXZpb2xldC91bHRyYXZpb2xldC9wYWNrYWdlcy91aS9zcmMvY29tcG9uZW50cy9Ob3RpZmljYXRpb24vaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ0BlbW90aW9uL3JlYWN0J1xuaW1wb3J0IHsgQ2xhc3NOYW1lcywgR2xvYmFsLCBjc3MsIHVzZVRoZW1lIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgVG9hc3RDb250YWluZXIgYXMgQmFzZVRvYXN0Q29udGFpbmVyLFxuICBTbGlkZSxcbiAgdG9hc3QgYXMgYmFzZVRvYXN0LFxufSBmcm9tICdyZWFjdC10b2FzdGlmeSdcbmltcG9ydCB0eXBlIHtcbiAgVGhlbWUgYXMgVGhlbWVUb2FzdGlmeSxcbiAgVG9hc3RPcHRpb25zLFxuICBUeXBlT3B0aW9ucyxcbn0gZnJvbSAncmVhY3QtdG9hc3RpZnknXG5pbXBvcnQgeyBCdXR0b24gfSBmcm9tICcuLi9CdXR0b24nXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9yZWFjdC10b2FzdGlmeS5jc3M/aW5saW5lJ1xuXG5jb25zdCBQUkVGSVggPSAnLlRvYXN0aWZ5J1xuXG50eXBlIFN0eWxlc1Byb3BzID0ge1xuICB0aGVtZTogVGhlbWVcbn1cblxudHlwZSBDbG9zZUJ1dHRvblByb3BzID0ge1xuICBjbG9zZVRvYXN0OiAoZXZlbnQ6IFJlYWN0Lk1vdXNlRXZlbnQ8SFRNTEVsZW1lbnQ+KSA9PiB2b2lkXG4gIHR5cGU6IFR5cGVPcHRpb25zXG4gIGFyaWFMYWJlbD86IHN0cmluZ1xuICB0aGVtZTogVGhlbWVUb2FzdGlmeVxufVxuXG5jb25zdCBzdHlsZXMgPSB7XG4gIHRvYXN0OiAoeyB0aGVtZSB9OiBTdHlsZXNQcm9wcykgPT4gY3NzYFxuICAgIGJvcmRlci1yYWRpdXM6ICR7dGhlbWUucmFkaWkuZGVmYXVsdH07XG5cbiAgICAmJHtQUkVGSVh9X190b2FzdCB7XG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiAke3RoZW1lLmNvbG9ycy5vdGhlci5lbGV2YXRpb24uYmFja2dyb3VuZC5yYWlzZWR9O1xuICAgICAgY29sb3I6ICR7dGhlbWUuY29sb3JzLm5ldXRyYWwudGV4dH07XG4gICAgICBwYWRkaW5nOiAke3RoZW1lLnNwYWNlWycyJ119O1xuICAgICAgYm94LXNoYWRvdzogJHt0aGVtZS5zaGFkb3dzLnJhaXNlZFswXX0sICR7dGhlbWUuc2hhZG93cy5yYWlzZWRbMV19O1xuICAgIH1cblxuICAgICYke1BSRUZJWH1fX3RvYXN0LWNvbnRhaW5lciB7XG4gICAgICB3aWR0aDogMzEycHg7XG4gICAgfVxuXG4gICAgJHtQUkVGSVh9X190b2FzdC1ib2R5IHtcbiAgICAgIG1hcmdpbjogMDtcbiAgICAgIHBhZGRpbmc6IDA7XG4gICAgICBkaXNwbGF5OiBub25lO1xuICAgIH1cbiAgYCxcbn1cblxuY29uc3QgY2xvc2VCdXR0b24gPSAocHJvcHM6IENsb3NlQnV0dG9uUHJvcHMpID0+IChcbiAgPEJ1dHRvblxuICAgIGFyaWEtbGFiZWw9XCJjbG9zZVwiXG4gICAgaWNvbj1cImNsb3NlXCJcbiAgICBzZW50aW1lbnQ9XCJuZXV0cmFsXCJcbiAgICB2YXJpYW50PVwiZ2hvc3RcIlxuICAgIG9uQ2xpY2s9e3Byb3BzLmNsb3NlVG9hc3R9XG4gICAgc2l6ZT1cInhzbWFsbFwiXG4gIC8+XG4pXG5cbmV4cG9ydCBjb25zdCBub3RpZmljYXRpb24gPSAoXG4gIGNoaWxkcmVuOiAoKHByb3BzOiBDbG9zZUJ1dHRvblByb3BzKSA9PiBSZWFjdE5vZGUpIHwgUmVhY3ROb2RlLFxuICB0aXRsZTogc3RyaW5nLFxuICBpY29uPzogUmVhY3ROb2RlLFxuICBpc0Nsb3NhYmxlPzogYm9vbGVhbixcbiAgY29udGFpbmVySWQ/OiBzdHJpbmcsXG4gIG9wdGlvbnM/OiBUb2FzdE9wdGlvbnMsXG4pID0+XG4gIGJhc2VUb2FzdCgnJywge1xuICAgIC4uLm9wdGlvbnMsXG4gICAgY2xvc2VCdXR0b246IHByb3BzID0+IChcbiAgICAgIDxTdGFjayBkaXJlY3Rpb249XCJyb3dcIiBnYXA9ezJ9PlxuICAgICAgICA8ZGl2PntpY29ufTwvZGl2PlxuICAgICAgICA8U3RhY2sgZGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgPFRleHQgYXM9XCJoM1wiIHZhcmlhbnQ9XCJib2R5U21hbGxTdHJvbmdlclwiPlxuICAgICAgICAgICAge3RpdGxlfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICB7dHlwZW9mIGNoaWxkcmVuID09PSAnZnVuY3Rpb24nID8gY2hpbGRyZW4ocHJvcHMpIDogY2hpbGRyZW59XG4gICAgICAgIDwvU3RhY2s+XG4gICAgICAgIHtpc0Nsb3NhYmxlID8gY2xvc2VCdXR0b24ocHJvcHMpIDogbnVsbH1cbiAgICAgIDwvU3RhY2s+XG4gICAgKSxcbiAgICBjb250YWluZXJJZDogY29udGFpbmVySWQgPz8gJ25vdGlmaWNhdGlvbicsXG4gIH0pXG5cbnR5cGUgTm90aWZpY2F0aW9uQ29udGFpbmVyUHJvcHMgPSB7XG4gIC8qKlxuICAgKiBEZWxheSAoaW4gbXMpIGJlZm9yZSB0aGUgbm90aWZpY2F0aW9uIGF1dG9jbG9zZXMuIFRvIGRpc2FibGUgYXV0b2Nsb3NlLCBzZXQgdG8gZmFsc2VcbiAgICovXG4gIGF1dG9DbG9zZT86IGZhbHNlIHwgbnVtYmVyXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIGRpc3BsYXkgdGhlIG5ld2VzdCB0b2FzdCBvbiB0b3AuXG4gICAqIGBEZWZhdWx0OiBmYWxzZWBcbiAgICovXG4gIG5ld2VzdE9uVG9wPzogYm9vbGVhblxuICAvKipcbiAgICogTGltaXQgdGhlIG51bWJlciBvZiB0b2FzdCBkaXNwbGF5ZWQgYXQgdGhlIHNhbWUgdGltZVxuICAgKi9cbiAgbGltaXQ/OiBudW1iZXJcbiAgLyoqXG4gICAqIFBvc2l0aW9uIG9uIHRoZSBub3RpZmljYXRpb24gY29udGFpbmVyXG4gICAqL1xuICBwb3NpdGlvbj86IFRvYXN0T3B0aW9uc1sncG9zaXRpb24nXVxuICAnZGF0YS10ZXN0aWQnPzogc3RyaW5nXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICAvKipcbiAgICogR2l2ZSBhIHBlcnNvbmFsaXplZCBjb250YWluZXJJZCBpbiBjYXNlIHRoZXJlIGFyZSBtdWx0aXBsZSBub3RpZmljYXRpb25zIHdpdGggZGlmZmVyZW50IHN0eWxlZCB0byBkaXNwbGF5XG4gICAqL1xuICBjb250YWluZXJJZD86IHN0cmluZ1xufVxuXG5leHBvcnQgY29uc3QgTm90aWZpY2F0aW9uQ29udGFpbmVyID0gKHtcbiAgbmV3ZXN0T25Ub3AsXG4gIGxpbWl0LFxuICBhdXRvQ2xvc2UgPSBmYWxzZSxcbiAgcG9zaXRpb24gPSAndG9wLXJpZ2h0JyxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbiAgY2xhc3NOYW1lLFxuICBjb250YWluZXJJZCA9ICdub3RpZmljYXRpb24nLFxufTogTm90aWZpY2F0aW9uQ29udGFpbmVyUHJvcHMpID0+IHtcbiAgY29uc3QgdGhlbWUgPSB1c2VUaGVtZSgpXG5cbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgPEdsb2JhbCBzdHlsZXM9e3N0eWxlfSAvPlxuICAgICAgPENsYXNzTmFtZXM+XG4gICAgICAgIHsoeyBjc3M6IGxvY2FsQ3NzIH0pID0+IChcbiAgICAgICAgICA8QmFzZVRvYXN0Q29udGFpbmVyXG4gICAgICAgICAgICBkYXRhLXRlc3RpZD17ZGF0YVRlc3RJZH1cbiAgICAgICAgICAgIHRvYXN0Q2xhc3NOYW1lPXtsb2NhbENzcyhzdHlsZXMudG9hc3QoeyB0aGVtZSB9KSl9XG4gICAgICAgICAgICBpY29uPXtmYWxzZX1cbiAgICAgICAgICAgIGF1dG9DbG9zZT17YXV0b0Nsb3NlfVxuICAgICAgICAgICAgbmV3ZXN0T25Ub3A9e25ld2VzdE9uVG9wfVxuICAgICAgICAgICAgbGltaXQ9e2xpbWl0fVxuICAgICAgICAgICAgcG9zaXRpb249e3Bvc2l0aW9ufVxuICAgICAgICAgICAgaGlkZVByb2dyZXNzQmFyXG4gICAgICAgICAgICBkcmFnZ2FibGU9e2ZhbHNlfVxuICAgICAgICAgICAgdHJhbnNpdGlvbj17U2xpZGV9XG4gICAgICAgICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICAgICAgICAgIGNvbnRhaW5lcklkPXtjb250YWluZXJJZH1cbiAgICAgICAgICAvPlxuICAgICAgICApfVxuICAgICAgPC9DbGFzc05hbWVzPlxuICAgIDwvPlxuICApXG59XG4iXX0= */")
|
|
13
13
|
};
|
|
14
14
|
const closeButton = (props) => /* @__PURE__ */ jsx(Button, { "aria-label": "close", icon: "close", sentiment: "neutral", variant: "ghost", onClick: props.closeToast, size: "xsmall" });
|
|
15
15
|
const notification = (children, title, icon, isClosable, containerId, options) => toast("", {
|
|
@@ -21,16 +21,17 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
|
|
|
21
21
|
theme
|
|
22
22
|
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
23
23
|
theme
|
|
24
|
-
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
24
|
+
}) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
25
25
|
const StyledTextInputV2 = /* @__PURE__ */ _styled__default.default(index$1.TextInputV2, process.env.NODE_ENV === "production" ? {
|
|
26
26
|
target: "eefux4u0"
|
|
27
27
|
} : {
|
|
28
28
|
target: "eefux4u0",
|
|
29
29
|
label: "StyledTextInputV2"
|
|
30
|
-
})(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
30
|
+
})(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 610px;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
|
|
31
31
|
const SearchInput = React.forwardRef(({
|
|
32
32
|
placeholder,
|
|
33
33
|
label,
|
|
34
|
+
labelDescription,
|
|
34
35
|
loading,
|
|
35
36
|
size,
|
|
36
37
|
popupPlacement,
|
|
@@ -42,7 +43,23 @@ const SearchInput = React.forwardRef(({
|
|
|
42
43
|
shortcut = false,
|
|
43
44
|
error,
|
|
44
45
|
disabled,
|
|
45
|
-
className
|
|
46
|
+
className,
|
|
47
|
+
minLength,
|
|
48
|
+
maxLength,
|
|
49
|
+
tooltip,
|
|
50
|
+
onFocus,
|
|
51
|
+
onBlur,
|
|
52
|
+
name,
|
|
53
|
+
id,
|
|
54
|
+
"aria-live": ariaLive,
|
|
55
|
+
"aria-atomic": ariaAtomic,
|
|
56
|
+
"aria-labelledby": ariaLabelledby,
|
|
57
|
+
readOnly,
|
|
58
|
+
required,
|
|
59
|
+
autoFocus,
|
|
60
|
+
autoComplete,
|
|
61
|
+
onKeyDown,
|
|
62
|
+
role
|
|
46
63
|
}, ref) => {
|
|
47
64
|
const focusedLinkIndex = React.useRef(0);
|
|
48
65
|
const popupRef = React.useRef(null);
|
|
@@ -168,6 +185,6 @@ const SearchInput = React.forwardRef(({
|
|
|
168
185
|
}, [isMacOS, shortcut]);
|
|
169
186
|
return /* @__PURE__ */ jsxRuntime.jsx("div", { style: {
|
|
170
187
|
width: "100%"
|
|
171
|
-
}, children: /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsxRuntime.jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsxRuntime.jsx(KeyGroup.KeyGroup, { disabled, keys }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled, className }) }) });
|
|
188
|
+
}, children: /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsxRuntime.jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsxRuntime.jsx(KeyGroup.KeyGroup, { disabled, keys }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled, className, "aria-atomic": ariaAtomic, autoComplete, "aria-labelledby": ariaLabelledby, "aria-live": ariaLive, id, name, readOnly, required, autoFocus, maxLength, minLength, onBlur, onFocus, onKeyDown, role, tooltip, labelDescription }) }) });
|
|
172
189
|
});
|
|
173
190
|
exports.SearchInput = SearchInput;
|