@ultraviolet/ui 3.0.0-beta.13 → 3.0.0-beta.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (75) hide show
  1. package/dist/components/Alert/styles.css.cjs +1 -0
  2. package/dist/components/Alert/styles.css.js +1 -0
  3. package/dist/components/Breadcrumbs/components/Item.cjs +25 -42
  4. package/dist/components/Breadcrumbs/components/Item.d.ts +3 -2
  5. package/dist/components/Breadcrumbs/components/Item.js +25 -40
  6. package/dist/components/Breadcrumbs/components/styles.css.cjs +16 -0
  7. package/dist/components/Breadcrumbs/components/styles.css.d.ts +12 -0
  8. package/dist/components/Breadcrumbs/components/styles.css.js +16 -0
  9. package/dist/components/Breadcrumbs/index.cjs +2 -19
  10. package/dist/components/Breadcrumbs/index.js +2 -17
  11. package/dist/components/Breadcrumbs/styles.css.cjs +5 -0
  12. package/dist/components/Breadcrumbs/styles.css.d.ts +1 -0
  13. package/dist/components/Breadcrumbs/styles.css.js +5 -0
  14. package/dist/components/Button/index.cjs +3 -2
  15. package/dist/components/Button/index.d.ts +2 -1
  16. package/dist/components/Button/index.js +3 -2
  17. package/dist/components/Card/index.cjs +9 -43
  18. package/dist/components/Card/index.js +4 -36
  19. package/dist/components/Card/styles.css.cjs +7 -0
  20. package/dist/components/Card/styles.css.d.ts +2 -0
  21. package/dist/components/Card/styles.css.js +7 -0
  22. package/dist/components/Checkbox/index.cjs +7 -205
  23. package/dist/components/Checkbox/index.d.ts +0 -20
  24. package/dist/components/Checkbox/index.js +8 -204
  25. package/dist/components/Checkbox/styles.css.cjs +17 -0
  26. package/dist/components/Checkbox/styles.css.d.ts +7 -0
  27. package/dist/components/Checkbox/styles.css.js +17 -0
  28. package/dist/components/CheckboxGroup/index.cjs +3 -34
  29. package/dist/components/CheckboxGroup/index.js +3 -32
  30. package/dist/components/CheckboxGroup/styles.css.cjs +7 -0
  31. package/dist/components/CheckboxGroup/styles.css.d.ts +2 -0
  32. package/dist/components/CheckboxGroup/styles.css.js +7 -0
  33. package/dist/components/Chip/index.d.ts +1 -1
  34. package/dist/components/Chip/styles.css.cjs +0 -1
  35. package/dist/components/Chip/styles.css.js +0 -1
  36. package/dist/components/Dialog/index.d.ts +1 -0
  37. package/dist/components/Drawer/index.cjs +18 -13
  38. package/dist/components/Drawer/index.js +18 -13
  39. package/dist/components/ExpandableCard/index.cjs +13 -97
  40. package/dist/components/ExpandableCard/index.js +9 -91
  41. package/dist/components/ExpandableCard/styles.css.cjs +17 -0
  42. package/dist/components/ExpandableCard/styles.css.d.ts +7 -0
  43. package/dist/components/ExpandableCard/styles.css.js +17 -0
  44. package/dist/components/Popup/styles.css.cjs +1 -0
  45. package/dist/components/Popup/styles.css.js +1 -0
  46. package/dist/components/Popup/variables.css.cjs +0 -1
  47. package/dist/components/Popup/variables.css.js +0 -1
  48. package/dist/components/RadioGroup/index.cjs +2 -20
  49. package/dist/components/RadioGroup/index.js +2 -18
  50. package/dist/components/RadioGroup/styles.css.cjs +5 -0
  51. package/dist/components/RadioGroup/styles.css.d.ts +1 -0
  52. package/dist/components/RadioGroup/styles.css.js +5 -0
  53. package/dist/components/SelectInput/SelectBar.cjs +9 -9
  54. package/dist/components/SelectInput/SelectBar.js +9 -9
  55. package/dist/components/SelectableCard/index.cjs +19 -18
  56. package/dist/components/SelectableCard/index.js +16 -15
  57. package/dist/components/Tag/index.cjs +16 -104
  58. package/dist/components/Tag/index.d.ts +2 -2
  59. package/dist/components/Tag/index.js +12 -99
  60. package/dist/components/Tag/styles.css.cjs +10 -0
  61. package/dist/components/Tag/styles.css.d.ts +34 -0
  62. package/dist/components/Tag/styles.css.js +10 -0
  63. package/dist/components/Text/style.css.cjs +1 -0
  64. package/dist/components/Text/style.css.js +1 -0
  65. package/dist/components/Text/variables.css.cjs +0 -1
  66. package/dist/components/Text/variables.css.js +0 -1
  67. package/dist/components/ToggleGroup/index.cjs +2 -20
  68. package/dist/components/ToggleGroup/index.js +2 -18
  69. package/dist/components/ToggleGroup/styles.css.cjs +5 -0
  70. package/dist/components/ToggleGroup/styles.css.d.ts +1 -0
  71. package/dist/components/ToggleGroup/styles.css.js +5 -0
  72. package/dist/ui.css +1 -1
  73. package/package.json +2 -2
  74. package/dist/components/Breadcrumbs/constants.cjs +0 -4
  75. package/dist/components/Breadcrumbs/constants.js +0 -4
@@ -2,97 +2,13 @@
2
2
  "use strict";
3
3
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
- const _styled = require("@emotion/styled/base");
6
5
  const react$1 = require("@emotion/react");
7
6
  const Icon = require("@ultraviolet/icons");
8
7
  const react = require("react");
9
- const index$1 = require("../Stack/index.cjs");
10
- const index = require("../Tooltip/index.cjs");
8
+ const index = require("../Stack/index.cjs");
9
+ const index$1 = require("../Tooltip/index.cjs");
11
10
  const Title = require("./components/Title.cjs");
12
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
13
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
14
- const StyledArrowIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowDownIcon, process.env.NODE_ENV === "production" ? {
15
- target: "e143cpd66"
16
- } : {
17
- target: "e143cpd66",
18
- label: "StyledArrowIcon"
19
- })(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/ExpandableCard/index.tsx"],"names":[],"mappings":"AAkB6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */");
20
- const DropableArea = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
21
- target: "e143cpd65"
22
- } : {
23
- target: "e143cpd65",
24
- label: "DropableArea"
25
- })("height:", ({
26
- theme
27
- }) => theme.space["3"], ";border-bottom:2px solid;border-color:transparent;padding:", ({
28
- theme
29
- }) => theme.space["0.5"], ' 0;width:100%;bottom:-5px;position:absolute;&[data-first="true"]{top:-', ({
30
- theme
31
- }) => theme.space["3"], ";}&::after{content:'';left:0;bottom:-4px;height:0px;width:0px;border:3px solid;border-color:inherit;border-radius:", ({
32
- theme
33
- }) => theme.radii.circle, ";display:flex;margin-top:-", ({
34
- theme
35
- }) => theme.space["1"], ";margin-left:-", ({
36
- theme
37
- }) => theme.space["0.25"], ";position:absolute;}" + (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/ExpandableCard/index.tsx"],"names":[],"mappings":"AAoB+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
38
- const StyledSummary = /* @__PURE__ */ _styled__default.default("summary", process.env.NODE_ENV === "production" ? {
39
- target: "e143cpd64"
40
- } : {
41
- target: "e143cpd64",
42
- label: "StyledSummary"
43
- })("display:flex;flex-direction:row;align-items:center;gap:", ({
44
- theme
45
- }) => theme.space["2"], ";padding:", ({
46
- theme
47
- }) => theme.space["3"], ';list-style-type:none;cursor:pointer;&[data-disabled="true"]{background:', ({
48
- theme
49
- }) => theme.colors.neutral.backgroundWeak, ";border-radius:", ({
50
- theme
51
- }) => theme.radii.default, ";cursor:not-allowed;}" + (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/ExpandableCard/index.tsx"],"names":[],"mappings":"AAiDoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
52
- const StyledContent = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
53
- target: "e143cpd63"
54
- } : {
55
- target: "e143cpd63",
56
- label: "StyledContent"
57
- })("border-top:1px solid ", ({
58
- theme
59
- }) => theme.colors.neutral.border, ";padding:", ({
60
- theme
61
- }) => theme.space["3"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx"],"names":[],"mappings":"AAiEgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
62
- const StyledDetails = /* @__PURE__ */ _styled__default.default("details", process.env.NODE_ENV === "production" ? {
63
- target: "e143cpd62"
64
- } : {
65
- target: "e143cpd62",
66
- label: "StyledDetails"
67
- })("border:1px solid ", ({
68
- theme
69
- }) => theme.colors.neutral.border, ";border-radius:", ({
70
- theme
71
- }) => theme.radii.default, ";width:100%;transition:border-color 0.2s ease-in-out;&[open]{border-color:", ({
72
- theme
73
- }) => theme.colors.primary.border, ";&>", StyledContent, "{border-color:", ({
74
- theme
75
- }) => theme.colors.primary.border, ";}", StyledArrowIcon, '{transform:rotate(180deg);}}&[data-clicking="true"]{box-shadow:', ({
76
- theme
77
- }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";border-color:", ({
78
- theme
79
- }) => theme.colors.primary.border, ";}" + (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/ExpandableCard/index.tsx"],"names":[],"mappings":"AAuEoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
80
- const StyledStack = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
81
- target: "e143cpd61"
82
- } : {
83
- target: "e143cpd61",
84
- label: "StyledStack"
85
- })("position:relative;&:hover>", StyledDetails, "{border-color:", ({
86
- theme
87
- }) => theme.colors.primary.border, ";}" + (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/ExpandableCard/index.tsx"],"names":[],"mappings":"AA+FiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
88
- const DragIconContainer = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
89
- target: "e143cpd60"
90
- } : {
91
- target: "e143cpd60",
92
- label: "DragIconContainer"
93
- })("height:100%;opacity:0;transition:opacity 0.2s ease-in-out;cursor:grab;padding-top:calc(", ({
94
- theme
95
- }) => theme.space["3"], ' + 2px);&[data-visible="true"]{opacity:1;}&:focus-within{opacity:1;}&:active{cursor:grabbing;opacity:1;}' + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx"],"names":[],"mappings":"AAsGuC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/ExpandableCard/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, DragIcon } from '@ultraviolet/icons'\nimport type {\n  DetailsHTMLAttributes,\n  DragEvent,\n  ForwardedRef,\n  KeyboardEvent,\n  ReactNode,\n} from 'react'\nimport { forwardRef, useCallback, useRef, useState } from 'react'\nimport type { XOR } from '../../types'\nimport { Stack } from '../Stack'\nimport { Tooltip } from '../Tooltip'\nimport { ExpandableCardTitle } from './components/Title'\n\nconst StyledArrowIcon = styled(ArrowDownIcon)``\n\nconst DropableArea = styled.div`\n  height: ${({ theme }) => theme.space['3']};\n  border-bottom: 2px solid;\n  border-color: transparent;\n  padding: ${({ theme }) => theme.space['0.5']} 0;\n  width: 100%;\n  bottom: -5px;\n  position: absolute;\n\n  &[data-first=\"true\"] {\n    top: -${({ theme }) => theme.space['3']};\n  }\n\n  &::after {\n    content: '';\n    left: 0;\n    bottom: -4px;\n    height: 0px;\n    width: 0px;\n    border: 3px solid;\n    border-color: inherit;\n    border-radius: ${({ theme }) => theme.radii.circle};\n    display: flex;\n    margin-top:  -${({ theme }) => theme.space['1']};\n    margin-left: -${({ theme }) => theme.space['0.25']};\n    position: absolute;\n  }\n`\n\nconst StyledSummary = styled.summary`\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  gap:  ${({ theme }) => theme.space['2']};\n  padding: ${({ theme }) => theme.space['3']};\n  list-style-type: none;\n\n  cursor: pointer;\n  &[data-disabled=\"true\"] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-radius: ${({ theme }) => theme.radii.default};\n    cursor: not-allowed;\n  }\n`\n\nconst StyledContent = styled.div`\n  border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  padding: ${({ theme }) => theme.space['3']};\n\n`\n\nconst StyledDetails = styled.details`\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n  width: 100%;\n  transition: border-color 0.2s ease-in-out;\n\n  &[open] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n    & > ${StyledContent} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n   ${StyledArrowIcon} {\n      transform: rotate(180deg);\n    }\n  }\n\n  &[data-clicking=\"true\"] {\n    box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n    border-color: ${({ theme }) => theme.colors.primary.border};\n\n  }\n`\nconst StyledStack = styled(Stack)`\n  position: relative;\n    &:hover > ${StyledDetails} {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n  `\n\nconst DragIconContainer = styled(Stack)`\n  height: 100%;\n  opacity: 0;\n  transition: opacity 0.2s ease-in-out;\n  cursor: grab;\n  padding-top: calc(${({ theme }) => theme.space['3']} + 2px);\n\n  &[data-visible=\"true\"] {\n    opacity: 1;\n  }\n  &:focus-within {\n    opacity: 1;\n  }\n\n  &:active {\n    cursor: grabbing;\n    opacity: 1;\n  }\n`\n\nexport const EXPANDABLE_CARD_SIZE = ['medium', 'large'] as const\n\ntype DraggableListType = { value?: string }\ntype ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number]\n\ntype DraggableProps =\n  | {\n      draggable: true\n      value: string\n      draggableTooltip?: string\n      onDrop?: (newValue: string, oldValue: string) => void\n      /**\n       * Index of the card in the  of draggable cards\n       */\n      index: number\n      /**\n       * You this prop to handle drag and drop with the keyboard (accessibility)\n       */\n      onKeyDown?: (event: KeyboardEvent<HTMLDivElement>) => void\n    }\n  | {\n      draggable?: never\n      value?: never\n      draggableTooltip?: never\n      onDrop?: never\n      index?: never\n      onKeyDown?: never\n    }\ntype CommonProps = {\n  header: ReactNode\n  size?: ExpandableCardSize\n  name?: string\n  children: ReactNode\n  disabled?: boolean\n  'data-testid'?: string\n  className?: string\n  /** Uncontrolled but open by default */\n  open?: DetailsHTMLAttributes<HTMLDetailsElement>['open']\n} & DraggableProps\n\ntype ExpandableCardProps = XOR<\n  [CommonProps & { expanded: boolean; onToggleExpand: () => void }, CommonProps]\n>\nconst BaseExpandableCard = forwardRef(\n  (\n    {\n      header,\n      name,\n      size = 'medium',\n      children,\n      disabled,\n      expanded,\n      onToggleExpand,\n      className,\n      draggable,\n      value,\n      draggableTooltip = 'Click and drag to move',\n      onDrop,\n      index,\n      onKeyDown,\n      'data-testid': dataTestId,\n      open,\n    }: ExpandableCardProps,\n    ref: ForwardedRef<HTMLDetailsElement>,\n  ) => {\n    const headerRef = useRef<HTMLDivElement>(null)\n    const containerRef = useRef<HTMLDivElement>(null)\n    const draggableRef = useRef<HTMLDivElement>(null)\n    const draggableFirstRef = useRef<HTMLDivElement>(null)\n    const [isHovered, setIsHovered] = useState(false)\n    const [clicking, setClicking] = useState(false)\n\n    const theme = useTheme()\n\n    const handleMouseEnter = () => {\n      setIsHovered(true)\n    }\n    const handleMouseLeave = () => {\n      setIsHovered(false)\n    }\n\n    const onDragStart = useCallback(\n      (event: DragEvent<HTMLDivElement>) => {\n        event.dataTransfer.setData('text/plain', JSON.stringify({ value }))\n      },\n      [value],\n    )\n\n    const onDragEnd = useCallback(() => setClicking(false), [])\n\n    const onDrag = useCallback(\n      (\n        event: DragEvent<HTMLDivElement>,\n        borderColor: string,\n        isFirst?: boolean,\n      ) => {\n        const refElement = isFirst ? draggableFirstRef : draggableRef\n        event.preventDefault()\n        if (refElement.current) {\n          refElement.current.style.borderColor = borderColor\n        }\n      },\n      [],\n    )\n\n    const handleDrop = useCallback(\n      (event: DragEvent<HTMLDivElement>, isFirst?: boolean) => {\n        event.preventDefault()\n        if (draggableRef.current) {\n          draggableRef.current.style.borderColor = 'transparent'\n        }\n\n        if (event?.dataTransfer) {\n          const data = JSON.parse(\n            event.dataTransfer.getData('text'),\n          ) as DraggableListType\n\n          onDrop?.(isFirst ? '' : value, data.value ?? '')\n        }\n      },\n      [onDrop, value],\n    )\n\n    return (\n      <StyledStack\n        data-draggable={draggable}\n        data-name={name}\n        data-value={value}\n        direction=\"row\"\n        draggable={draggable}\n        gap={2}\n        onDragEnd={onDragEnd}\n        onDragStart={onDragStart}\n        onMouseEnter={handleMouseEnter}\n        onMouseLeave={handleMouseLeave}\n        ref={containerRef}\n        width=\"100%\"\n      >\n        {draggable ? (\n          <DragIconContainer\n            data-testid={`draggable-icon-${value}`}\n            data-visible={isHovered}\n            justifyContent=\"center\"\n            onKeyDown={(event: KeyboardEvent<HTMLDivElement>) => {\n              if (event.key === 'Enter' || event.key === ' ') {\n                event.preventDefault()\n                if (clicking && containerRef.current) {\n                  setClicking(false)\n                  setIsHovered(false)\n                } else if (containerRef.current) {\n                  setClicking(true)\n                  setIsHovered(true)\n                }\n              }\n              if (clicking) {\n                onKeyDown?.(event)\n              }\n            }}\n            onMouseDown={() => setClicking(true)}\n            onMouseUp={() => setClicking(false)}\n          >\n            <Tooltip\n              text={draggableTooltip}\n              visible={clicking ? false : undefined} // Hide the tooltip when dragging the card\n            >\n              <DragIcon\n                disabled={disabled}\n                prominence={clicking ? 'strong' : 'weak'}\n                sentiment=\"neutral\"\n                size=\"small\"\n              />\n            </Tooltip>\n          </DragIconContainer>\n        ) : null}\n        <StyledDetails\n          className={className}\n          data-clicking={clicking}\n          data-testid={dataTestId}\n          key={clicking ? 'closed' : 'open'}\n          name={name}\n          open={open ?? expanded}\n          ref={ref}\n          tabIndex={disabled ? -1 : undefined}\n        >\n          <StyledSummary\n            data-disabled={!!disabled}\n            data-testid={dataTestId ? `${dataTestId}-summary` : undefined}\n            onClick={event => {\n              if (disabled || onToggleExpand) {\n                onToggleExpand?.()\n                event.preventDefault()\n              }\n            }}\n            onKeyDown={\n              onToggleExpand\n                ? event => {\n                    if (\n                      event.key === ' ' &&\n                      event.target === headerRef.current\n                    ) {\n                      onToggleExpand()\n                      event.preventDefault()\n                    }\n                  }\n                : undefined\n            }\n            ref={headerRef}\n          >\n            <StyledArrowIcon disabled={disabled} sentiment=\"neutral\" />\n            {typeof header === 'string' ? (\n              <ExpandableCardTitle disabled={disabled} size={size}>\n                {header}\n              </ExpandableCardTitle>\n            ) : (\n              header\n            )}\n          </StyledSummary>\n          <StyledContent>{children}</StyledContent>\n        </StyledDetails>\n        {draggable && index === 0 ? (\n          <DropableArea\n            data-first\n            onDragLeave={event => onDrag(event, 'transparent', true)}\n            onDragOver={event =>\n              onDrag(event, theme.colors.primary.border, true)\n            }\n            onDrop={event => handleDrop(event, true)}\n            ref={draggableFirstRef}\n          />\n        ) : null}\n        {draggable ? (\n          <DropableArea\n            data-testid={`${value}-dropable-area`}\n            onDragLeave={event => onDrag(event, 'transparent')}\n            onDragOver={event => onDrag(event, theme.colors.primary.border)}\n            onDrop={handleDrop}\n            ref={draggableRef}\n          />\n        ) : null}\n      </StyledStack>\n    )\n  },\n)\n\n// /**\n//  * ExpandableCard is a card that can be collapsed and expanded to reveal more content.\n//  */\nexport const ExpandableCard = Object.assign(BaseExpandableCard, {\n  Title: ExpandableCardTitle,\n})\n"]} */"));
11
+ const styles_css = require("./styles.css.cjs");
96
12
  const BaseExpandableCard = react.forwardRef(({
97
13
  header,
98
14
  name,
@@ -106,7 +22,7 @@ const BaseExpandableCard = react.forwardRef(({
106
22
  value,
107
23
  draggableTooltip = "Click and drag to move",
108
24
  onDrop,
109
- index: index$12,
25
+ index: index$2,
110
26
  onKeyDown,
111
27
  "data-testid": dataTestId,
112
28
  open
@@ -147,8 +63,8 @@ const BaseExpandableCard = react.forwardRef(({
147
63
  onDrop?.(isFirst ? "" : value, data.value ?? "");
148
64
  }
149
65
  }, [onDrop, value]);
150
- return /* @__PURE__ */ jsxRuntime.jsxs(StyledStack, { "data-draggable": draggable, "data-name": name, "data-value": value, direction: "row", draggable, gap: 2, onDragEnd, onDragStart, onMouseEnter: handleMouseEnter, onMouseLeave: handleMouseLeave, ref: containerRef, width: "100%", children: [
151
- draggable ? /* @__PURE__ */ jsxRuntime.jsx(DragIconContainer, { "data-testid": `draggable-icon-${value}`, "data-visible": isHovered, justifyContent: "center", onKeyDown: (event) => {
66
+ return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { className: styles_css.stackClass, "data-draggable": draggable, "data-name": name, "data-value": value, direction: "row", draggable, gap: 2, onDragEnd, onDragStart, onMouseEnter: handleMouseEnter, onMouseLeave: handleMouseLeave, ref: containerRef, width: "100%", children: [
67
+ draggable ? /* @__PURE__ */ jsxRuntime.jsx(index.Stack, { className: styles_css.dragIconContainer, "data-testid": `draggable-icon-${value}`, "data-visible": isHovered, justifyContent: "center", onKeyDown: (event) => {
152
68
  if (event.key === "Enter" || event.key === " ") {
153
69
  event.preventDefault();
154
70
  if (clicking && containerRef.current) {
@@ -163,15 +79,15 @@ const BaseExpandableCard = react.forwardRef(({
163
79
  onKeyDown?.(event);
164
80
  }
165
81
  }, onMouseDown: () => setClicking(true), onMouseUp: () => setClicking(false), children: /* @__PURE__ */ jsxRuntime.jsx(
166
- index.Tooltip,
82
+ index$1.Tooltip,
167
83
  {
168
84
  text: draggableTooltip,
169
85
  visible: clicking ? false : void 0,
170
86
  children: /* @__PURE__ */ jsxRuntime.jsx(Icon.DragIcon, { disabled, prominence: clicking ? "strong" : "weak", sentiment: "neutral", size: "small" })
171
87
  }
172
88
  ) }) : null,
173
- /* @__PURE__ */ jsxRuntime.jsxs(StyledDetails, { className, "data-clicking": clicking, "data-testid": dataTestId, name, open: open ?? expanded, ref, tabIndex: disabled ? -1 : void 0, children: [
174
- /* @__PURE__ */ jsxRuntime.jsxs(StyledSummary, { "data-disabled": !!disabled, "data-testid": dataTestId ? `${dataTestId}-summary` : void 0, onClick: (event) => {
89
+ /* @__PURE__ */ jsxRuntime.jsxs("details", { className: `${className ? `${className} ` : ""}${styles_css.detailsClass}`, "data-clicking": clicking, "data-testid": dataTestId, name, open: open ?? expanded, ref, tabIndex: disabled ? -1 : void 0, children: [
90
+ /* @__PURE__ */ jsxRuntime.jsxs("summary", { className: styles_css.summaryClass, "data-disabled": !!disabled, "data-testid": dataTestId ? `${dataTestId}-summary` : void 0, onClick: (event) => {
175
91
  if (disabled || onToggleExpand) {
176
92
  onToggleExpand?.();
177
93
  event.preventDefault();
@@ -182,13 +98,13 @@ const BaseExpandableCard = react.forwardRef(({
182
98
  event.preventDefault();
183
99
  }
184
100
  } : void 0, ref: headerRef, children: [
185
- /* @__PURE__ */ jsxRuntime.jsx(StyledArrowIcon, { disabled, sentiment: "neutral" }),
101
+ /* @__PURE__ */ jsxRuntime.jsx(Icon.ArrowDownIcon, { className: styles_css.arrowIcon, disabled, sentiment: "neutral" }),
186
102
  typeof header === "string" ? /* @__PURE__ */ jsxRuntime.jsx(Title.ExpandableCardTitle, { disabled, size, children: header }) : header
187
103
  ] }),
188
- /* @__PURE__ */ jsxRuntime.jsx(StyledContent, { children })
104
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: styles_css.content, children })
189
105
  ] }, clicking ? "closed" : "open"),
190
- draggable && index$12 === 0 ? /* @__PURE__ */ jsxRuntime.jsx(DropableArea, { "data-first": true, onDragLeave: (event) => onDrag(event, "transparent", true), onDragOver: (event) => onDrag(event, theme.colors.primary.border, true), onDrop: (event) => handleDrop(event, true), ref: draggableFirstRef }) : null,
191
- draggable ? /* @__PURE__ */ jsxRuntime.jsx(DropableArea, { "data-testid": `${value}-dropable-area`, onDragLeave: (event) => onDrag(event, "transparent"), onDragOver: (event) => onDrag(event, theme.colors.primary.border), onDrop: handleDrop, ref: draggableRef }) : null
106
+ draggable && index$2 === 0 ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: styles_css.dropableArea, "data-first": true, onDragLeave: (event) => onDrag(event, "transparent", true), onDragOver: (event) => onDrag(event, theme.colors.primary.border, true), onDrop: (event) => handleDrop(event, true), ref: draggableFirstRef }) : null,
107
+ draggable ? /* @__PURE__ */ jsxRuntime.jsx("div", { className: styles_css.dropableArea, "data-testid": `${value}-dropable-area`, onDragLeave: (event) => onDrag(event, "transparent"), onDragOver: (event) => onDrag(event, theme.colors.primary.border), onDrop: handleDrop, ref: draggableRef }) : null
192
108
  ] });
193
109
  });
194
110
  const ExpandableCard = Object.assign(BaseExpandableCard, {