@ultraviolet/ui 3.0.0-beta.2 → 3.0.0-beta.4

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 (36) hide show
  1. package/dist/components/ActionBar/index.cjs +5 -30
  2. package/dist/components/ActionBar/index.js +5 -28
  3. package/dist/components/ActionBar/styles.css.cjs +8 -0
  4. package/dist/components/ActionBar/styles.css.d.ts +2 -0
  5. package/dist/components/ActionBar/styles.css.js +8 -0
  6. package/dist/components/Badge/constant.cjs +8 -0
  7. package/dist/components/Badge/constant.d.ts +14 -0
  8. package/dist/components/Badge/constant.js +8 -0
  9. package/dist/components/Badge/index.cjs +10 -88
  10. package/dist/components/Badge/index.d.ts +2 -20
  11. package/dist/components/Badge/index.js +8 -84
  12. package/dist/components/Badge/styles.css.cjs +6 -0
  13. package/dist/components/Badge/styles.css.d.ts +43 -0
  14. package/dist/components/Badge/styles.css.js +6 -0
  15. package/dist/components/Button/index.cjs +1 -1
  16. package/dist/components/Button/index.js +1 -1
  17. package/dist/components/Button/styles.css.d.ts +4 -4
  18. package/dist/components/Chip/index.d.ts +1 -1
  19. package/dist/components/ExpandableCard/index.cjs +10 -9
  20. package/dist/components/ExpandableCard/index.d.ts +3 -1
  21. package/dist/components/ExpandableCard/index.js +10 -9
  22. package/dist/components/NumberInput/index.d.ts +1 -1
  23. package/dist/components/Popup/index.cjs +21 -7
  24. package/dist/components/Popup/index.js +21 -7
  25. package/dist/components/Slider/styles.d.ts +1 -1
  26. package/dist/components/Snippet/index.cjs +15 -14
  27. package/dist/components/Snippet/index.js +15 -14
  28. package/dist/components/Tabs/index.d.ts +1 -1
  29. package/dist/components/TextArea/index.cjs +6 -6
  30. package/dist/components/TextArea/index.js +6 -6
  31. package/dist/components/TextInput/index.d.ts +1 -1
  32. package/dist/components/Tooltip/index.d.ts +1 -1
  33. package/dist/ui.css +1 -1
  34. package/dist/utils/animationVanillaExtract.css.d.ts +1 -0
  35. package/dist/utils/index.d.ts +1 -0
  36. package/package.json +3 -3
@@ -16,7 +16,7 @@ const StyledArrowIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowDownI
16
16
  } : {
17
17
  target: "e143cpd66",
18
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":"AAY6C","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */");
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 => {\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
20
  const DropableArea = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
21
21
  target: "e143cpd65"
22
22
  } : {
@@ -34,7 +34,7 @@ const DropableArea = /* @__PURE__ */ _styled__default.default("div", process.env
34
34
  theme
35
35
  }) => theme.space["1"], ";margin-left:-", ({
36
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":"AAc+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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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
38
  const StyledSummary = /* @__PURE__ */ _styled__default.default("summary", process.env.NODE_ENV === "production" ? {
39
39
  target: "e143cpd64"
40
40
  } : {
@@ -48,7 +48,7 @@ const StyledSummary = /* @__PURE__ */ _styled__default.default("summary", proces
48
48
  theme
49
49
  }) => theme.colors.neutral.backgroundWeak, ";border-radius:", ({
50
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":"AA2CoC","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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
52
  const StyledContent = /* @__PURE__ */ _styled__default.default("div", process.env.NODE_ENV === "production" ? {
53
53
  target: "e143cpd63"
54
54
  } : {
@@ -58,7 +58,7 @@ const StyledContent = /* @__PURE__ */ _styled__default.default("div", process.en
58
58
  theme
59
59
  }) => theme.colors.neutral.border, ";padding:", ({
60
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":"AA2DgC","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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
62
  const StyledDetails = /* @__PURE__ */ _styled__default.default("details", process.env.NODE_ENV === "production" ? {
63
63
  target: "e143cpd62"
64
64
  } : {
@@ -76,7 +76,7 @@ const StyledDetails = /* @__PURE__ */ _styled__default.default("details", proces
76
76
  theme
77
77
  }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";border-color:", ({
78
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":"AAiEoC","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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
80
  const StyledStack = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
81
81
  target: "e143cpd61"
82
82
  } : {
@@ -84,7 +84,7 @@ const StyledStack = /* @__PURE__ */ _styled__default.default(index$1.Stack, proc
84
84
  label: "StyledStack"
85
85
  })("position:relative;&:hover>", StyledDetails, "{border-color:", ({
86
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":"AAyFiC","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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
88
  const DragIconContainer = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
89
89
  target: "e143cpd60"
90
90
  } : {
@@ -92,7 +92,7 @@ const DragIconContainer = /* @__PURE__ */ _styled__default.default(index$1.Stack
92
92
  label: "DragIconContainer"
93
93
  })("height:100%;opacity:0;transition:opacity 0.2s ease-in-out;cursor:grab;padding-top:calc(", ({
94
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":"AAgGuC","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 { DragEvent, ForwardedRef, KeyboardEvent, ReactNode } 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} & 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    }: 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 => {\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={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"]} */"));
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 => {\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"]} */"));
96
96
  const BaseExpandableCard = react.forwardRef(({
97
97
  header,
98
98
  name,
@@ -108,7 +108,8 @@ const BaseExpandableCard = react.forwardRef(({
108
108
  onDrop,
109
109
  index: index$12,
110
110
  onKeyDown,
111
- "data-testid": dataTestId
111
+ "data-testid": dataTestId,
112
+ open
112
113
  }, ref) => {
113
114
  const headerRef = react.useRef(null);
114
115
  const containerRef = react.useRef(null);
@@ -169,7 +170,7 @@ const BaseExpandableCard = react.forwardRef(({
169
170
  children: /* @__PURE__ */ jsxRuntime.jsx(Icon.DragIcon, { disabled, prominence: clicking ? "strong" : "weak", sentiment: "neutral", size: "small" })
170
171
  }
171
172
  ) }) : null,
172
- /* @__PURE__ */ jsxRuntime.jsxs(StyledDetails, { className, "data-clicking": clicking, "data-testid": dataTestId, name, open: expanded, ref, tabIndex: disabled ? -1 : void 0, children: [
173
+ /* @__PURE__ */ jsxRuntime.jsxs(StyledDetails, { className, "data-clicking": clicking, "data-testid": dataTestId, name, open: open ?? expanded, ref, tabIndex: disabled ? -1 : void 0, children: [
173
174
  /* @__PURE__ */ jsxRuntime.jsxs(StyledSummary, { "data-disabled": !!disabled, "data-testid": dataTestId ? `${dataTestId}-summary` : void 0, onClick: (event) => {
174
175
  if (disabled || onToggleExpand) {
175
176
  onToggleExpand?.();
@@ -1,4 +1,4 @@
1
- import type { KeyboardEvent, ReactNode } from 'react';
1
+ import type { DetailsHTMLAttributes, KeyboardEvent, ReactNode } from 'react';
2
2
  export declare const EXPANDABLE_CARD_SIZE: readonly ["medium", "large"];
3
3
  type ExpandableCardSize = (typeof EXPANDABLE_CARD_SIZE)[number];
4
4
  type DraggableProps = {
@@ -30,6 +30,8 @@ type CommonProps = {
30
30
  disabled?: boolean;
31
31
  'data-testid'?: string;
32
32
  className?: string;
33
+ /** Uncontrolled but open by default */
34
+ open?: DetailsHTMLAttributes<HTMLDetailsElement>['open'];
33
35
  } & DraggableProps;
34
36
  export declare const ExpandableCard: import("react").ForwardRefExoticComponent<(({
35
37
  expanded?: undefined;