@ultraviolet/ui 2.0.0-beta.10 → 2.0.0-beta.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/DateInput/index.cjs +2 -2
- package/dist/components/DateInput/index.js +2 -2
- package/dist/components/List/HeaderCell.cjs +3 -3
- package/dist/components/List/HeaderCell.js +4 -4
- package/dist/components/List/HeaderRow.cjs +2 -4
- package/dist/components/List/HeaderRow.js +2 -4
- package/dist/components/List/Row.cjs +32 -16
- package/dist/components/List/Row.js +32 -16
- package/dist/components/List/index.cjs +3 -3
- package/dist/components/List/index.js +3 -3
- package/dist/components/Menu/MenuContent.cjs +9 -8
- package/dist/components/Menu/MenuContent.d.ts +3 -1
- package/dist/components/Menu/MenuContent.js +10 -9
- package/dist/components/Menu/MenuProvider.cjs +9 -8
- package/dist/components/Menu/MenuProvider.d.ts +3 -0
- package/dist/components/Menu/MenuProvider.js +10 -9
- package/dist/components/Menu/components/Item.cjs +19 -7
- package/dist/components/Menu/components/Item.js +21 -9
- package/dist/components/Menu/helpers.cjs +3 -0
- package/dist/components/Menu/helpers.js +3 -0
- package/dist/components/Menu/index.d.ts +2 -1
- package/dist/components/Menu/types.d.ts +2 -1
- package/dist/components/Modal/components/Disclosure.cjs +5 -5
- package/dist/components/Modal/components/Disclosure.d.ts +1 -1
- package/dist/components/Modal/components/Disclosure.js +6 -6
- package/dist/components/Modal/index.cjs +4 -1
- package/dist/components/Modal/index.js +5 -2
- package/dist/components/Modal/types.d.ts +2 -1
- package/dist/components/NumberInput/index.cjs +23 -31
- package/dist/components/NumberInput/index.js +23 -31
- package/dist/components/Popover/index.cjs +2 -2
- package/dist/components/Popover/index.d.ts +4 -2
- package/dist/components/Popover/index.js +2 -2
- package/dist/components/Popup/helpers.cjs +41 -1
- package/dist/components/Popup/helpers.d.ts +1 -1
- package/dist/components/Popup/helpers.js +41 -1
- package/dist/components/Popup/index.cjs +8 -12
- package/dist/components/Popup/index.js +8 -12
- package/dist/components/SearchInput/types.d.ts +1 -1
- package/dist/components/SelectInput/Dropdown.cjs +9 -9
- package/dist/components/SelectInput/Dropdown.js +9 -9
- package/dist/components/Slider/styles.d.ts +26 -15
- package/dist/components/StepList/index.cjs +3 -3
- package/dist/components/StepList/index.js +3 -3
- package/dist/components/Table/HeaderCell.cjs +4 -4
- package/dist/components/Table/HeaderCell.js +5 -5
- package/dist/components/Table/index.cjs +2 -2
- package/dist/components/Table/index.js +2 -2
- package/dist/components/TimeInput/index.cjs +3 -3
- package/dist/components/TimeInput/index.js +3 -3
- package/dist/components/Tooltip/index.cjs +1 -1
- package/dist/components/Tooltip/index.d.ts +36 -4
- package/dist/components/Tooltip/index.js +1 -1
- package/dist/components/UnitInput/index.cjs +18 -37
- package/dist/components/UnitInput/index.js +18 -37
- package/package.json +4 -4
|
@@ -26,13 +26,13 @@ const ExpandableWrapper = /* @__PURE__ */ _styled("tr", process.env.NODE_ENV ===
|
|
|
26
26
|
theme
|
|
27
27
|
}) => theme.radii.default, ";transform:translate3d(0, -", ({
|
|
28
28
|
theme
|
|
29
|
-
}) => theme.space["2"],
|
|
29
|
+
}) => theme.space["2"], ", 0);position:relative;td,td:first-child,td:last-child{transition:box-shadow 200ms ease,border-color 200ms ease;}td{border:1px solid ", ({
|
|
30
30
|
theme
|
|
31
31
|
}) => theme.colors.neutral.border, ";border-top:none;border-radius:0 0 ", ({
|
|
32
32
|
theme
|
|
33
33
|
}) => theme.radii.default, " ", ({
|
|
34
34
|
theme
|
|
35
|
-
}) => theme.radii.default, ";pointer-events:none;transition:box-shadow 200ms ease,border-color 200ms ease;}" + (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/List/Row.tsx"],"names":[],"mappings":"AAwBmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
35
|
+
}) => theme.radii.default, ";}" + (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/List/Row.tsx"],"names":[],"mappings":"AAwBmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
36
36
|
const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV === "production" ? {
|
|
37
37
|
shouldForwardProp: (prop) => !["inRange"].includes(prop),
|
|
38
38
|
target: "ei4uyz14"
|
|
@@ -48,7 +48,7 @@ const StyledCheckbox = /* @__PURE__ */ _styled(Checkbox, process.env.NODE_ENV ==
|
|
|
48
48
|
fill: ${theme.colors.neutral.backgroundHover};
|
|
49
49
|
stroke: ${theme.colors.neutral.borderHover};
|
|
50
50
|
}
|
|
51
|
-
` : "", ";" + (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/List/Row.tsx"],"names":[],"mappings":"AAqDwB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
51
|
+
` : "", ";" + (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/List/Row.tsx"],"names":[],"mappings":"AAiDwB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
52
52
|
const colorChange = (theme) => keyframes`
|
|
53
53
|
5% {
|
|
54
54
|
background-color: ${theme.colors.primary.background};
|
|
@@ -70,21 +70,37 @@ const StyledRow = /* @__PURE__ */ _styled("tr", process.env.NODE_ENV === "produc
|
|
|
70
70
|
theme
|
|
71
71
|
}) => theme.typography.bodySmall.fontSize, ";column-gap:", ({
|
|
72
72
|
theme
|
|
73
|
-
}) => theme.space["2"],
|
|
73
|
+
}) => theme.space["2"], ";&[role='button row']{cursor:pointer;}position:relative;td,td:first-child,td:last-child{transition:box-shadow 200ms ease,border-color 200ms ease;}td{border-top:1px solid ", ({
|
|
74
|
+
theme
|
|
75
|
+
}) => theme.colors.neutral.border, ";border-bottom:1px solid ", ({
|
|
76
|
+
theme
|
|
77
|
+
}) => theme.colors.neutral.border, ";}td:first-child{border-left:1px solid ", ({
|
|
74
78
|
theme
|
|
75
79
|
}) => theme.colors.neutral.border, ";border-radius:", ({
|
|
76
80
|
theme
|
|
77
|
-
}) => theme.radii.default, "
|
|
81
|
+
}) => theme.radii.default, " 0 0 ", ({
|
|
78
82
|
theme
|
|
79
|
-
}) => theme.
|
|
83
|
+
}) => theme.radii.default, ";}td:last-child{border-right:1px solid ", ({
|
|
80
84
|
theme
|
|
81
|
-
}) => theme.colors.
|
|
85
|
+
}) => theme.colors.neutral.border, ";border-radius:0 ", ({
|
|
82
86
|
theme
|
|
83
87
|
}) => theme.radii.default, " ", ({
|
|
84
88
|
theme
|
|
85
|
-
}) => theme.radii.default, " 0
|
|
89
|
+
}) => theme.radii.default, " 0;}&:not([aria-disabled='true']):hover td,&:not([aria-disabled='true']):hover td:first-child,&:not([aria-disabled='true']):hover td:last-child{border-color:", ({
|
|
90
|
+
theme
|
|
91
|
+
}) => theme.colors.primary.border, ";}&:not([aria-disabled='true']):hover+", ExpandableWrapper, " td{border-color:", ({
|
|
92
|
+
theme
|
|
93
|
+
}) => theme.colors.primary.border, ";}&[aria-expanded='true'] td{border-bottom-color:", ({
|
|
94
|
+
theme
|
|
95
|
+
}) => theme.colors.neutral.border, ";&:first-child{border-left:1px solid ", ({
|
|
96
|
+
theme
|
|
97
|
+
}) => theme.colors.neutral.border, ";border-radius:", ({
|
|
98
|
+
theme
|
|
99
|
+
}) => theme.radii.default, " 0 0 0;}&:last-child{border-right:1px solid ", ({
|
|
100
|
+
theme
|
|
101
|
+
}) => theme.colors.neutral.border, ";border-radius:0 ", ({
|
|
86
102
|
theme
|
|
87
|
-
}) => theme.
|
|
103
|
+
}) => theme.radii.default, " 0 0;}}", ({
|
|
88
104
|
theme,
|
|
89
105
|
sentiment
|
|
90
106
|
}) => `
|
|
@@ -100,11 +116,11 @@ const StyledRow = /* @__PURE__ */ _styled("tr", process.env.NODE_ENV === "produc
|
|
|
100
116
|
box-shadow: ${theme.shadows.hoverPrimary};
|
|
101
117
|
}
|
|
102
118
|
` : ""}
|
|
103
|
-
`, " &[data-highlight='true']{
|
|
119
|
+
`, " &[data-highlight='true']{box-shadow:", ({
|
|
104
120
|
theme
|
|
105
|
-
}) => theme.
|
|
121
|
+
}) => theme.shadows.hoverPrimary, ";td,td:first-child,td:last-child{border-color:", ({
|
|
106
122
|
theme
|
|
107
|
-
}) => theme.
|
|
123
|
+
}) => theme.colors.primary.border, ";}}&[aria-disabled='true']{background-color:", ({
|
|
108
124
|
theme
|
|
109
125
|
}) => theme.colors.neutral.backgroundDisabled, ";color:", ({
|
|
110
126
|
theme
|
|
@@ -120,7 +136,7 @@ const StyledRow = /* @__PURE__ */ _styled("tr", process.env.NODE_ENV === "produc
|
|
|
120
136
|
`), " animation:", ({
|
|
121
137
|
highlightAnimation,
|
|
122
138
|
theme
|
|
123
|
-
}) => highlightAnimation ? colorChange(theme) : void 0, " 3s linear;" + (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/List/Row.tsx"],"names":[],"mappings":"AAqFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
139
|
+
}) => highlightAnimation ? colorChange(theme) : void 0, " 3s linear;" + (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/List/Row.tsx"],"names":[],"mappings":"AAiFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
124
140
|
const StyledCheckboxContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
|
|
125
141
|
target: "ei4uyz12"
|
|
126
142
|
} : {
|
|
@@ -131,7 +147,7 @@ const StyledCheckboxContainer = /* @__PURE__ */ _styled("div", process.env.NODE_
|
|
|
131
147
|
styles: "display:flex"
|
|
132
148
|
} : {
|
|
133
149
|
name: "zjik7",
|
|
134
|
-
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx"],"names":[],"mappings":"AAiL0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */",
|
|
150
|
+
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx"],"names":[],"mappings":"AA2L0C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */",
|
|
135
151
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
136
152
|
});
|
|
137
153
|
const NoPaddingCell = /* @__PURE__ */ _styled(Cell, process.env.NODE_ENV === "production" ? {
|
|
@@ -145,7 +161,7 @@ const NoPaddingCell = /* @__PURE__ */ _styled(Cell, process.env.NODE_ENV === "pr
|
|
|
145
161
|
theme
|
|
146
162
|
}) => theme.space["2"], ";}max-width:", ({
|
|
147
163
|
maxWidth
|
|
148
|
-
}) => maxWidth, ";" + (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/List/Row.tsx"],"names":[],"mappings":"AAyLE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
164
|
+
}) => maxWidth, ";" + (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/List/Row.tsx"],"names":[],"mappings":"AAmME","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
149
165
|
const ExpandableCell = /* @__PURE__ */ _styled(Cell, process.env.NODE_ENV === "production" ? {
|
|
150
166
|
shouldForwardProp: (prop) => !["padding"].includes(prop),
|
|
151
167
|
target: "ei4uyz10"
|
|
@@ -156,7 +172,7 @@ const ExpandableCell = /* @__PURE__ */ _styled(Cell, process.env.NODE_ENV === "p
|
|
|
156
172
|
})("padding:", ({
|
|
157
173
|
theme,
|
|
158
174
|
padding
|
|
159
|
-
}) => padding ? theme.space[padding] : theme.space["2"], ";" + (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/List/Row.tsx"],"names":[],"mappings":"AAqMoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  &:before {\n    content: \"\";\n    position: absolute;\n    top: 0; /* Adjust based on border width and spacing */\n    left: 0;\n    right: 0;\n    bottom: 0; /* Adjust based on border width and spacing */\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default};\n    pointer-events: none;\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  &:not([aria-disabled='true']):hover::before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper}:before {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true']:before {\n    border-radius: ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0 0;\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
175
|
+
}) => padding ? theme.space[padding] : theme.space["2"], ";" + (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/List/Row.tsx"],"names":[],"mappings":"AA+MoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/List/Row.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { keyframes, useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { ArrowDownIcon, ArrowUpIcon } from '@ultraviolet/icons'\nimport type { ReactNode, RefObject } from 'react'\nimport {\n  Children,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useId,\n  useRef,\n} from 'react'\nimport type { SENTIMENTS, space } from '../../theme'\nimport { Button } from '../Button'\nimport { Checkbox } from '../Checkbox'\nimport { Tooltip } from '../Tooltip'\nimport { Cell } from './Cell'\nimport { useListContext } from './ListContext'\nimport { SELECTABLE_CHECKBOX_SIZE } from './constants'\nimport type { ColumnProps } from './types'\n\nconst ExpandableWrapper = styled.tr`\n  width: 100%;\n  display: table-row;\n  vertical-align: middle;\n  cursor: auto;\n  background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  transform: translate3d(0, -${({ theme }) => theme.space['2']}, 0);\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-top: none;\n    border-radius: 0 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default};\n  }\n`\n\nconst StyledCheckbox = styled(Checkbox, {\n  shouldForwardProp: prop => !['inRange'].includes(prop),\n})<{ inRange: boolean }>`\n  ${({ theme, inRange }) =>\n    inRange\n      ? `\n        rect {\n          fill: ${theme.colors.neutral.backgroundHover};\n          stroke: ${theme.colors.neutral.borderHover};\n        }\n  `\n      : ''}\n\n`\n\nconst colorChange = (theme: Theme) => keyframes`\n  5% {\n    background-color: ${theme.colors.primary.background};\n  }\n  80% {\n    background-color: ${theme.colors.primary.background};\n  }\n`\n\nexport const StyledRow = styled('tr', {\n  shouldForwardProp: prop =>\n    !['highlightAnimation', 'sentiment', 'columns', 'columnsStartAt'].includes(\n      prop,\n    ),\n})<{\n  sentiment: (typeof SENTIMENTS)[number]\n  columns: ColumnProps[]\n  columnsStartAt?: number\n  highlightAnimation?: boolean\n}>`\n  /* List itself also apply style about common templating between HeaderRow and other Rows */\n\n  display: table-row;\n  vertical-align: middle;\n  position: relative;\n  box-shadow: none;\n  background-color: ${({ theme }) => theme.colors.neutral.background};\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  column-gap: ${({ theme }) => theme.space['2']};\n\n  &[role='button row'] {\n    cursor: pointer;\n  }\n\n  position: relative;\n\n  td, td:first-child, td:last-child {\n    transition:\n      box-shadow 200ms ease,\n      border-color 200ms ease;\n  }\n\n  td {\n    border-top: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-bottom: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  }\n  td:first-child {\n    border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: ${({ theme }) => theme.radii.default} 0 0 ${({ theme }) => theme.radii.default};\n  }\n  td:last-child {\n    border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n    border-radius: 0 ${({ theme }) => theme.radii.default} ${({ theme }) => theme.radii.default} 0;\n  }\n\n  &:not([aria-disabled='true']):hover td, &:not([aria-disabled='true']):hover td:first-child, &:not([aria-disabled='true']):hover td:last-child {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &:not([aria-disabled='true']):hover + ${ExpandableWrapper} td {\n    border-color: ${({ theme }) => theme.colors.primary.border};\n  }\n\n  &[aria-expanded='true'] td {\n    &:first-child {\n      border-left: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: ${({ theme }) => theme.radii.default} 0 0 0;\n    }\n    &:last-child {\n      border-right: 1px solid ${({ theme }) => theme.colors.neutral.border};\n      border-radius: 0 ${({ theme }) => theme.radii.default} 0 0;\n    }\n    border-bottom-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  ${({ theme, sentiment }) =>\n    `\n    color: ${theme.colors[sentiment].text};\n    border-color: ${theme.colors[sentiment].border};\n    background-color: ${theme.colors[sentiment].background};\n    & [data-expandable-content] {\n      border-color: ${theme.colors[sentiment].border};\n    }\n\n    ${\n      sentiment === 'neutral'\n        ? `&:not([aria-disabled='true']):hover {\n          border-color: ${theme.colors.primary.border};\n          box-shadow: ${theme.shadows.hoverPrimary};\n        }\n        `\n        : ''\n    }\n  `}\n\n  &[data-highlight='true'] {\n    td, td:first-child, td:last-child {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    box-shadow: ${({ theme }) => theme.shadows.hoverPrimary};\n  }\n\n  &[aria-disabled='true'] {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    cursor: not-allowed;\n  }\n\n  ${({ columns, columnsStartAt }) =>\n    columns.map(\n      (column, index) => `\n    td:nth-of-type(${index + 1 + (columnsStartAt ?? 0)}) {\n      ${column.width ? `width: ${column.width};` : ''}\n      ${column.minWidth ? `min-width: ${column.minWidth};` : ''}\n      ${column.maxWidth ? `max-width: ${column.maxWidth};` : ''}\n    }\n  `,\n    )}\n\n  animation: ${({ highlightAnimation, theme }) =>\n    highlightAnimation ? colorChange(theme) : undefined}\n    3s linear;\n`\n\nconst StyledCheckboxContainer = styled.div`\n  display: flex;\n`\n\nconst NoPaddingCell = styled(Cell, {\n  shouldForwardProp: prop => !['maxWidth'].includes(prop),\n})<{\n  maxWidth: string\n}>`\n  padding: 0;\n\n  &:first-of-type {\n    padding-left: ${({ theme }) => theme.space['2']};\n  }\n\n  max-width: ${({ maxWidth }) => maxWidth};\n`\n\nconst ExpandableCell = styled(Cell, {\n  shouldForwardProp: prop => !['padding'].includes(prop),\n})<{ padding?: keyof typeof space }>`\n  padding: ${({ theme, padding }) => (padding ? theme.space[padding] : theme.space['2'])};\n`\n\ntype RowProps = {\n  children: ReactNode\n  id: string\n  expandable?: ReactNode\n  /**\n   * Row cannot be selected if this prop is provided. boolean true disabled selection, a string disable selection and a tooltip will be displayed on checkbox hover.\n   * */\n  selectDisabled?: boolean | string\n  disabled?: boolean\n  sentiment?: (typeof SENTIMENTS)[number]\n  expanded?: boolean\n  className?: string\n  /**\n   * Define a custom padding for the content in the expandable\n   */\n  expandablePadding?: keyof typeof space\n  highlightAnimation?: boolean\n  'data-testid'?: string\n}\n\nexport const Row = forwardRef<HTMLTableRowElement, RowProps>(\n  (\n    {\n      children,\n      id,\n      expandable,\n      disabled,\n      selectDisabled,\n      sentiment = 'neutral',\n      highlightAnimation,\n      expanded,\n      className,\n      expandablePadding,\n      'data-testid': dataTestid,\n    },\n    forwardedRef,\n  ) => {\n    const {\n      selectable,\n      registerExpandableRow,\n      expandedRowIds,\n      expandRow,\n      collapseRow,\n      registerSelectableRow,\n      selectedRowIds,\n      expandButton,\n      inRange,\n      columns,\n      refList,\n      setRefList,\n      handleOnChange,\n    } = useListContext()\n\n    const theme = useTheme()\n\n    const expandedRowId = useId()\n\n    const checkboxRef = useRef<HTMLInputElement>(null)\n\n    const isSelectDisabled =\n      disabled || (selectDisabled !== undefined && selectDisabled !== false)\n\n    const hasExpandable = !!expandable\n    useEffect(() => {\n      if (hasExpandable) {\n        const unregisterCallback = registerExpandableRow(id, expanded)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, hasExpandable, registerExpandableRow, expanded, expandRow])\n\n    useEffect(() => {\n      if (!isSelectDisabled) {\n        const unregisterCallback = registerSelectableRow(id)\n\n        return unregisterCallback\n      }\n\n      return undefined\n    }, [id, registerSelectableRow, isSelectDisabled])\n\n    const toggleRowExpand = useCallback(() => {\n      if (expandedRowIds[id]) {\n        collapseRow(id)\n      } else {\n        expandRow(id)\n      }\n    }, [collapseRow, expandRow, expandedRowIds, id])\n\n    const canClickRowToExpand = !disabled && !!expandable && !expandButton\n\n    useEffect(() => {\n      if (\n        checkboxRef.current !== null &&\n        !refList.includes(checkboxRef as RefObject<HTMLInputElement>)\n      ) {\n        setRefList([...refList, checkboxRef as RefObject<HTMLInputElement>])\n      }\n    }, [refList, setRefList])\n\n    const childrenLength =\n      Children.count(children) + (selectable ? 1 : 0) + (expandButton ? 1 : 0)\n\n    return (\n      <>\n        <StyledRow\n          className={className}\n          ref={forwardedRef}\n          role={canClickRowToExpand ? 'button row' : undefined}\n          onClick={canClickRowToExpand ? toggleRowExpand : undefined}\n          onKeyDown={\n            canClickRowToExpand\n              ? event => {\n                  if (event.key === ' ') {\n                    toggleRowExpand()\n                    event.preventDefault()\n                  }\n                }\n              : undefined\n          }\n          tabIndex={canClickRowToExpand ? 0 : -1}\n          sentiment={sentiment}\n          aria-disabled={disabled}\n          aria-expanded={expandable ? expandedRowIds[id] : undefined}\n          aria-controls={\n            expandable && expandedRowIds[id] ? expandedRowId : undefined\n          }\n          data-highlight={selectable && !!selectedRowIds[id]}\n          data-testid={dataTestid}\n          columns={columns}\n          columnsStartAt={(selectable ? 1 : 0) + (expandButton ? 1 : 0)}\n          highlightAnimation={highlightAnimation}\n        >\n          {selectable ? (\n            <NoPaddingCell\n              preventClick={canClickRowToExpand}\n              maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}\n            >\n              <StyledCheckboxContainer>\n                <Tooltip\n                  text={\n                    typeof selectDisabled === 'string'\n                      ? selectDisabled\n                      : undefined\n                  }\n                >\n                  <StyledCheckbox\n                    name=\"list-select-checkbox\"\n                    aria-label=\"select\"\n                    checked={selectedRowIds[id]}\n                    value={id}\n                    ref={checkboxRef}\n                    disabled={isSelectDisabled}\n                    inRange={inRange?.includes(id)}\n                    onChange={() => handleOnChange(id, selectedRowIds[id])}\n                  />\n                </Tooltip>\n              </StyledCheckboxContainer>\n            </NoPaddingCell>\n          ) : null}\n          {expandButton ? (\n            <NoPaddingCell maxWidth={theme.sizing[SELECTABLE_CHECKBOX_SIZE]}>\n              <Button\n                disabled={disabled || !expandable}\n                onClick={toggleRowExpand}\n                size=\"small\"\n                sentiment={sentiment}\n                variant=\"ghost\"\n                aria-label=\"expand\"\n                data-testid=\"list-expand-button\"\n              >\n                {expandedRowIds[id] ? <ArrowUpIcon /> : <ArrowDownIcon />}\n              </Button>\n            </NoPaddingCell>\n          ) : null}\n          {children}\n        </StyledRow>\n        {expandable && expandedRowIds[id] ? (\n          <ExpandableWrapper\n            id={expandedRowId}\n            data-expandable-content\n            onClick={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n            onKeyDown={\n              canClickRowToExpand\n                ? e => {\n                    e.stopPropagation()\n                  }\n                : undefined\n            }\n          >\n            <ExpandableCell\n              colSpan={childrenLength}\n              padding={expandablePadding}\n            >\n              {expandable}\n            </ExpandableCell>\n          </ExpandableWrapper>\n        ) : null}\n      </>\n    )\n  },\n)\n"]} */"));
|
|
160
176
|
const Row = forwardRef(({
|
|
161
177
|
children,
|
|
162
178
|
id,
|