@amboss/design-system 1.26.6 → 1.28.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/cjs/components/DataTable/DataTable.d.ts +12 -6
- package/build/cjs/components/DataTable/DataTable.js +1 -1
- package/build/cjs/components/DataTable/TableBody.d.ts +3 -3
- package/build/cjs/components/DataTable/TableBody.js +1 -1
- package/build/cjs/components/DataTable/TableCell.d.ts +7 -3
- package/build/cjs/components/DataTable/TableCell.js +1 -1
- package/build/cjs/components/DataTable/TableHeader.d.ts +3 -3
- package/build/cjs/components/DataTable/TableHeader.js +1 -1
- package/build/cjs/components/DropdownMenu/DropdownMenu.d.ts +3 -7
- package/build/cjs/components/DropdownMenu/DropdownMenu.js +1 -1
- package/build/cjs/components/DropdownMenu/Menu.d.ts +14 -0
- package/build/cjs/components/DropdownMenu/Menu.js +1 -0
- package/build/cjs/components/DropdownMenu/utils.d.ts +6 -0
- package/build/cjs/components/DropdownMenu/utils.js +1 -0
- package/build/cjs/components/Tooltip/TooltipContent.js +1 -1
- package/build/cjs/components/Tooltip/utils.js +1 -1
- package/build/cjs/web-tokens/assets/icons.json +1 -0
- package/build/cjs/web-tokens/assets/icons16.json +1 -0
- package/build/esm/components/DataTable/DataTable.d.ts +12 -6
- package/build/esm/components/DataTable/DataTable.js +1 -1
- package/build/esm/components/DataTable/TableBody.d.ts +3 -3
- package/build/esm/components/DataTable/TableBody.js +1 -1
- package/build/esm/components/DataTable/TableCell.d.ts +7 -3
- package/build/esm/components/DataTable/TableCell.js +1 -1
- package/build/esm/components/DataTable/TableHeader.d.ts +3 -3
- package/build/esm/components/DataTable/TableHeader.js +1 -1
- package/build/esm/components/DropdownMenu/DropdownMenu.d.ts +3 -7
- package/build/esm/components/DropdownMenu/DropdownMenu.js +1 -1
- package/build/esm/components/DropdownMenu/Menu.d.ts +14 -0
- package/build/esm/components/DropdownMenu/Menu.js +1 -0
- package/build/esm/components/DropdownMenu/utils.d.ts +6 -0
- package/build/esm/components/DropdownMenu/utils.js +1 -0
- package/build/esm/components/Tooltip/TooltipContent.js +1 -1
- package/build/esm/components/Tooltip/utils.js +1 -1
- package/build/esm/web-tokens/assets/icons.json +1 -0
- package/build/esm/web-tokens/assets/icons16.json +1 -0
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),Object.defineProperty(exports,"TooltipContent",{enumerable:!0,get:function(){return TooltipContent}});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=_interop_require_default._(require("@emotion/styled")),_react1=require("@emotion/react"),_reactdom=require("react-dom"),_useDocument=require("../../shared/useDocument"),_useWindow=require("../../shared/useWindow"),_SubThemeProvider=require("../SubThemeProvider/SubThemeProvider"),_zindexjson=_interop_require_default._(require("../../web-tokens/_zindex.json")),_utils=require("./utils"),StyledContainer=(0,_styled.default)("div",{target:"ett48xg0",label:"StyledContainer"})(({theme,horizontalPlacement,verticalPlacement,maxWidth,contentPadding,hasInvertedSubTheme=!0})=>{let animationDistance="top"===verticalPlacement?`${_utils.ANIMATION_DISTANCE}px`:`-${_utils.ANIMATION_DISTANCE}px`,animation=(0,_react1.keyframes)({to:{opacity:1,transform:"center"===horizontalPlacement?`translate(-50%, ${animationDistance})`:`translateY(${animationDistance})`}}),contentPaddingMap={s:theme.variables.size.spacing.xxs,m:theme.variables.size.spacing.s},invertedSubThemeStyles={padding:`${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,...contentPadding&&{padding:contentPaddingMap[contentPadding]}};return{position:"absolute",zIndex:_zindexjson.default.tooltip.value,opacity:0,animation:`200ms ease-out forwards ${animation}`,maxWidth,width:"max-content",boxSizing:"border-box",backgroundColor:hasInvertedSubTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,borderRadius:hasInvertedSubTheme?theme.variables.size.borderRadius.xs:theme.variables.size.borderRadius.s,...hasInvertedSubTheme&&invertedSubThemeStyles,..."center"===horizontalPlacement&&{transform:"translate(-50%)"},...!hasInvertedSubTheme&&{":after":{content:'" "',position:"absolute",top:0,left:0,width:"100%",height:"100%",pointerEvents:"none",borderRadius:"inherit",boxShadow:theme.values.elevation[3]}}}},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tooltip/TooltipContent.tsx","sources":["src/components/Tooltip/TooltipContent.tsx"],"sourcesContent":["import type { MutableRefObject, ReactElement } from \"react\";\nimport React, {\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../web-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!hasInvertedSubTheme && {\n        \":after\": {\n          content: '\" \"',\n          position: \"absolute\",\n          top: 0,\n          left: 0,\n          width: \"100%\",\n          height: \"100%\",\n          pointerEvents: \"none\",\n          borderRadius: \"inherit\",\n          boxShadow: theme.values.elevation[3],\n        },\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      zIndex: 1,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation[3],\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"],"names":[],"mappings":"AA4EwB"} */"),StyledArrow=(0,_styled.default)("div",{target:"ett48xg1",label:"StyledArrow"})(({theme,hasInvertedSubTheme,verticalPlacement,horizontalPlacement,size=_utils.ARROW_SIZE})=>{let offset=(0,_utils.getArrowOffset)(size),adjustmentForShadow=hasInvertedSubTheme?0:1,arrowContainerHeight=size+_utils.DISTANCE_FROM_TRIGGER,arrowSideLength=Math.sqrt(size**2+(size+adjustmentForShadow)**2);return{position:"absolute",width:40,height:arrowContainerHeight,zIndex:1,overflow:"hidden",..."top"===verticalPlacement&&{top:`calc(100% - ${adjustmentForShadow}px)`},..."bottom"===verticalPlacement&&{top:`-${arrowContainerHeight-adjustmentForShadow}px`},..."center"===horizontalPlacement&&{left:"50%",transform:"translate(-50%)"},..."right"===horizontalPlacement&&{left:`${offset-(20-size)}px`},..."left"===horizontalPlacement&&{right:`${offset-(20-size)}px`},"&::after":{content:'" "',position:"absolute",top:"top"===verticalPlacement?0:"100%",left:"50%",width:arrowSideLength,height:arrowSideLength,backgroundColor:hasInvertedSubTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,transform:"translate(-50%, -50%) rotate(45deg)",...!hasInvertedSubTheme&&{boxShadow:theme.values.elevation[3]}}}},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tooltip/TooltipContent.tsx","sources":["src/components/Tooltip/TooltipContent.tsx"],"sourcesContent":["import type { MutableRefObject, ReactElement } from \"react\";\nimport React, {\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../web-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!hasInvertedSubTheme && {\n        \":after\": {\n          content: '\" \"',\n          position: \"absolute\",\n          top: 0,\n          left: 0,\n          width: \"100%\",\n          height: \"100%\",\n          pointerEvents: \"none\",\n          borderRadius: \"inherit\",\n          boxShadow: theme.values.elevation[3],\n        },\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      zIndex: 1,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation[3],\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"],"names":[],"mappings":"AA6JoB"} */"),initialStyle={top:0,left:0,verticalPlacement:"top",horizontalPlacement:"center"};let lastTooltipHideTimestamp=0;function TooltipContent({placement="auto",content,tooltipId,triggerRef,portalContainer,dataE2eTestId,dataDSId,isVisible,"aria-hidden":ariaHidden,role,tabIndex,contentPadding,maxWidth=224,hasInvertedSubTheme=!0,defaultVerticalPlacement,onTooltipPointerEnter,onTooltipPointerLeave,hideArrow=!1}){let animationDuration;let[style,setStyle]=(0,_react.useState)(initialStyle),tooltipRef=(0,_react.useRef)(null),document=(0,_useDocument.useDocument)(),window=(0,_useWindow.useWindow)(),arrowSize=(0,_react.useMemo)(()=>hideArrow?0:hasInvertedSubTheme?_utils.ARROW_SIZE:_utils.ARROW_SIZE_BIG,[hasInvertedSubTheme,hideArrow]),calculateStyle=(0,_react.useCallback)(()=>{triggerRef.current&&tooltipRef.current&&setStyle((0,_utils.getTooltipStyle)(placement,defaultVerticalPlacement,triggerRef,tooltipRef,document,window,arrowSize))},[triggerRef,tooltipRef,document,window,placement,arrowSize,defaultVerticalPlacement]);if((0,_react.useLayoutEffect)(()=>{isVisible&&calculateStyle()},[isVisible,calculateStyle,contentPadding,content]),(0,_react.useEffect)(()=>{let observer;return"undefined"!=typeof IntersectionObserver&&isVisible&&tooltipRef.current?((observer=new IntersectionObserver(entries=>{entries.forEach(()=>{calculateStyle()})},{threshold:.9})).observe(tooltipRef.current),window.addEventListener("resize",calculateStyle),window.addEventListener("scroll",calculateStyle,!0)):isVisible||(lastTooltipHideTimestamp=Date.now()),()=>{observer&&observer.disconnect(),window.removeEventListener("resize",calculateStyle),window.removeEventListener("scroll",calculateStyle)}},[isVisible,calculateStyle,window,tooltipRef]),!isVisible)return null;let tooltipElm=_react.default.createElement(StyledContainer,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":dataDSId,style:{top:style.top,left:style.left,animationDuration:(animationDuration="200ms",lastTooltipHideTimestamp&&Date.now()-lastTooltipHideTimestamp<700&&(animationDuration="0ms"),animationDuration)},ref:tooltipRef,id:tooltipId,role:role,"aria-hidden":ariaHidden,hasInvertedSubTheme:hasInvertedSubTheme,tabIndex:tabIndex,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,maxWidth:maxWidth,contentPadding:contentPadding,onPointerEnter:onTooltipPointerEnter,onPointerLeave:onTooltipPointerLeave},content,!hideArrow&&_react.default.createElement(StyledArrow,{"data-e2e-test-id":`${dataE2eTestId}_arrow`,hasInvertedSubTheme:hasInvertedSubTheme,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,size:arrowSize})),wrapperElm=hasInvertedSubTheme?_react.default.createElement(_SubThemeProvider.SubThemeProvider,{name:"inverted"},tooltipElm):tooltipElm;return(0,_reactdom.createPortal)(wrapperElm,portalContainer||document.body)}
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),Object.defineProperty(exports,"TooltipContent",{enumerable:!0,get:function(){return TooltipContent}});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=_interop_require_default._(require("@emotion/styled")),_react1=require("@emotion/react"),_reactdom=require("react-dom"),_useDocument=require("../../shared/useDocument"),_useWindow=require("../../shared/useWindow"),_SubThemeProvider=require("../SubThemeProvider/SubThemeProvider"),_zindexjson=_interop_require_default._(require("../../web-tokens/_zindex.json")),_utils=require("./utils"),StyledContainer=(0,_styled.default)("div",{target:"e8g5w0i0",label:"StyledContainer"})(({theme,horizontalPlacement,verticalPlacement,maxWidth,contentPadding,hasInvertedSubTheme=!0})=>{let animationDistance="top"===verticalPlacement?`${_utils.ANIMATION_DISTANCE}px`:`-${_utils.ANIMATION_DISTANCE}px`,animation=(0,_react1.keyframes)({to:{opacity:1,transform:"center"===horizontalPlacement?`translate(-50%, ${animationDistance})`:`translateY(${animationDistance})`}}),contentPaddingMap={s:theme.variables.size.spacing.xxs,m:theme.variables.size.spacing.s},invertedSubThemeStyles={padding:`${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,...contentPadding&&{padding:contentPaddingMap[contentPadding]}};return{position:"absolute",zIndex:_zindexjson.default.tooltip.value,opacity:0,animation:`200ms ease-out forwards ${animation}`,maxWidth,width:"max-content",boxSizing:"border-box",backgroundColor:hasInvertedSubTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,borderRadius:hasInvertedSubTheme?theme.variables.size.borderRadius.xs:theme.variables.size.borderRadius.s,...hasInvertedSubTheme&&invertedSubThemeStyles,..."center"===horizontalPlacement&&{transform:"translate(-50%)"},...!hasInvertedSubTheme&&{":after":{content:'" "',position:"absolute",top:0,left:0,width:"100%",height:"100%",pointerEvents:"none",borderRadius:"inherit",boxShadow:theme.values.elevation[3]}}}},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tooltip/TooltipContent.tsx","sources":["src/components/Tooltip/TooltipContent.tsx"],"sourcesContent":["import type { MutableRefObject, ReactElement } from \"react\";\nimport React, {\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../web-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!hasInvertedSubTheme && {\n        \":after\": {\n          content: '\" \"',\n          position: \"absolute\",\n          top: 0,\n          left: 0,\n          width: \"100%\",\n          height: \"100%\",\n          pointerEvents: \"none\",\n          borderRadius: \"inherit\",\n          boxShadow: theme.values.elevation[3],\n        },\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      zIndex: 1,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation[3],\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  useEffect(() => {\n    if (isVisible && tooltipRef.current) {\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"],"names":[],"mappings":"AA4EwB"} */"),StyledArrow=(0,_styled.default)("div",{target:"e8g5w0i1",label:"StyledArrow"})(({theme,hasInvertedSubTheme,verticalPlacement,horizontalPlacement,size=_utils.ARROW_SIZE})=>{let offset=(0,_utils.getArrowOffset)(size),adjustmentForShadow=hasInvertedSubTheme?0:1,arrowContainerHeight=size+_utils.DISTANCE_FROM_TRIGGER,arrowSideLength=Math.sqrt(size**2+(size+adjustmentForShadow)**2);return{position:"absolute",width:40,height:arrowContainerHeight,zIndex:1,overflow:"hidden",..."top"===verticalPlacement&&{top:`calc(100% - ${adjustmentForShadow}px)`},..."bottom"===verticalPlacement&&{top:`-${arrowContainerHeight-adjustmentForShadow}px`},..."center"===horizontalPlacement&&{left:"50%",transform:"translate(-50%)"},..."right"===horizontalPlacement&&{left:`${offset-(20-size)}px`},..."left"===horizontalPlacement&&{right:`${offset-(20-size)}px`},"&::after":{content:'" "',position:"absolute",top:"top"===verticalPlacement?0:"100%",left:"50%",width:arrowSideLength,height:arrowSideLength,backgroundColor:hasInvertedSubTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,transform:"translate(-50%, -50%) rotate(45deg)",...!hasInvertedSubTheme&&{boxShadow:theme.values.elevation[3]}}}},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tooltip/TooltipContent.tsx","sources":["src/components/Tooltip/TooltipContent.tsx"],"sourcesContent":["import type { MutableRefObject, ReactElement } from \"react\";\nimport React, {\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../web-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!hasInvertedSubTheme && {\n        \":after\": {\n          content: '\" \"',\n          position: \"absolute\",\n          top: 0,\n          left: 0,\n          width: \"100%\",\n          height: \"100%\",\n          pointerEvents: \"none\",\n          borderRadius: \"inherit\",\n          boxShadow: theme.values.elevation[3],\n        },\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      zIndex: 1,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation[3],\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  useEffect(() => {\n    if (isVisible && tooltipRef.current) {\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"],"names":[],"mappings":"AA6JoB"} */"),initialStyle={top:0,left:0,verticalPlacement:"top",horizontalPlacement:"center"};let lastTooltipHideTimestamp=0;function TooltipContent({placement="auto",content,tooltipId,triggerRef,portalContainer,dataE2eTestId,dataDSId,isVisible,"aria-hidden":ariaHidden,role,tabIndex,contentPadding,maxWidth=224,hasInvertedSubTheme=!0,defaultVerticalPlacement,onTooltipPointerEnter,onTooltipPointerLeave,hideArrow=!1}){let animationDuration;let[style,setStyle]=(0,_react.useState)(initialStyle),tooltipRef=(0,_react.useRef)(null),document=(0,_useDocument.useDocument)(),window=(0,_useWindow.useWindow)(),arrowSize=(0,_react.useMemo)(()=>hideArrow?0:hasInvertedSubTheme?_utils.ARROW_SIZE:_utils.ARROW_SIZE_BIG,[hasInvertedSubTheme,hideArrow]),calculateStyle=(0,_react.useCallback)(()=>{triggerRef.current&&tooltipRef.current&&setStyle((0,_utils.getTooltipStyle)(placement,defaultVerticalPlacement,triggerRef,tooltipRef,document,window,arrowSize))},[triggerRef,tooltipRef,document,window,placement,arrowSize,defaultVerticalPlacement]);if((0,_react.useLayoutEffect)(()=>{isVisible&&calculateStyle()},[isVisible,calculateStyle,contentPadding,content]),(0,_react.useEffect)(()=>(isVisible&&tooltipRef.current?(window.addEventListener("resize",calculateStyle),window.addEventListener("scroll",calculateStyle,!0)):isVisible||(lastTooltipHideTimestamp=Date.now()),()=>{window.removeEventListener("resize",calculateStyle),window.removeEventListener("scroll",calculateStyle)}),[isVisible,calculateStyle,window,tooltipRef]),!isVisible)return null;let tooltipElm=_react.default.createElement(StyledContainer,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":dataDSId,style:{top:style.top,left:style.left,animationDuration:(animationDuration="200ms",lastTooltipHideTimestamp&&Date.now()-lastTooltipHideTimestamp<700&&(animationDuration="0ms"),animationDuration)},ref:tooltipRef,id:tooltipId,role:role,"aria-hidden":ariaHidden,hasInvertedSubTheme:hasInvertedSubTheme,tabIndex:tabIndex,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,maxWidth:maxWidth,contentPadding:contentPadding,onPointerEnter:onTooltipPointerEnter,onPointerLeave:onTooltipPointerLeave},content,!hideArrow&&_react.default.createElement(StyledArrow,{"data-e2e-test-id":`${dataE2eTestId}_arrow`,hasInvertedSubTheme:hasInvertedSubTheme,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,size:arrowSize})),wrapperElm=hasInvertedSubTheme?_react.default.createElement(_SubThemeProvider.SubThemeProvider,{name:"inverted"},tooltipElm):tooltipElm;return(0,_reactdom.createPortal)(wrapperElm,portalContainer||document.body)}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),function(target,all){for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:all[name]})}(exports,{ANIMATION_DISTANCE:function(){return ANIMATION_DISTANCE},ARROW_SIZE:function(){return ARROW_SIZE},ARROW_SIZE_BIG:function(){return ARROW_SIZE_BIG},DISTANCE_FROM_TRIGGER:function(){return DISTANCE_FROM_TRIGGER},getArrowOffset:function(){return getArrowOffset},getTooltipStyle:function(){return getTooltipStyle}});const DISTANCE_FROM_TRIGGER=4,ANIMATION_DISTANCE=8,ARROW_SIZE=6,ARROW_SIZE_BIG=8,getArrowOffset=size=>2*size;function getTooltipStyle(placement,defaultVerticalPlacement="top",triggerRef,tooltipRef,document,window,arrowSize){let triggerRect=triggerRef.current.getBoundingClientRect(),tooltipRect=tooltipRef.current.getBoundingClientRect(),viewportWidth=document.documentElement.clientWidth,viewportHeight=document.documentElement.clientHeight,top=0,left=0,verticalPlacement=defaultVerticalPlacement,horizontalPlacement="center",arrowOffset=getArrowOffset(arrowSize),tooltipMargin=arrowSize+ANIMATION_DISTANCE+DISTANCE_FROM_TRIGGER;switch(placement){case"top":case"bottom":verticalPlacement=placement;break;case"top-left":case"top-right":case"bottom-left":case"bottom-right":{let placements=placement.split("-");verticalPlacement=placements[0],horizontalPlacement=placements[1];break}default:"top"===verticalPlacement?triggerRect.top<tooltipRect.height&&(verticalPlacement="bottom"):"bottom"===verticalPlacement&&triggerRect.bottom+tooltipRect.height
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),function(target,all){for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:all[name]})}(exports,{ANIMATION_DISTANCE:function(){return ANIMATION_DISTANCE},ARROW_SIZE:function(){return ARROW_SIZE},ARROW_SIZE_BIG:function(){return ARROW_SIZE_BIG},DISTANCE_FROM_TRIGGER:function(){return DISTANCE_FROM_TRIGGER},getArrowOffset:function(){return getArrowOffset},getTooltipStyle:function(){return getTooltipStyle}});const DISTANCE_FROM_TRIGGER=4,ANIMATION_DISTANCE=8,ARROW_SIZE=6,ARROW_SIZE_BIG=8,getArrowOffset=size=>2*size;function getTooltipStyle(placement,defaultVerticalPlacement="top",triggerRef,tooltipRef,document,window,arrowSize){let triggerRect=triggerRef.current.getBoundingClientRect(),tooltipRect=tooltipRef.current.getBoundingClientRect(),viewportWidth=document.documentElement.clientWidth,viewportHeight=document.documentElement.clientHeight,top=0,left=0,verticalPlacement=defaultVerticalPlacement,horizontalPlacement="center",arrowOffset=getArrowOffset(arrowSize),tooltipMargin=arrowSize+ANIMATION_DISTANCE+DISTANCE_FROM_TRIGGER;switch(placement){case"top":case"bottom":verticalPlacement=placement;break;case"top-left":case"top-right":case"bottom-left":case"bottom-right":{let placements=placement.split("-");verticalPlacement=placements[0],horizontalPlacement=placements[1];break}default:"top"===verticalPlacement?triggerRect.top<tooltipRect.height&&(verticalPlacement="bottom"):"bottom"===verticalPlacement&&triggerRect.bottom+tooltipRect.height>=viewportHeight&&triggerRect.top>tooltipRect.height&&(verticalPlacement="top"),triggerRect.left>=tooltipRect.width/2?triggerRect.left+triggerRect.width/2+tooltipRect.width/2>=viewportWidth&&(horizontalPlacement="left"):horizontalPlacement="right"}switch(top="top"===verticalPlacement?triggerRect.top-tooltipRect.height-tooltipMargin:triggerRect.bottom+tooltipMargin,horizontalPlacement){case"left":left=triggerRect.left+triggerRect.width/2-tooltipRect.width+arrowOffset+arrowSize;break;case"right":left=triggerRect.left+triggerRect.width/2-arrowOffset-arrowSize;break;default:left=triggerRect.left+triggerRect.width/2}return{top:top+window.scrollY,left:left+window.scrollX,horizontalPlacement,verticalPlacement}}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"alert-circle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-alert-circle\"><circle cx=\"12\" cy=\"12\" r=\"10\"></circle><line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\"></line><line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\"></line></svg>",
|
|
3
3
|
"alert-triangle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-alert-triangle\"><path d=\"M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z\"></path><line x1=\"12\" y1=\"9\" x2=\"12\" y2=\"13\"></line><line x1=\"12\" y1=\"17\" x2=\"12.01\" y2=\"17\"></line></svg>",
|
|
4
|
+
"alert-octagon": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-alert-octagon\"><polygon points=\"7.86 2 16.14 2 22 7.86 22 16.14 16.14 22 7.86 22 2 16.14 2 7.86 7.86 2\"></polygon><line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\"></line><line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\"></line></svg>",
|
|
4
5
|
"align-left": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-align-left\"><line x1=\"17\" y1=\"10\" x2=\"3\" y2=\"10\"></line><line x1=\"21\" y1=\"6\" x2=\"3\" y2=\"6\"></line><line x1=\"21\" y1=\"14\" x2=\"3\" y2=\"14\"></line><line x1=\"17\" y1=\"18\" x2=\"3\" y2=\"18\"></line></svg>",
|
|
5
6
|
"anamnesis": "<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n<path d=\"M3 11.8437C3 11.2915 3.44772 10.8437 4 10.8437H11C11.5523 10.8437 12 10.396 12 9.84374V9C12 8.44772 12.4477 8 13 8H20C20.5523 8 21 8.44772 21 9V21C21 21.5523 20.5523 22 20 22H4C3.44772 22 3 21.5523 3 21V11.8437Z\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n<path d=\"M19 9V2H5V11\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n<path d=\"M8 5L16 5\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n<path d=\"M8 8H9\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n</svg>\n",
|
|
6
7
|
"apple": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" viewBox=\"0 0 24 24\">\n <path fill=\"currentColor\" fill-rule=\"evenodd\"\n d=\"M16.749 0c.214 1.458-.379 2.886-1.162 3.896-.837 1.084-2.28 1.922-3.679 1.879-.255-1.396.398-2.834 1.194-3.8C13.974.906 15.469.088 16.749 0zm4.201 20.518c.721-1.105.99-1.662 1.55-2.91-4.07-1.547-4.723-7.335-.694-9.556-1.23-1.54-2.956-2.434-4.586-2.434-1.174 0-1.979.306-2.71.585-.61.232-1.169.445-1.849.445-.734 0-1.385-.233-2.066-.477-.748-.269-1.533-.55-2.508-.55-1.83 0-3.776 1.117-5.01 3.028C1.34 11.34 1.636 16.4 4.45 20.71c1.006 1.542 2.35 3.275 4.107 3.29.73.008 1.215-.21 1.74-.445.601-.27 1.255-.563 2.386-.569 1.138-.007 1.78.29 2.374.564.511.236.986.454 1.71.447 1.758-.014 3.176-1.935 4.182-3.477z\"\n clip-rule=\"evenodd\" />\n</svg>",
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"alert-circle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"16\" height=\"16\" fill=\"none\" viewBox=\"0 0 16 16\">\n <circle cx=\"8\" cy=\"8\" r=\"7\" stroke=\"currentColor\" stroke-width=\"2\" />\n <circle cx=\"8\" cy=\"11\" r=\"1\" fill=\"currentColor\" />\n <path stroke=\"currentColor\" stroke-linecap=\"round\" stroke-width=\"2\" d=\"M8 5v3\" />\n</svg>",
|
|
3
3
|
"alert-triangle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"16\" height=\"16\" fill=\"none\" viewBox=\"0 0 16 16\">\n <path stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"\n d=\"M6.854 1.691L1.18 11.846c-.239.442-.24.986-.004 1.43.236.442.673.718 1.15.724h11.349a1.328 1.328 0 001.15-.725 1.526 1.526 0 00-.005-1.43L9.146 1.692C8.903 1.262 8.468 1 8 1s-.903.262-1.146.691z\"\n clip-rule=\"evenodd\" />\n <g fill=\"currentColor\">\n <path d=\"M9 6a1 1 0 00-2 0zM7 8a1 1 0 002 0zm0-2v2h2V6z\" />\n <rect width=\"2\" height=\"2\" x=\"7\" y=\"10\" rx=\"1\" />\n </g>\n</svg>",
|
|
4
|
+
"alert-octagon": "<svg fill=\"none\" height=\"16\" viewBox=\"0 0 16 16\" width=\"16\"\n xmlns=\"http://www.w3.org/2000/svg\"\n xmlns:xlink=\"http://www.w3.org/1999/xlink\">\n <clipPath id=\"a\">\n <path d=\"m0 0h16v16h-16z\"/>\n </clipPath>\n <g clip-path=\"url(#a)\" fill=\"currentColor\">\n <path d=\"m8 4c.55229 0 1 .44772 1 1v3c0 .55228-.44771 1-1 1-.55228 0-1-.44772-1-1v-3c0-.55228.44772-1 1-1z\"/>\n <path d=\"m8 10c-.55228 0-1 .4477-1 1s.44772 1 1 1c.55229 0 1-.4477 1-1s-.44771-1-1-1z\"/>\n <path clip-rule=\"evenodd\" d=\"m4.39489.292893c.18097-.180964.43097-.292893.70711-.292893h5.796c.2652 0 .5196.105357.7071.292893l4.102 4.101997c.1875.18754.2929.44189.2929.70711v5.796c0 .2652-.1054.5196-.2929.7071l-4.102 4.102c-.1875.1875-.4419.2929-.7071.2929h-5.796c-.26522 0-.51957-.1054-.70711-.2929l-4.101997-4.102c-.187536-.1875-.292893-.4419-.292893-.7071v-5.796c0-.26522.105357-.51957.292893-.70711zm1.12132 1.707107-3.51621 3.51621v4.96759l3.51621 3.5162h4.96759l3.5162-3.5162v-4.96759l-3.5162-3.51621z\" fill-rule=\"evenodd\"/>\n </g>\n</svg>\n",
|
|
4
5
|
"align-left": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"16\" height=\"16\" fill=\"none\" viewBox=\"0 0 16 16\">\n <g stroke=\"currentColor\" stroke-linecap=\"round\" stroke-width=\"2\">\n <path d=\"M3 8h7M3 12h10M3 4h11\" />\n </g>\n</svg>",
|
|
5
6
|
"anamnesis": "<svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n<path d=\"M1 6.88641C1 6.33412 1.44772 5.88641 2 5.88641H6.86364C7.41592 5.88641 7.86364 5.43869 7.86364 4.88641V4.81824C7.86364 4.26595 8.31135 3.81824 8.86364 3.81824H13.7273C14.2796 3.81824 14.7273 4.26595 14.7273 4.81824V14.0001C14.7273 14.5523 14.2796 15.0001 13.7273 15.0001H2C1.44772 15.0001 1 14.5523 1 14.0001V6.88641Z\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n<path d=\"M13.3546 4.45455V1H2.3728V6.72727\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n</svg>\n",
|
|
6
7
|
"apple": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"16\" height=\"16\" fill=\"none\" viewBox=\"0 0 16 16\">\n <path fill=\"currentColor\" fill-rule=\"evenodd\"\n d=\"M11.166 0c.143.972-.253 1.924-.774 2.598-.559.722-1.52 1.28-2.453 1.252-.17-.93.265-1.89.796-2.534C9.316.605 10.313.06 11.165 0zm2.8 13.679c.481-.737.66-1.108 1.034-1.94-2.714-1.032-3.149-4.89-.463-6.37-.82-1.028-1.97-1.624-3.057-1.624-.783 0-1.32.205-1.807.39-.406.155-.779.297-1.232.297-.49 0-.923-.155-1.377-.318-.5-.18-1.023-.367-1.673-.367-1.22 0-2.517.745-3.34 2.02-1.157 1.793-.96 5.166.916 8.039.671 1.028 1.567 2.184 2.739 2.194.486.005.81-.14 1.16-.297.4-.18.836-.375 1.59-.38.758-.004 1.187.194 1.583.376.34.157.657.303 1.14.298 1.172-.01 2.117-1.29 2.788-2.318z\"\n clip-rule=\"evenodd\" />\n</svg>",
|
|
@@ -39,20 +39,26 @@ export type DataTableProps = {
|
|
|
39
39
|
*/
|
|
40
40
|
isEmpty?: boolean;
|
|
41
41
|
/**
|
|
42
|
-
* Is first column sticky on scroll
|
|
42
|
+
* Is first column sticky on horizontal scroll
|
|
43
43
|
*/
|
|
44
44
|
isFirstColumnSticky?: boolean;
|
|
45
|
+
/**
|
|
46
|
+
* Is first column sticky on horizontal scroll
|
|
47
|
+
*/
|
|
48
|
+
isLastColumnSticky?: boolean;
|
|
45
49
|
"data-e2e-test-id"?: string;
|
|
46
50
|
/** Column most recently used to sort data */
|
|
47
51
|
currentlySortedByColumn?: string;
|
|
48
|
-
/**
|
|
49
|
-
* Callback to handle sorting by column
|
|
50
|
-
*/
|
|
51
52
|
/**
|
|
52
53
|
* Empty cell content.
|
|
53
54
|
*/
|
|
54
55
|
emptyCellContent?: string;
|
|
56
|
+
/** Define a vertically scrollable table with max height */
|
|
57
|
+
maxHeight?: string | MQ<string>;
|
|
58
|
+
/**
|
|
59
|
+
* Callback to handle sorting by column
|
|
60
|
+
*/
|
|
55
61
|
onSort?: (columnName: string, desiredSortDirection: DataTableColumn["sortDirection"]) => void;
|
|
56
62
|
};
|
|
57
|
-
export declare function BaseDataTable({ caption, columns, rows, footer, currentlySortedByColumn, isFirstColumnSticky, "data-e2e-test-id": dataE2eTestId, isLoading, loadingStateScreenReaderText, isEmpty, children, emptyTableContentHeight, layout, width, bodyCellVerticalPadding, emptyCellContent, onSort, }: React.PropsWithChildren<DataTableProps>): React.ReactElement;
|
|
58
|
-
export declare function DataTable({ caption, columns, rows, footer, currentlySortedByColumn, isFirstColumnSticky, "data-e2e-test-id": dataE2eTestId, isLoading, loadingStateScreenReaderText, isEmpty, children, emptyTableContentHeight, layout, width, bodyCellVerticalPadding, emptyCellContent, onSort, }: React.PropsWithChildren<DataTableProps>): React.ReactElement;
|
|
63
|
+
export declare function BaseDataTable({ caption, columns, rows, footer, currentlySortedByColumn, isFirstColumnSticky, isLastColumnSticky, "data-e2e-test-id": dataE2eTestId, isLoading, loadingStateScreenReaderText, isEmpty, children, emptyTableContentHeight, layout, width, bodyCellVerticalPadding, emptyCellContent, maxHeight, onSort, }: React.PropsWithChildren<DataTableProps>): React.ReactElement;
|
|
64
|
+
export declare function DataTable({ caption, columns, rows, footer, currentlySortedByColumn, isFirstColumnSticky, isLastColumnSticky, "data-e2e-test-id": dataE2eTestId, isLoading, loadingStateScreenReaderText, isEmpty, children, emptyTableContentHeight, layout, width, bodyCellVerticalPadding, emptyCellContent, maxHeight, onSort, }: React.PropsWithChildren<DataTableProps>): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useEffect,useState,useRef}from"react";import styled from"@emotion/styled";import TableBody from"./TableBody";import TableHeader from"./TableHeader";import{Container}from"../Container/Container";import{ScreenReaderText}from"../../shared/ScreenReaderText";import{mqValue}from"../../shared/mediaQueries";let TableContainer=styled("div",{target:"erdx5by0",label:"TableContainer"})(({theme,footer})=>({overflow:"auto",backgroundColor:theme.values.color.background.primary.default,borderRadius:"inherit",...footer&&{borderBottomLeftRadius:0,borderBottomRightRadius:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on scroll\n   */\n  isFirstColumnSticky?: boolean;\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Callback to handle sorting by column\n   */\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\">;\n\nconst TableContainer = styled.div<TableContainerProps>(({ theme, footer }) => ({\n  overflow: \"auto\",\n  backgroundColor: theme.values.color.background.primary.default,\n  borderRadius: \"inherit\",\n  ...(footer && {\n    borderBottomLeftRadius: 0,\n    borderBottomRightRadius: 0,\n  }),\n}));\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolled, setIsScrolled] = useState(false);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current &&\n      tableRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach((entry) => {\n            setIsScrolled(!entry.isIntersecting);\n          });\n        },\n        {\n          root: containerRef.current,\n          threshold: 1,\n        }\n      );\n\n      observer.observe(tableRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, []);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer ref={containerRef} footer={footer}>\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolled={isScrolled}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolled={isScrolled}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AAoFuB"} */"),StyledTable=styled("table",{target:"erdx5by1",label:"StyledTable"})(({layout,tableWidth})=>({borderSpacing:0,tableLayout:layout,...mqValue({width:tableWidth})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on scroll\n   */\n  isFirstColumnSticky?: boolean;\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Callback to handle sorting by column\n   */\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\">;\n\nconst TableContainer = styled.div<TableContainerProps>(({ theme, footer }) => ({\n  overflow: \"auto\",\n  backgroundColor: theme.values.color.background.primary.default,\n  borderRadius: \"inherit\",\n  ...(footer && {\n    borderBottomLeftRadius: 0,\n    borderBottomRightRadius: 0,\n  }),\n}));\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolled, setIsScrolled] = useState(false);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current &&\n      tableRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach((entry) => {\n            setIsScrolled(!entry.isIntersecting);\n          });\n        },\n        {\n          root: containerRef.current,\n          threshold: 1,\n        }\n      );\n\n      observer.observe(tableRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, []);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer ref={containerRef} footer={footer}>\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolled={isScrolled}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolled={isScrolled}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AAkGoB"} */"),StyledFooter=styled("div",{target:"erdx5by2",label:"StyledFooter"})(({theme})=>({padding:theme.variables.size.spacing.xs,backgroundColor:theme.values.color.background.primary.default,borderBottomLeftRadius:"inherit",borderBottomRightRadius:"inherit"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on scroll\n   */\n  isFirstColumnSticky?: boolean;\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Callback to handle sorting by column\n   */\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\">;\n\nconst TableContainer = styled.div<TableContainerProps>(({ theme, footer }) => ({\n  overflow: \"auto\",\n  backgroundColor: theme.values.color.background.primary.default,\n  borderRadius: \"inherit\",\n  ...(footer && {\n    borderBottomLeftRadius: 0,\n    borderBottomRightRadius: 0,\n  }),\n}));\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolled, setIsScrolled] = useState(false);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current &&\n      tableRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach((entry) => {\n            setIsScrolled(!entry.isIntersecting);\n          });\n        },\n        {\n          root: containerRef.current,\n          threshold: 1,\n        }\n      );\n\n      observer.observe(tableRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, []);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer ref={containerRef} footer={footer}>\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolled={isScrolled}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolled={isScrolled}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AA4GqB"} */");export function BaseDataTable({caption,columns,rows=[],footer,currentlySortedByColumn,isFirstColumnSticky=!0,"data-e2e-test-id":dataE2eTestId,isLoading=!1,loadingStateScreenReaderText="Loading",isEmpty=!1,children,emptyTableContentHeight="15rem",layout="auto",width="100%",bodyCellVerticalPadding="m",emptyCellContent="--",onSort}){let[isScrolled,setIsScrolled]=useState(!1),tableRef=useRef(null),containerRef=useRef(null);useEffect(()=>{let observer;return"undefined"!=typeof IntersectionObserver&&containerRef&&containerRef.current&&tableRef.current&&(observer=new IntersectionObserver(entries=>{entries.forEach(entry=>{setIsScrolled(!entry.isIntersecting)})},{root:containerRef.current,threshold:1})).observe(tableRef.current),()=>{observer&&observer.disconnect()}},[]);let footerElm=footer?React.createElement(StyledFooter,null,footer):null;return React.createElement(React.Fragment,null,React.createElement(TableContainer,{ref:containerRef,footer:footer},React.createElement(StyledTable,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"DataTable",ref:tableRef,layout:layout,tableWidth:width},React.createElement(ScreenReaderText,{as:"caption"},caption),React.createElement(TableHeader,{columns:columns,isFirstColumnSticky:isFirstColumnSticky,currentlySortedByColumn:currentlySortedByColumn,isTableScrolled:isScrolled,onSort:onSort}),React.createElement(TableBody,{columns:columns,rows:rows,bodyCellVerticalPadding:bodyCellVerticalPadding,isTableScrolled:isScrolled,isEmpty:isEmpty,isLoading:isLoading,loadingStateScreenReaderText:loadingStateScreenReaderText,emptyTableContentHeight:emptyTableContentHeight,footer:footer,isFirstColumnSticky:isFirstColumnSticky,emptyCellContent:emptyCellContent},children))),footerElm)}export function DataTable({caption,columns,rows=[],footer,currentlySortedByColumn,isFirstColumnSticky=!0,"data-e2e-test-id":dataE2eTestId,isLoading=!1,loadingStateScreenReaderText="Loading",isEmpty=!1,children,emptyTableContentHeight="15rem",layout="auto",width="100%",bodyCellVerticalPadding="m",emptyCellContent="--",onSort}){return React.createElement(Container,{elevation:1},React.createElement(BaseDataTable,{caption:caption,columns:columns,rows:rows,footer:footer,currentlySortedByColumn:currentlySortedByColumn,isFirstColumnSticky:isFirstColumnSticky,"data-e2e-test-id":dataE2eTestId,isLoading:isLoading,loadingStateScreenReaderText:loadingStateScreenReaderText,isEmpty:isEmpty,emptyTableContentHeight:emptyTableContentHeight,layout:layout,width:width,bodyCellVerticalPadding:bodyCellVerticalPadding,emptyCellContent:emptyCellContent,onSort:onSort},children))}
|
|
1
|
+
import React,{useEffect,useState,useRef,useCallback}from"react";import styled from"@emotion/styled";import TableBody from"./TableBody";import TableHeader from"./TableHeader";import{Container}from"../Container/Container";import{ScreenReaderText}from"../../shared/ScreenReaderText";import{mqValue}from"../../shared/mediaQueries";function getTableScrolledToRight(elm){let elmRect=elm.getBoundingClientRect();return elm.scrollLeft+elmRect.width>=elm.scrollWidth}let TableContainer=styled("div",{target:"ecq7ipt0",label:"TableContainer"})(({theme,footer,maxHeight})=>({overflow:"auto",backgroundColor:theme.values.color.background.primary.default,borderRadius:"inherit",...mqValue({maxHeight}),...footer&&{borderBottomLeftRadius:0,borderBottomRightRadius:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef, useCallback } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isFirstColumnSticky?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isLastColumnSticky?: boolean;\n\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  /** Define a vertically scrollable table with max height */\n  maxHeight?: string | MQ<string>;\n\n  /**\n   * Callback to handle sorting by column\n   */\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\nfunction getTableScrolledToRight(elm: HTMLDivElement) {\n  const elmRect = elm.getBoundingClientRect();\n\n  return elm.scrollLeft + elmRect.width >= elm.scrollWidth;\n}\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\" | \"maxHeight\">;\n\nconst TableContainer = styled.div<TableContainerProps>(\n  ({ theme, footer, maxHeight }) => ({\n    overflow: \"auto\",\n    backgroundColor: theme.values.color.background.primary.default,\n    borderRadius: \"inherit\",\n\n    ...mqValue({\n      maxHeight,\n    }),\n\n    ...(footer && {\n      borderBottomLeftRadius: 0,\n      borderBottomRightRadius: 0,\n    }),\n  })\n);\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolledToRight, setIsScrolledToRight] =\n    useState(isLastColumnSticky);\n  const [isScrolledToLeft, setIsScrolledToLeft] = useState(isFirstColumnSticky);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  const handleScroll = useCallback((evt) => {\n    const elm = evt.target;\n\n    setIsScrolledToLeft(elm.scrollLeft === 0);\n    setIsScrolledToRight(getTableScrolledToRight(elm));\n  }, []);\n\n  useEffect(() => {\n    let resizeObserver: ResizeObserver;\n\n    if (\n      typeof ResizeObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current\n    ) {\n      resizeObserver = new ResizeObserver((entries) => {\n        entries.forEach(() => {\n          setIsScrolledToLeft(containerRef.current.scrollLeft === 0);\n          setIsScrolledToRight(getTableScrolledToRight(containerRef.current));\n        });\n      });\n\n      resizeObserver.observe(containerRef.current);\n    }\n\n    return () => {\n      if (resizeObserver) {\n        resizeObserver.disconnect();\n      }\n    };\n  }, [containerRef]);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer\n        ref={containerRef}\n        footer={footer}\n        maxHeight={maxHeight}\n        onScroll={handleScroll}\n      >\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            maxHeight={maxHeight}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        isLastColumnSticky={isLastColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        maxHeight={maxHeight}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AAkGuB"} */"),StyledTable=styled("table",{target:"ecq7ipt1",label:"StyledTable"})(({layout,tableWidth})=>({borderSpacing:0,tableLayout:layout,...mqValue({width:tableWidth})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef, useCallback } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isFirstColumnSticky?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isLastColumnSticky?: boolean;\n\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  /** Define a vertically scrollable table with max height */\n  maxHeight?: string | MQ<string>;\n\n  /**\n   * Callback to handle sorting by column\n   */\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\nfunction getTableScrolledToRight(elm: HTMLDivElement) {\n  const elmRect = elm.getBoundingClientRect();\n\n  return elm.scrollLeft + elmRect.width >= elm.scrollWidth;\n}\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\" | \"maxHeight\">;\n\nconst TableContainer = styled.div<TableContainerProps>(\n  ({ theme, footer, maxHeight }) => ({\n    overflow: \"auto\",\n    backgroundColor: theme.values.color.background.primary.default,\n    borderRadius: \"inherit\",\n\n    ...mqValue({\n      maxHeight,\n    }),\n\n    ...(footer && {\n      borderBottomLeftRadius: 0,\n      borderBottomRightRadius: 0,\n    }),\n  })\n);\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolledToRight, setIsScrolledToRight] =\n    useState(isLastColumnSticky);\n  const [isScrolledToLeft, setIsScrolledToLeft] = useState(isFirstColumnSticky);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  const handleScroll = useCallback((evt) => {\n    const elm = evt.target;\n\n    setIsScrolledToLeft(elm.scrollLeft === 0);\n    setIsScrolledToRight(getTableScrolledToRight(elm));\n  }, []);\n\n  useEffect(() => {\n    let resizeObserver: ResizeObserver;\n\n    if (\n      typeof ResizeObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current\n    ) {\n      resizeObserver = new ResizeObserver((entries) => {\n        entries.forEach(() => {\n          setIsScrolledToLeft(containerRef.current.scrollLeft === 0);\n          setIsScrolledToRight(getTableScrolledToRight(containerRef.current));\n        });\n      });\n\n      resizeObserver.observe(containerRef.current);\n    }\n\n    return () => {\n      if (resizeObserver) {\n        resizeObserver.disconnect();\n      }\n    };\n  }, [containerRef]);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer\n        ref={containerRef}\n        footer={footer}\n        maxHeight={maxHeight}\n        onScroll={handleScroll}\n      >\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            maxHeight={maxHeight}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        isLastColumnSticky={isLastColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        maxHeight={maxHeight}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AAuHoB"} */"),StyledFooter=styled("div",{target:"ecq7ipt2",label:"StyledFooter"})(({theme})=>({padding:theme.variables.size.spacing.xs,backgroundColor:theme.values.color.background.primary.default,borderBottomLeftRadius:"inherit",borderBottomRightRadius:"inherit"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/DataTable.tsx","sources":["src/components/DataTable/DataTable.tsx"],"sourcesContent":["import React, { useEffect, useState, useRef, useCallback } from \"react\";\nimport styled from \"@emotion/styled\";\nimport TableBody from \"./TableBody\";\nimport TableHeader from \"./TableHeader\";\nimport { Container } from \"../Container/Container\";\nimport type { DataTableColumn, DataTableRow } from \"./types\";\nimport { ScreenReaderText } from \"../../shared/ScreenReaderText\";\nimport { mqValue } from \"../../shared/mediaQueries\";\nimport type { MQ } from \"../../types\";\n\nexport type DataTableProps = {\n  /** Screen reader text for table caption */\n  caption: string;\n\n  /** CSS table layout. In 'auto' layout, columns are sized according to content width. In 'fixed' layout, columns can be provided a fixed or percentage width. */\n  layout?: \"auto\" | \"fixed\";\n\n  /** Table width */\n  width?: string | MQ<string>;\n\n  /** Vertical padding on row cells to set table density, condensed (s), default (m), comfortable (l) */\n  bodyCellVerticalPadding?: \"s\" | \"m\" | \"l\" | MQ<\"s\" | \"m\" | \"l\">;\n\n  /** Height of empty/error or loading table content */\n  emptyTableContentHeight?: string | MQ<string>;\n\n  /** Footer element */\n  footer?: React.ReactElement;\n\n  /**\n   * Meta data for columns\n   * @param DataTableColumn[].name - Column name\n   * @param DataTableColumn[].label - Column label for display\n   * @param DataTableColumn[].align - Align cell content, 'left' | 'right' | 'center'\n   * @param DataTableColumn[].width - Column width\n   * @param DataTableColumn[].renderCell - Callback to render custom cell content.\n   * @param DataTableColumn[].isSortable - Is table sortable by column\n   * @param DataTableColumn[].sortDirection - Sorted as 'asc' | 'desc'\n   */\n  columns: DataTableColumn[];\n\n  /**\n   * Table content as an array of objects with values for each column\n   */\n  rows?: DataTableRow[];\n\n  /**\n   * Is loading data. You can display custom loading content with children prop.\n   */\n  isLoading?: boolean;\n\n  /** Screen reader text for loading spinner */\n  loadingStateScreenReaderText?: string;\n\n  /** Is empty or in error state. You can display custom empty/error content with children prop.\n   */\n  isEmpty?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isFirstColumnSticky?: boolean;\n\n  /**\n   * Is first column sticky on horizontal scroll\n   */\n  isLastColumnSticky?: boolean;\n\n  \"data-e2e-test-id\"?: string;\n\n  /**  Column most recently used to sort data */\n  currentlySortedByColumn?: string;\n\n  /**\n   * Empty cell content.\n   */\n  emptyCellContent?: string;\n\n  /** Define a vertically scrollable table with max height */\n  maxHeight?: string | MQ<string>;\n\n  /**\n   * Callback to handle sorting by column\n   */\n  onSort?: (\n    columnName: string,\n    desiredSortDirection: DataTableColumn[\"sortDirection\"]\n  ) => void;\n};\n\nfunction getTableScrolledToRight(elm: HTMLDivElement) {\n  const elmRect = elm.getBoundingClientRect();\n\n  return elm.scrollLeft + elmRect.width >= elm.scrollWidth;\n}\n\ntype TableContainerProps = Pick<DataTableProps, \"footer\" | \"maxHeight\">;\n\nconst TableContainer = styled.div<TableContainerProps>(\n  ({ theme, footer, maxHeight }) => ({\n    overflow: \"auto\",\n    backgroundColor: theme.values.color.background.primary.default,\n    borderRadius: \"inherit\",\n\n    ...mqValue({\n      maxHeight,\n    }),\n\n    ...(footer && {\n      borderBottomLeftRadius: 0,\n      borderBottomRightRadius: 0,\n    }),\n  })\n);\n\ntype StyledTableProps = Pick<DataTableProps, \"layout\"> & {\n  tableWidth: DataTableProps[\"width\"];\n};\n\nconst StyledTable = styled.table<StyledTableProps>(\n  ({ layout, tableWidth }) => ({\n    borderSpacing: 0,\n    tableLayout: layout,\n    ...mqValue({\n      width: tableWidth,\n    }),\n  })\n);\n\nconst StyledFooter = styled.div(({ theme }) => ({\n  padding: theme.variables.size.spacing.xs,\n  backgroundColor: theme.values.color.background.primary.default,\n  borderBottomLeftRadius: \"inherit\",\n  borderBottomRightRadius: \"inherit\",\n}));\n\nexport function BaseDataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  const [isScrolledToRight, setIsScrolledToRight] =\n    useState(isLastColumnSticky);\n  const [isScrolledToLeft, setIsScrolledToLeft] = useState(isFirstColumnSticky);\n  const tableRef = useRef<HTMLTableElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n\n  const handleScroll = useCallback((evt) => {\n    const elm = evt.target;\n\n    setIsScrolledToLeft(elm.scrollLeft === 0);\n    setIsScrolledToRight(getTableScrolledToRight(elm));\n  }, []);\n\n  useEffect(() => {\n    let resizeObserver: ResizeObserver;\n\n    if (\n      typeof ResizeObserver !== \"undefined\" &&\n      containerRef &&\n      containerRef.current\n    ) {\n      resizeObserver = new ResizeObserver((entries) => {\n        entries.forEach(() => {\n          setIsScrolledToLeft(containerRef.current.scrollLeft === 0);\n          setIsScrolledToRight(getTableScrolledToRight(containerRef.current));\n        });\n      });\n\n      resizeObserver.observe(containerRef.current);\n    }\n\n    return () => {\n      if (resizeObserver) {\n        resizeObserver.disconnect();\n      }\n    };\n  }, [containerRef]);\n\n  const footerElm = footer ? <StyledFooter>{footer}</StyledFooter> : null;\n\n  return (\n    <>\n      <TableContainer\n        ref={containerRef}\n        footer={footer}\n        maxHeight={maxHeight}\n        onScroll={handleScroll}\n      >\n        <StyledTable\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"DataTable\"\n          ref={tableRef}\n          layout={layout}\n          tableWidth={width}\n        >\n          <ScreenReaderText as=\"caption\">{caption}</ScreenReaderText>\n          <TableHeader\n            columns={columns}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            currentlySortedByColumn={currentlySortedByColumn}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            maxHeight={maxHeight}\n            onSort={onSort}\n          />\n          <TableBody\n            columns={columns}\n            rows={rows}\n            bodyCellVerticalPadding={bodyCellVerticalPadding}\n            isTableScrolledToLeft={isScrolledToLeft}\n            isTableScrolledToRight={isScrolledToRight}\n            isEmpty={isEmpty}\n            isLoading={isLoading}\n            loadingStateScreenReaderText={loadingStateScreenReaderText}\n            emptyTableContentHeight={emptyTableContentHeight}\n            footer={footer}\n            isFirstColumnSticky={isFirstColumnSticky}\n            isLastColumnSticky={isLastColumnSticky}\n            emptyCellContent={emptyCellContent}\n          >\n            {children}\n          </TableBody>\n        </StyledTable>\n      </TableContainer>\n      {footerElm}\n    </>\n  );\n}\n\nexport function DataTable({\n  caption,\n  columns,\n  rows = [],\n  footer,\n  currentlySortedByColumn,\n  isFirstColumnSticky = true,\n  isLastColumnSticky = false,\n  \"data-e2e-test-id\": dataE2eTestId,\n  isLoading = false,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty = false,\n  children,\n  emptyTableContentHeight = \"15rem\",\n  layout = \"auto\",\n  width = \"100%\",\n  bodyCellVerticalPadding = \"m\",\n  emptyCellContent = \"--\",\n  maxHeight,\n  onSort,\n}: React.PropsWithChildren<DataTableProps>): React.ReactElement {\n  return (\n    <Container elevation={1}>\n      <BaseDataTable\n        caption={caption}\n        columns={columns}\n        rows={rows}\n        footer={footer}\n        currentlySortedByColumn={currentlySortedByColumn}\n        isFirstColumnSticky={isFirstColumnSticky}\n        isLastColumnSticky={isLastColumnSticky}\n        data-e2e-test-id={dataE2eTestId}\n        isLoading={isLoading}\n        loadingStateScreenReaderText={loadingStateScreenReaderText}\n        isEmpty={isEmpty}\n        emptyTableContentHeight={emptyTableContentHeight}\n        layout={layout}\n        width={width}\n        bodyCellVerticalPadding={bodyCellVerticalPadding}\n        emptyCellContent={emptyCellContent}\n        maxHeight={maxHeight}\n        onSort={onSort}\n      >\n        {children}\n      </BaseDataTable>\n    </Container>\n  );\n}\n"],"names":[],"mappings":"AAiIqB"} */");export function BaseDataTable({caption,columns,rows=[],footer,currentlySortedByColumn,isFirstColumnSticky=!0,isLastColumnSticky=!1,"data-e2e-test-id":dataE2eTestId,isLoading=!1,loadingStateScreenReaderText="Loading",isEmpty=!1,children,emptyTableContentHeight="15rem",layout="auto",width="100%",bodyCellVerticalPadding="m",emptyCellContent="--",maxHeight,onSort}){let[isScrolledToRight,setIsScrolledToRight]=useState(isLastColumnSticky),[isScrolledToLeft,setIsScrolledToLeft]=useState(isFirstColumnSticky),tableRef=useRef(null),containerRef=useRef(null),handleScroll=useCallback(evt=>{let elm=evt.target;setIsScrolledToLeft(0===elm.scrollLeft),setIsScrolledToRight(getTableScrolledToRight(elm))},[]);useEffect(()=>{let resizeObserver;return"undefined"!=typeof ResizeObserver&&containerRef&&containerRef.current&&(resizeObserver=new ResizeObserver(entries=>{entries.forEach(()=>{setIsScrolledToLeft(0===containerRef.current.scrollLeft),setIsScrolledToRight(getTableScrolledToRight(containerRef.current))})})).observe(containerRef.current),()=>{resizeObserver&&resizeObserver.disconnect()}},[containerRef]);let footerElm=footer?React.createElement(StyledFooter,null,footer):null;return React.createElement(React.Fragment,null,React.createElement(TableContainer,{ref:containerRef,footer:footer,maxHeight:maxHeight,onScroll:handleScroll},React.createElement(StyledTable,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"DataTable",ref:tableRef,layout:layout,tableWidth:width},React.createElement(ScreenReaderText,{as:"caption"},caption),React.createElement(TableHeader,{columns:columns,isFirstColumnSticky:isFirstColumnSticky,isLastColumnSticky:isLastColumnSticky,currentlySortedByColumn:currentlySortedByColumn,isTableScrolledToLeft:isScrolledToLeft,isTableScrolledToRight:isScrolledToRight,maxHeight:maxHeight,onSort:onSort}),React.createElement(TableBody,{columns:columns,rows:rows,bodyCellVerticalPadding:bodyCellVerticalPadding,isTableScrolledToLeft:isScrolledToLeft,isTableScrolledToRight:isScrolledToRight,isEmpty:isEmpty,isLoading:isLoading,loadingStateScreenReaderText:loadingStateScreenReaderText,emptyTableContentHeight:emptyTableContentHeight,footer:footer,isFirstColumnSticky:isFirstColumnSticky,isLastColumnSticky:isLastColumnSticky,emptyCellContent:emptyCellContent},children))),footerElm)}export function DataTable({caption,columns,rows=[],footer,currentlySortedByColumn,isFirstColumnSticky=!0,isLastColumnSticky=!1,"data-e2e-test-id":dataE2eTestId,isLoading=!1,loadingStateScreenReaderText="Loading",isEmpty=!1,children,emptyTableContentHeight="15rem",layout="auto",width="100%",bodyCellVerticalPadding="m",emptyCellContent="--",maxHeight,onSort}){return React.createElement(Container,{elevation:1},React.createElement(BaseDataTable,{caption:caption,columns:columns,rows:rows,footer:footer,currentlySortedByColumn:currentlySortedByColumn,isFirstColumnSticky:isFirstColumnSticky,isLastColumnSticky:isLastColumnSticky,"data-e2e-test-id":dataE2eTestId,isLoading:isLoading,loadingStateScreenReaderText:loadingStateScreenReaderText,isEmpty:isEmpty,emptyTableContentHeight:emptyTableContentHeight,layout:layout,width:width,bodyCellVerticalPadding:bodyCellVerticalPadding,emptyCellContent:emptyCellContent,maxHeight:maxHeight,onSort:onSort},children))}
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
+
import type { TableCellProps } from "./TableCell";
|
|
2
3
|
import type { DataTableProps } from "./DataTable";
|
|
3
4
|
export type TableBodyProps = {
|
|
4
5
|
className?: string;
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
declare function TableBody({ className, columns, rows, bodyCellVerticalPadding, isTableScrolled, isLoading, loadingStateScreenReaderText, isEmpty, emptyTableContentHeight, footer, children, isFirstColumnSticky, emptyCellContent, }: React.PropsWithChildren<TableBodyProps>): React.ReactElement;
|
|
6
|
+
} & Pick<DataTableProps, "columns" | "rows" | "isLoading" | "loadingStateScreenReaderText" | "isEmpty" | "emptyTableContentHeight" | "bodyCellVerticalPadding" | "footer" | "isFirstColumnSticky" | "isLastColumnSticky" | "emptyCellContent"> & Pick<TableCellProps, "isTableScrolledToLeft" | "isTableScrolledToRight">;
|
|
7
|
+
declare function TableBody({ className, columns, rows, bodyCellVerticalPadding, isTableScrolledToLeft, isTableScrolledToRight, isLoading, loadingStateScreenReaderText, isEmpty, emptyTableContentHeight, footer, children, isFirstColumnSticky, isLastColumnSticky, emptyCellContent, }: React.PropsWithChildren<TableBodyProps>): React.ReactElement;
|
|
8
8
|
declare const _default: React.MemoExoticComponent<typeof TableBody>;
|
|
9
9
|
export default _default;
|