@amboss/design-system 3.14.1 → 3.15.1
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/TableBody.js +1 -1
- package/build/cjs/components/EntityList/BaseEntityList.d.ts +1 -1
- package/build/cjs/components/EntityList/BaseEntityList.js +1 -1
- package/build/cjs/components/EntityList/EntityItemBody.js +1 -1
- package/build/cjs/components/EntityList/types.d.ts +10 -1
- package/build/cjs/components/EntityTree/BaseEntityTree.d.ts +10 -1
- package/build/cjs/components/EntityTree/BaseEntityTree.js +1 -1
- package/build/cjs/components/Patterns/Modal/Modal.js +3 -3
- package/build/cjs/components/Tooltip/TooltipContent.js +1 -1
- package/build/cjs/types/index.d.ts +10 -10
- package/build/cjs/types/index.js +1 -1
- package/build/esm/components/DataTable/TableBody.js +1 -1
- package/build/esm/components/EntityList/BaseEntityList.d.ts +1 -1
- package/build/esm/components/EntityList/BaseEntityList.js +1 -1
- package/build/esm/components/EntityList/EntityItemBody.js +1 -1
- package/build/esm/components/EntityList/types.d.ts +10 -1
- package/build/esm/components/EntityTree/BaseEntityTree.d.ts +10 -1
- package/build/esm/components/EntityTree/BaseEntityTree.js +1 -1
- package/build/esm/components/Patterns/Modal/Modal.js +3 -3
- package/build/esm/components/Tooltip/TooltipContent.js +1 -1
- package/build/esm/types/index.d.ts +10 -10
- package/build/esm/types/index.js +1 -1
- package/package.json +5 -5
|
@@ -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=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_react1=require("@emotion/react"),_useDocument=require("../../shared/useDocument"),_useWindow=require("../../shared/useWindow"),_SubThemeProvider=require("../SubThemeProvider/SubThemeProvider"),_Portal=require("../Portal/Portal"),_utils=require("./utils"),StyledContainer=(0,_styled.default)("div",{target:"e1t0ow2j0",label:"StyledContainer"})(({theme,horizontalPlacement,verticalPlacement,maxWidth,contentPadding,subTheme,isHidden})=>{let animationDistance="top"===verticalPlacement?`${_utils.ANIMATION_DISTANCE}px`:`-${_utils.ANIMATION_DISTANCE}px`,animation=(0,_react1.keyframes)({to:{opacity:theme.variables.opacity.visible,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:(0,_Portal.usePortalChildZIndex)(theme.variables.zIndex.tooltip),opacity:theme.variables.opacity.hidden,animation:`200ms ease-out forwards ${animation}`,maxWidth,width:"max-content",boxSizing:"border-box",backgroundColor:subTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,borderRadius:subTheme?theme.variables.size.borderRadius.xs:theme.variables.size.borderRadius.s,...isHidden&&{visibility:"hidden"},...!!subTheme&&invertedSubThemeStyles,..."center"===horizontalPlacement&&{transform:"translate(-50%)"},...!subTheme&&{":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 type { Property } from \"csstype\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport type { SubThemeTypes } from \"../../web-tokens/_subThemeType\";\nimport {\n  Portal,\n  usePortalChildZIndex,\n  type PortalProps,\n} from \"../Portal/Portal\";\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    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  contentPadding?: \"s\" | \"m\";\n  subTheme?: SubThemeTypes;\n  maxWidth?: Property.MaxWidth | number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  hideArrow?: boolean;\n  /** Sets css visibility hidden when trigger is scrolled out of view */\n  isHiddenOnInvisibleTrigger?: boolean;\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  /** This callback is triggered when content overflows the viewport both above and below trigger. Use this for tracking purposes or re-rendering Popover with a larger maxWidth */\n  onOverflowViewport?: (\n    triggerRect: DOMRect,\n    tooltipRect: DOMRect,\n    viewportHeight: number\n  ) => void;\n} & Pick<PortalProps, \"portalContainer\">;\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\" | \"subTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  isHidden: boolean;\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n// Use 0.99 instead of 1.0 to handle sub-pixel rounding issues\n// that can cause intersectionRatio to report values like 0.99\nconst INTERSECTION_THRESHOLD = 0.99;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    subTheme,\n    isHidden,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: theme.variables.opacity.visible,\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    const zIndex = usePortalChildZIndex(theme.variables.zIndex.tooltip);\n\n    return {\n      position: \"absolute\",\n      zIndex,\n      opacity: theme.variables.opacity.hidden,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: subTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: subTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(isHidden && {\n        visibility: \"hidden\",\n      }),\n\n      ...(!!subTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!subTheme && {\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, \"subTheme\"> & {\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    subTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = subTheme ? 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: subTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!subTheme && {\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  subTheme,\n  defaultVerticalPlacement,\n  hideArrow = false,\n  isHiddenOnInvisibleTrigger = false,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  onOverflowViewport,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const [isHidden, setIsHidden] = useState(false); // css visibilty hidden\n  const tooltipRef = useRef(null);\n  const isOverflowCallbackCalledRef = useRef(false);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return subTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [subTheme, hideArrow]);\n\n  const handleViewportOverflow = useCallback<\n    TooltipContentProps[\"onOverflowViewport\"]\n  >(\n    (triggerRect, tooltipRect, viewportHeight) => {\n      if (!isOverflowCallbackCalledRef.current) {\n        onOverflowViewport?.(triggerRect, tooltipRect, viewportHeight);\n        isOverflowCallbackCalledRef.current = true;\n      }\n    },\n    [onOverflowViewport, isOverflowCallbackCalledRef]\n  );\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          onOverflowViewport: handleViewportOverflow,\n        })\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n    handleViewportOverflow,\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, maxWidth]);\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, true);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  useEffect(() => {\n    isOverflowCallbackCalledRef.current = false;\n  }, [isVisible, isOverflowCallbackCalledRef]);\n\n  /* Hide (visibility: hidden) tooltip if trigger is scrolled out of view */\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      isHiddenOnInvisibleTrigger &&\n      triggerRef.current &&\n      isVisible &&\n      typeof IntersectionObserver !== \"undefined\"\n    ) {\n      observer = new IntersectionObserver(\n        (entries) =>\n          entries.forEach((entry) => {\n            const rect = entry.boundingClientRect;\n\n            // Ignore intersection changes when element has zero dimensions\n            // This can happen during DOM updates/portal rendering\n            if (rect?.width === 0 && rect?.height === 0) {\n              return;\n            }\n\n            setIsHidden(!entry.isIntersecting);\n          }),\n        {\n          root: null, // viewport\n          threshold: INTERSECTION_THRESHOLD,\n        }\n      );\n\n      observer.observe(triggerRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, [triggerRef, isVisible, isHiddenOnInvisibleTrigger]);\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      subTheme={subTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      isHidden={isHidden}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          subTheme={subTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = subTheme ? (\n    <SubThemeProvider name={subTheme}>{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return <Portal portalContainer={portalContainer}>{wrapperElm}</Portal>;\n}\n"],"names":[],"mappings":"AAyFwB"} */"),StyledArrow=(0,_styled.default)("div",{target:"e1t0ow2j1",label:"StyledArrow"})(({theme,subTheme,verticalPlacement,horizontalPlacement,size=_utils.ARROW_SIZE})=>{let offset=(0,_utils.getArrowOffset)(size),adjustmentForShadow=+!subTheme,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:subTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,transform:"translate(-50%, -50%) rotate(45deg)",...!subTheme&&{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 type { Property } from \"csstype\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport type { SubThemeTypes } from \"../../web-tokens/_subThemeType\";\nimport {\n  Portal,\n  usePortalChildZIndex,\n  type PortalProps,\n} from \"../Portal/Portal\";\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    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  contentPadding?: \"s\" | \"m\";\n  subTheme?: SubThemeTypes;\n  maxWidth?: Property.MaxWidth | number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  hideArrow?: boolean;\n  /** Sets css visibility hidden when trigger is scrolled out of view */\n  isHiddenOnInvisibleTrigger?: boolean;\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  /** This callback is triggered when content overflows the viewport both above and below trigger. Use this for tracking purposes or re-rendering Popover with a larger maxWidth */\n  onOverflowViewport?: (\n    triggerRect: DOMRect,\n    tooltipRect: DOMRect,\n    viewportHeight: number\n  ) => void;\n} & Pick<PortalProps, \"portalContainer\">;\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\" | \"subTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  isHidden: boolean;\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n// Use 0.99 instead of 1.0 to handle sub-pixel rounding issues\n// that can cause intersectionRatio to report values like 0.99\nconst INTERSECTION_THRESHOLD = 0.99;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    subTheme,\n    isHidden,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: theme.variables.opacity.visible,\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    const zIndex = usePortalChildZIndex(theme.variables.zIndex.tooltip);\n\n    return {\n      position: \"absolute\",\n      zIndex,\n      opacity: theme.variables.opacity.hidden,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: subTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: subTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(isHidden && {\n        visibility: \"hidden\",\n      }),\n\n      ...(!!subTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!subTheme && {\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, \"subTheme\"> & {\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    subTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = subTheme ? 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: subTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!subTheme && {\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  subTheme,\n  defaultVerticalPlacement,\n  hideArrow = false,\n  isHiddenOnInvisibleTrigger = false,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  onOverflowViewport,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const [isHidden, setIsHidden] = useState(false); // css visibilty hidden\n  const tooltipRef = useRef(null);\n  const isOverflowCallbackCalledRef = useRef(false);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return subTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [subTheme, hideArrow]);\n\n  const handleViewportOverflow = useCallback<\n    TooltipContentProps[\"onOverflowViewport\"]\n  >(\n    (triggerRect, tooltipRect, viewportHeight) => {\n      if (!isOverflowCallbackCalledRef.current) {\n        onOverflowViewport?.(triggerRect, tooltipRect, viewportHeight);\n        isOverflowCallbackCalledRef.current = true;\n      }\n    },\n    [onOverflowViewport, isOverflowCallbackCalledRef]\n  );\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          onOverflowViewport: handleViewportOverflow,\n        })\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n    handleViewportOverflow,\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, maxWidth]);\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, true);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  useEffect(() => {\n    isOverflowCallbackCalledRef.current = false;\n  }, [isVisible, isOverflowCallbackCalledRef]);\n\n  /* Hide (visibility: hidden) tooltip if trigger is scrolled out of view */\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      isHiddenOnInvisibleTrigger &&\n      triggerRef.current &&\n      isVisible &&\n      typeof IntersectionObserver !== \"undefined\"\n    ) {\n      observer = new IntersectionObserver(\n        (entries) =>\n          entries.forEach((entry) => {\n            const rect = entry.boundingClientRect;\n\n            // Ignore intersection changes when element has zero dimensions\n            // This can happen during DOM updates/portal rendering\n            if (rect?.width === 0 && rect?.height === 0) {\n              return;\n            }\n\n            setIsHidden(!entry.isIntersecting);\n          }),\n        {\n          root: null, // viewport\n          threshold: INTERSECTION_THRESHOLD,\n        }\n      );\n\n      observer.observe(triggerRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, [triggerRef, isVisible, isHiddenOnInvisibleTrigger]);\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      subTheme={subTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      isHidden={isHidden}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          subTheme={subTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = subTheme ? (\n    <SubThemeProvider name={subTheme}>{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return <Portal portalContainer={portalContainer}>{wrapperElm}</Portal>;\n}\n"],"names":[],"mappings":"AAiLoB"} */"),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,subTheme,defaultVerticalPlacement,hideArrow=!1,isHiddenOnInvisibleTrigger=!1,onTooltipPointerEnter,onTooltipPointerLeave,onOverflowViewport}){let animationDuration,[style,setStyle]=(0,_react.useState)(initialStyle),[isHidden,setIsHidden]=(0,_react.useState)(!1),tooltipRef=(0,_react.useRef)(null),isOverflowCallbackCalledRef=(0,_react.useRef)(!1),document=(0,_useDocument.useDocument)(),window=(0,_useWindow.useWindow)(),arrowSize=(0,_react.useMemo)(()=>hideArrow?0:subTheme?_utils.ARROW_SIZE:_utils.ARROW_SIZE_BIG,[subTheme,hideArrow]),handleViewportOverflow=(0,_react.useCallback)((triggerRect,tooltipRect,viewportHeight)=>{isOverflowCallbackCalledRef.current||(onOverflowViewport?.(triggerRect,tooltipRect,viewportHeight),isOverflowCallbackCalledRef.current=!0)},[onOverflowViewport,isOverflowCallbackCalledRef]),calculateStyle=(0,_react.useCallback)(()=>{triggerRef.current&&tooltipRef.current&&setStyle((0,_utils.getTooltipStyle)({placement,defaultVerticalPlacement,triggerRef,tooltipRef,document,window,arrowSize,onOverflowViewport:handleViewportOverflow}))},[triggerRef,tooltipRef,document,window,placement,arrowSize,defaultVerticalPlacement,handleViewportOverflow]);if((0,_react.useLayoutEffect)(()=>{isVisible&&calculateStyle()},[isVisible,calculateStyle,contentPadding,content,maxWidth]),(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,!0)}),[isVisible,calculateStyle,window,tooltipRef]),(0,_react.useEffect)(()=>{isOverflowCallbackCalledRef.current=!1},[isVisible,isOverflowCallbackCalledRef]),(0,_react.useEffect)(()=>{let observer;return isHiddenOnInvisibleTrigger&&triggerRef.current&&isVisible&&"undefined"!=typeof IntersectionObserver&&(observer=new IntersectionObserver(entries=>entries.forEach(entry=>{let rect=entry.boundingClientRect;(rect?.width!==0||rect?.height!==0)&&setIsHidden(!entry.isIntersecting)}),{root:null,threshold:.99})).observe(triggerRef.current),()=>{observer&&observer.disconnect()}},[triggerRef,isVisible,isHiddenOnInvisibleTrigger]),!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,subTheme:subTheme,tabIndex:tabIndex,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,maxWidth:maxWidth,contentPadding:contentPadding,isHidden:isHidden,onPointerEnter:onTooltipPointerEnter,onPointerLeave:onTooltipPointerLeave},content,!hideArrow&&_react.default.createElement(StyledArrow,{"data-e2e-test-id":`${dataE2eTestId}_arrow`,subTheme:subTheme,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,size:arrowSize})),wrapperElm=subTheme?_react.default.createElement(_SubThemeProvider.SubThemeProvider,{name:subTheme},tooltipElm):tooltipElm;return _react.default.createElement(_Portal.Portal,{portalContainer:portalContainer},wrapperElm)}
|
|
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=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_react1=require("@emotion/react"),_useDocument=require("../../shared/useDocument"),_useWindow=require("../../shared/useWindow"),_SubThemeProvider=require("../SubThemeProvider/SubThemeProvider"),_Portal=require("../Portal/Portal"),_utils=require("./utils"),StyledContainer=(0,_styled.default)("div",{target:"efqithb0",label:"StyledContainer"})(({theme,horizontalPlacement,verticalPlacement,maxWidth,contentPadding,subTheme,isHidden})=>{let animationDistance="top"===verticalPlacement?`${_utils.ANIMATION_DISTANCE}px`:`-${_utils.ANIMATION_DISTANCE}px`,animation=(0,_react1.keyframes)({to:{opacity:theme.variables.opacity.visible,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:(0,_Portal.usePortalChildZIndex)(theme.variables.zIndex.tooltip),opacity:theme.variables.opacity.hidden,animation:`200ms ease-out forwards ${animation}`,maxWidth,width:"max-content",boxSizing:"border-box",backgroundColor:subTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,borderRadius:subTheme?theme.variables.size.borderRadius.xs:theme.variables.size.borderRadius.s,...isHidden&&{visibility:"hidden"},...!!subTheme&&invertedSubThemeStyles,..."center"===horizontalPlacement&&{transform:"translate(-50%)"},...!subTheme&&{":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 type { Property } from \"csstype\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport type { SubThemeTypes } from \"../../web-tokens/_subThemeType\";\nimport {\n  Portal,\n  usePortalChildZIndex,\n  type PortalProps,\n} from \"../Portal/Portal\";\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    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  contentPadding?: \"s\" | \"m\";\n  subTheme?: SubThemeTypes;\n  maxWidth?: Property.MaxWidth | number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  hideArrow?: boolean;\n  /** Sets css visibility hidden when trigger is scrolled out of view */\n  isHiddenOnInvisibleTrigger?: boolean;\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  /** This callback is triggered when content overflows the viewport both above and below trigger. Use this for tracking purposes or re-rendering Popover with a larger maxWidth */\n  onOverflowViewport?: (\n    triggerRect: DOMRect,\n    tooltipRect: DOMRect,\n    viewportHeight: number\n  ) => void;\n} & Pick<PortalProps, \"portalContainer\">;\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\" | \"subTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  isHidden: boolean;\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n// Use 0.99 instead of 1.0 to handle sub-pixel rounding issues\n// that can cause intersectionRatio to report values like 0.99\nconst INTERSECTION_THRESHOLD = 0.99;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    subTheme,\n    isHidden,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: theme.variables.opacity.visible,\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    const zIndex = usePortalChildZIndex(theme.variables.zIndex.tooltip);\n\n    return {\n      position: \"absolute\",\n      zIndex,\n      opacity: theme.variables.opacity.hidden,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: subTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: subTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(isHidden && {\n        visibility: \"hidden\",\n      }),\n\n      ...(!!subTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!subTheme && {\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, \"subTheme\"> & {\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    subTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = subTheme ? 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: subTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!subTheme && {\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  subTheme,\n  defaultVerticalPlacement,\n  hideArrow = false,\n  isHiddenOnInvisibleTrigger = false,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  onOverflowViewport,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const [isHidden, setIsHidden] = useState(false); // css visibilty hidden\n  const tooltipRef = useRef(null);\n  const isOverflowCallbackCalledRef = useRef(false);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return subTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [subTheme, hideArrow]);\n\n  const handleViewportOverflow = useCallback<\n    TooltipContentProps[\"onOverflowViewport\"]\n  >(\n    (triggerRect, tooltipRect, viewportHeight) => {\n      if (!isOverflowCallbackCalledRef.current) {\n        onOverflowViewport?.(triggerRect, tooltipRect, viewportHeight);\n        isOverflowCallbackCalledRef.current = true;\n      }\n    },\n    [onOverflowViewport, isOverflowCallbackCalledRef]\n  );\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          onOverflowViewport: handleViewportOverflow,\n        })\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n    handleViewportOverflow,\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, maxWidth]);\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, true);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  // Observe tooltip element size changes to recalculate positioning\n  useEffect(() => {\n    let resizeObserver: ResizeObserver;\n\n    if (\n      isVisible &&\n      tooltipRef.current &&\n      typeof ResizeObserver !== \"undefined\"\n    ) {\n      resizeObserver = new ResizeObserver((entries) => {\n        entries.forEach((entry) => {\n          if (entry.contentRect.width > 0 && entry.contentRect.height > 0) {\n            calculateStyle();\n          }\n        });\n      });\n\n      resizeObserver.observe(tooltipRef.current);\n    }\n\n    return () => {\n      if (resizeObserver) {\n        resizeObserver.disconnect();\n      }\n    };\n  }, [isVisible, calculateStyle, tooltipRef]);\n\n  useEffect(() => {\n    isOverflowCallbackCalledRef.current = false;\n  }, [isVisible, isOverflowCallbackCalledRef]);\n\n  /* Hide (visibility: hidden) tooltip if trigger is scrolled out of view */\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      isHiddenOnInvisibleTrigger &&\n      triggerRef.current &&\n      isVisible &&\n      typeof IntersectionObserver !== \"undefined\"\n    ) {\n      observer = new IntersectionObserver(\n        (entries) =>\n          entries.forEach((entry) => {\n            const rect = entry.boundingClientRect;\n\n            // Ignore intersection changes when element has zero dimensions\n            // This can happen during DOM updates/portal rendering\n            if (rect?.width === 0 && rect?.height === 0) {\n              return;\n            }\n\n            setIsHidden(!entry.isIntersecting);\n          }),\n        {\n          root: null, // viewport\n          threshold: INTERSECTION_THRESHOLD,\n        }\n      );\n\n      observer.observe(triggerRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, [triggerRef, isVisible, isHiddenOnInvisibleTrigger]);\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      subTheme={subTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      isHidden={isHidden}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          subTheme={subTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = subTheme ? (\n    <SubThemeProvider name={subTheme}>{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return <Portal portalContainer={portalContainer}>{wrapperElm}</Portal>;\n}\n"],"names":[],"mappings":"AAyFwB"} */"),StyledArrow=(0,_styled.default)("div",{target:"efqithb1",label:"StyledArrow"})(({theme,subTheme,verticalPlacement,horizontalPlacement,size=_utils.ARROW_SIZE})=>{let offset=(0,_utils.getArrowOffset)(size),adjustmentForShadow=+!subTheme,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:subTheme?theme.values.color.background.primary.default:theme.values.color.background.elevated.default,transform:"translate(-50%, -50%) rotate(45deg)",...!subTheme&&{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 type { Property } from \"csstype\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport type { SubThemeTypes } from \"../../web-tokens/_subThemeType\";\nimport {\n  Portal,\n  usePortalChildZIndex,\n  type PortalProps,\n} from \"../Portal/Portal\";\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    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  contentPadding?: \"s\" | \"m\";\n  subTheme?: SubThemeTypes;\n  maxWidth?: Property.MaxWidth | number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  hideArrow?: boolean;\n  /** Sets css visibility hidden when trigger is scrolled out of view */\n  isHiddenOnInvisibleTrigger?: boolean;\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  /** This callback is triggered when content overflows the viewport both above and below trigger. Use this for tracking purposes or re-rendering Popover with a larger maxWidth */\n  onOverflowViewport?: (\n    triggerRect: DOMRect,\n    tooltipRect: DOMRect,\n    viewportHeight: number\n  ) => void;\n} & Pick<PortalProps, \"portalContainer\">;\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\" | \"subTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  isHidden: boolean;\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n// Use 0.99 instead of 1.0 to handle sub-pixel rounding issues\n// that can cause intersectionRatio to report values like 0.99\nconst INTERSECTION_THRESHOLD = 0.99;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    subTheme,\n    isHidden,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: theme.variables.opacity.visible,\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    const zIndex = usePortalChildZIndex(theme.variables.zIndex.tooltip);\n\n    return {\n      position: \"absolute\",\n      zIndex,\n      opacity: theme.variables.opacity.hidden,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: subTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: subTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(isHidden && {\n        visibility: \"hidden\",\n      }),\n\n      ...(!!subTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(!subTheme && {\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, \"subTheme\"> & {\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    subTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = subTheme ? 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: subTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!subTheme && {\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  subTheme,\n  defaultVerticalPlacement,\n  hideArrow = false,\n  isHiddenOnInvisibleTrigger = false,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  onOverflowViewport,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const [isHidden, setIsHidden] = useState(false); // css visibilty hidden\n  const tooltipRef = useRef(null);\n  const isOverflowCallbackCalledRef = useRef(false);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return subTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [subTheme, hideArrow]);\n\n  const handleViewportOverflow = useCallback<\n    TooltipContentProps[\"onOverflowViewport\"]\n  >(\n    (triggerRect, tooltipRect, viewportHeight) => {\n      if (!isOverflowCallbackCalledRef.current) {\n        onOverflowViewport?.(triggerRect, tooltipRect, viewportHeight);\n        isOverflowCallbackCalledRef.current = true;\n      }\n    },\n    [onOverflowViewport, isOverflowCallbackCalledRef]\n  );\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          onOverflowViewport: handleViewportOverflow,\n        })\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n    handleViewportOverflow,\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, maxWidth]);\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, true);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  // Observe tooltip element size changes to recalculate positioning\n  useEffect(() => {\n    let resizeObserver: ResizeObserver;\n\n    if (\n      isVisible &&\n      tooltipRef.current &&\n      typeof ResizeObserver !== \"undefined\"\n    ) {\n      resizeObserver = new ResizeObserver((entries) => {\n        entries.forEach((entry) => {\n          if (entry.contentRect.width > 0 && entry.contentRect.height > 0) {\n            calculateStyle();\n          }\n        });\n      });\n\n      resizeObserver.observe(tooltipRef.current);\n    }\n\n    return () => {\n      if (resizeObserver) {\n        resizeObserver.disconnect();\n      }\n    };\n  }, [isVisible, calculateStyle, tooltipRef]);\n\n  useEffect(() => {\n    isOverflowCallbackCalledRef.current = false;\n  }, [isVisible, isOverflowCallbackCalledRef]);\n\n  /* Hide (visibility: hidden) tooltip if trigger is scrolled out of view */\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      isHiddenOnInvisibleTrigger &&\n      triggerRef.current &&\n      isVisible &&\n      typeof IntersectionObserver !== \"undefined\"\n    ) {\n      observer = new IntersectionObserver(\n        (entries) =>\n          entries.forEach((entry) => {\n            const rect = entry.boundingClientRect;\n\n            // Ignore intersection changes when element has zero dimensions\n            // This can happen during DOM updates/portal rendering\n            if (rect?.width === 0 && rect?.height === 0) {\n              return;\n            }\n\n            setIsHidden(!entry.isIntersecting);\n          }),\n        {\n          root: null, // viewport\n          threshold: INTERSECTION_THRESHOLD,\n        }\n      );\n\n      observer.observe(triggerRef.current);\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n    };\n  }, [triggerRef, isVisible, isHiddenOnInvisibleTrigger]);\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      subTheme={subTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      isHidden={isHidden}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          subTheme={subTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = subTheme ? (\n    <SubThemeProvider name={subTheme}>{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return <Portal portalContainer={portalContainer}>{wrapperElm}</Portal>;\n}\n"],"names":[],"mappings":"AAiLoB"} */"),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,subTheme,defaultVerticalPlacement,hideArrow=!1,isHiddenOnInvisibleTrigger=!1,onTooltipPointerEnter,onTooltipPointerLeave,onOverflowViewport}){let animationDuration,[style,setStyle]=(0,_react.useState)(initialStyle),[isHidden,setIsHidden]=(0,_react.useState)(!1),tooltipRef=(0,_react.useRef)(null),isOverflowCallbackCalledRef=(0,_react.useRef)(!1),document=(0,_useDocument.useDocument)(),window=(0,_useWindow.useWindow)(),arrowSize=(0,_react.useMemo)(()=>hideArrow?0:subTheme?_utils.ARROW_SIZE:_utils.ARROW_SIZE_BIG,[subTheme,hideArrow]),handleViewportOverflow=(0,_react.useCallback)((triggerRect,tooltipRect,viewportHeight)=>{isOverflowCallbackCalledRef.current||(onOverflowViewport?.(triggerRect,tooltipRect,viewportHeight),isOverflowCallbackCalledRef.current=!0)},[onOverflowViewport,isOverflowCallbackCalledRef]),calculateStyle=(0,_react.useCallback)(()=>{triggerRef.current&&tooltipRef.current&&setStyle((0,_utils.getTooltipStyle)({placement,defaultVerticalPlacement,triggerRef,tooltipRef,document,window,arrowSize,onOverflowViewport:handleViewportOverflow}))},[triggerRef,tooltipRef,document,window,placement,arrowSize,defaultVerticalPlacement,handleViewportOverflow]);if((0,_react.useLayoutEffect)(()=>{isVisible&&calculateStyle()},[isVisible,calculateStyle,contentPadding,content,maxWidth]),(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,!0)}),[isVisible,calculateStyle,window,tooltipRef]),(0,_react.useEffect)(()=>{let resizeObserver;return isVisible&&tooltipRef.current&&"undefined"!=typeof ResizeObserver&&(resizeObserver=new ResizeObserver(entries=>{entries.forEach(entry=>{entry.contentRect.width>0&&entry.contentRect.height>0&&calculateStyle()})})).observe(tooltipRef.current),()=>{resizeObserver&&resizeObserver.disconnect()}},[isVisible,calculateStyle,tooltipRef]),(0,_react.useEffect)(()=>{isOverflowCallbackCalledRef.current=!1},[isVisible,isOverflowCallbackCalledRef]),(0,_react.useEffect)(()=>{let observer;return isHiddenOnInvisibleTrigger&&triggerRef.current&&isVisible&&"undefined"!=typeof IntersectionObserver&&(observer=new IntersectionObserver(entries=>entries.forEach(entry=>{let rect=entry.boundingClientRect;(rect?.width!==0||rect?.height!==0)&&setIsHidden(!entry.isIntersecting)}),{root:null,threshold:.99})).observe(triggerRef.current),()=>{observer&&observer.disconnect()}},[triggerRef,isVisible,isHiddenOnInvisibleTrigger]),!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,subTheme:subTheme,tabIndex:tabIndex,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,maxWidth:maxWidth,contentPadding:contentPadding,isHidden:isHidden,onPointerEnter:onTooltipPointerEnter,onPointerLeave:onTooltipPointerLeave},content,!hideArrow&&_react.default.createElement(StyledArrow,{"data-e2e-test-id":`${dataE2eTestId}_arrow`,subTheme:subTheme,horizontalPlacement:style.horizontalPlacement,verticalPlacement:style.verticalPlacement,size:arrowSize})),wrapperElm=subTheme?_react.default.createElement(_SubThemeProvider.SubThemeProvider,{name:subTheme},tooltipElm):tooltipElm;return _react.default.createElement(_Portal.Portal,{portalContainer:portalContainer},wrapperElm)}
|
|
@@ -30,25 +30,25 @@ export declare const BaseVariationsRuntype: Union<[Literal<"primary">, Literal<"
|
|
|
30
30
|
export declare const BaseColorsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>;
|
|
31
31
|
export type BaseVariations = Static<typeof BaseVariationsRuntype>;
|
|
32
32
|
export type BaseColors = Static<typeof BaseColorsRuntype>;
|
|
33
|
-
declare const PictogramVariationsRuntype: Union<[Literal<"secondary">, Literal<"tertiary">, Literal<"quaternary">]>;
|
|
33
|
+
export declare const PictogramVariationsRuntype: Union<[Literal<"secondary">, Literal<"tertiary">, Literal<"quaternary">]>;
|
|
34
34
|
export type PictogramVariations = Static<typeof PictogramVariationsRuntype>;
|
|
35
|
-
declare const TextColorsRuntype: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"onAccent">]>;
|
|
35
|
+
export declare const TextColorsRuntype: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"onAccent">]>;
|
|
36
36
|
export type TextColors = Static<typeof TextColorsRuntype>;
|
|
37
|
-
declare const TextVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">]>;
|
|
37
|
+
export declare const TextVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">]>;
|
|
38
38
|
export type TextVariations = Static<typeof TextVariationsRuntype>;
|
|
39
|
-
declare const TextWeightRuntype: Union<[Literal<"normal">, Literal<"bold">, Literal<"black">, Literal<"inherit">]>;
|
|
39
|
+
export declare const TextWeightRuntype: Union<[Literal<"normal">, Literal<"bold">, Literal<"black">, Literal<"inherit">]>;
|
|
40
40
|
export type TextWeight = Static<typeof TextWeightRuntype>;
|
|
41
|
-
declare const IconsColorsRuntime: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"inherit">, Literal<"brand">, Literal<"quaternary">, Literal<"onAccent">]>;
|
|
41
|
+
export declare const IconsColorsRuntime: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"inherit">, Literal<"brand">, Literal<"quaternary">, Literal<"onAccent">]>;
|
|
42
42
|
export type IconsColors = Static<typeof IconsColorsRuntime>;
|
|
43
|
-
declare const IconsVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"inherit">, Literal<"brand">]>;
|
|
43
|
+
export declare const IconsVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"success">, Literal<"info">, Literal<"error">, Literal<"warning">, Literal<"inherit">, Literal<"brand">]>;
|
|
44
44
|
export type IconsVariations = Static<typeof IconsVariationsRuntype>;
|
|
45
45
|
export declare const IconSizesRuntype: Union<[Literal<"s">, Literal<"m">, Literal<"l">]>;
|
|
46
46
|
export type IconSizes = Static<typeof IconSizesRuntype>;
|
|
47
|
-
declare const CalloutVariationsRuntype: Union<[Literal<"info">, Literal<"success">, Literal<"error">, Literal<"warning">, Literal<"neutral">]>;
|
|
47
|
+
export declare const CalloutVariationsRuntype: Union<[Literal<"info">, Literal<"success">, Literal<"error">, Literal<"warning">, Literal<"neutral">]>;
|
|
48
48
|
export type CalloutVariations = Static<typeof CalloutVariationsRuntype>;
|
|
49
|
-
declare const LinkVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>]>;
|
|
49
|
+
export declare const LinkVariationsRuntype: Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>]>;
|
|
50
50
|
export type LinkVariations = Static<typeof LinkVariationsRuntype>;
|
|
51
|
-
declare const LinkColorsRuntype: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>]>;
|
|
51
|
+
export declare const LinkColorsRuntype: Union<[Union<[Union<[Literal<"primary">, Literal<"secondary">, Literal<"tertiary">]>, Literal<"accent">]>]>;
|
|
52
52
|
export type LinkColors = Static<typeof LinkColorsRuntype>;
|
|
53
53
|
export type MQ<T> = [T, T?, T?] | {
|
|
54
54
|
mq: [T, T?, T?];
|
|
@@ -58,7 +58,7 @@ export declare const ButtonSizeRuntype: Union<[Literal<"s">, Literal<"m">, Liter
|
|
|
58
58
|
export type ButtonSize = Static<typeof ButtonSizeRuntype>;
|
|
59
59
|
export declare const HyphensRuntype: Union<[Literal<"auto">, Literal<"none">, Literal<"manual">]>;
|
|
60
60
|
export type Hyphens = Static<typeof HyphensRuntype>;
|
|
61
|
-
declare const OverflowWrapRuntype: Union<[Literal<"normal">, Literal<"break-word">]>;
|
|
61
|
+
export declare const OverflowWrapRuntype: Union<[Literal<"normal">, Literal<"break-word">]>;
|
|
62
62
|
export type OverflowWrap = Static<typeof OverflowWrapRuntype>;
|
|
63
63
|
export declare const ToggleSizeRuntype: Union<[Literal<"s">, Literal<"m">]>;
|
|
64
64
|
export type ToggleSize = Static<typeof ToggleSizeRuntype>;
|
package/build/cjs/types/index.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={get BaseColorsRuntype(){return BaseColorsRuntype},get BaseVariationsRuntype(){return BaseVariationsRuntype},get BorderRadiusRuntype(){return BorderRadiusRuntype},get ButtonSizeRuntype(){return ButtonSizeRuntype},get ColumnAlignmentRunType(){return ColumnAlignmentRunType},get ColumsSizesRuntype(){return ColumsSizesRuntype},get HorizontalAlignmentRuntype(){return HorizontalAlignmentRuntype},get HyphensRuntype(){return HyphensRuntype},get IconSizesRuntype(){return IconSizesRuntype},get InputTextSizeRuntype(){return InputTextSizeRuntype},get LinkTextSizeRuntype(){return LinkTextSizeRuntype},get ListSizeRuntype(){return ListSizeRuntype},get MediaItemSizeRuntype(){return MediaItemSizeRuntype},get OrderRuntype(){return OrderRuntype},get SpaceSizesRuntype(){return SpaceSizesRuntype},get StackHorizontalAlignmentRuntype(){return StackHorizontalAlignmentRuntype},get StackVerticalAlignmentRuntype(){return StackVerticalAlignmentRuntype},get TextAlignmentRuntype(){return TextAlignmentRuntype},get TextSizeRuntype(){return TextSizeRuntype},get ToggleSizeRuntype(){return ToggleSizeRuntype},get VerticalAlignmentRuntype(){return VerticalAlignmentRuntype}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:Object.getOwnPropertyDescriptor(all,name).get});const _runtypes=require("runtypes"),LinkTextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l")),TextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),InputTextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),BorderRadiusRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("none"),(0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),TextAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center")),HorizontalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("spaceBetween")),VerticalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("top"),(0,_runtypes.Literal)("bottom"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("stretch")),StackVerticalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("top"),(0,_runtypes.Literal)("bottom"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("spaceBetween")),StackHorizontalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("stretch")),SpaceSizesRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("zero"),(0,_runtypes.Literal)("xxxs"),(0,_runtypes.Literal)("xxs"),(0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l"),(0,_runtypes.Literal)("xl"),(0,_runtypes.Literal)("xxl")),ColumsSizesRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)(1),(0,_runtypes.Literal)(2),(0,_runtypes.Literal)(3),(0,_runtypes.Literal)(4),(0,_runtypes.Literal)(5),(0,_runtypes.Literal)(6),(0,_runtypes.Literal)(7),(0,_runtypes.Literal)(8),(0,_runtypes.Literal)(9),(0,_runtypes.Literal)(10),(0,_runtypes.Literal)(11),(0,_runtypes.Literal)(12),(0,_runtypes.Literal)("auto"),(0,_runtypes.Literal)("narrow"),(0,_runtypes.Literal)("fill")),ColumnAlignmentRunType=(0,_runtypes.Union)((0,_runtypes.Literal)("auto"),(0,_runtypes.Literal)("start"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("end")),OrderRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("first"),(0,_runtypes.Literal)("last"),(0,_runtypes.Literal)("unset")),BaseVariationsRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("primary"),(0,_runtypes.Literal)("secondary"),(0,_runtypes.Literal)("tertiary")),BaseColorsRuntype=(0,_runtypes.Union)(BaseVariationsRuntype,(0,_runtypes.Literal)("accent"))
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={get BaseColorsRuntype(){return BaseColorsRuntype},get BaseVariationsRuntype(){return BaseVariationsRuntype},get BorderRadiusRuntype(){return BorderRadiusRuntype},get ButtonSizeRuntype(){return ButtonSizeRuntype},get CalloutVariationsRuntype(){return CalloutVariationsRuntype},get ColumnAlignmentRunType(){return ColumnAlignmentRunType},get ColumsSizesRuntype(){return ColumsSizesRuntype},get HorizontalAlignmentRuntype(){return HorizontalAlignmentRuntype},get HyphensRuntype(){return HyphensRuntype},get IconSizesRuntype(){return IconSizesRuntype},get IconsColorsRuntime(){return IconsColorsRuntime},get IconsVariationsRuntype(){return IconsVariationsRuntype},get InputTextSizeRuntype(){return InputTextSizeRuntype},get LinkColorsRuntype(){return LinkColorsRuntype},get LinkTextSizeRuntype(){return LinkTextSizeRuntype},get LinkVariationsRuntype(){return LinkVariationsRuntype},get ListSizeRuntype(){return ListSizeRuntype},get MediaItemSizeRuntype(){return MediaItemSizeRuntype},get OrderRuntype(){return OrderRuntype},get OverflowWrapRuntype(){return OverflowWrapRuntype},get PictogramVariationsRuntype(){return PictogramVariationsRuntype},get SpaceSizesRuntype(){return SpaceSizesRuntype},get StackHorizontalAlignmentRuntype(){return StackHorizontalAlignmentRuntype},get StackVerticalAlignmentRuntype(){return StackVerticalAlignmentRuntype},get TextAlignmentRuntype(){return TextAlignmentRuntype},get TextColorsRuntype(){return TextColorsRuntype},get TextSizeRuntype(){return TextSizeRuntype},get TextVariationsRuntype(){return TextVariationsRuntype},get TextWeightRuntype(){return TextWeightRuntype},get ToggleSizeRuntype(){return ToggleSizeRuntype},get VerticalAlignmentRuntype(){return VerticalAlignmentRuntype}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:Object.getOwnPropertyDescriptor(all,name).get});const _runtypes=require("runtypes"),LinkTextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l")),TextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),InputTextSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),BorderRadiusRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("none"),(0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),TextAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center")),HorizontalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("spaceBetween")),VerticalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("top"),(0,_runtypes.Literal)("bottom"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("stretch")),StackVerticalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("top"),(0,_runtypes.Literal)("bottom"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("spaceBetween")),StackHorizontalAlignmentRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("left"),(0,_runtypes.Literal)("right"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("stretch")),SpaceSizesRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("zero"),(0,_runtypes.Literal)("xxxs"),(0,_runtypes.Literal)("xxs"),(0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l"),(0,_runtypes.Literal)("xl"),(0,_runtypes.Literal)("xxl")),ColumsSizesRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)(1),(0,_runtypes.Literal)(2),(0,_runtypes.Literal)(3),(0,_runtypes.Literal)(4),(0,_runtypes.Literal)(5),(0,_runtypes.Literal)(6),(0,_runtypes.Literal)(7),(0,_runtypes.Literal)(8),(0,_runtypes.Literal)(9),(0,_runtypes.Literal)(10),(0,_runtypes.Literal)(11),(0,_runtypes.Literal)(12),(0,_runtypes.Literal)("auto"),(0,_runtypes.Literal)("narrow"),(0,_runtypes.Literal)("fill")),ColumnAlignmentRunType=(0,_runtypes.Union)((0,_runtypes.Literal)("auto"),(0,_runtypes.Literal)("start"),(0,_runtypes.Literal)("center"),(0,_runtypes.Literal)("end")),OrderRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("first"),(0,_runtypes.Literal)("last"),(0,_runtypes.Literal)("unset")),BaseVariationsRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("primary"),(0,_runtypes.Literal)("secondary"),(0,_runtypes.Literal)("tertiary")),BaseColorsRuntype=(0,_runtypes.Union)(BaseVariationsRuntype,(0,_runtypes.Literal)("accent")),PictogramVariationsRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("secondary"),(0,_runtypes.Literal)("tertiary"),(0,_runtypes.Literal)("quaternary")),TextColorsRuntype=(0,_runtypes.Union)(BaseColorsRuntype,(0,_runtypes.Literal)("success"),(0,_runtypes.Literal)("info"),(0,_runtypes.Literal)("error"),(0,_runtypes.Literal)("warning"),(0,_runtypes.Literal)("onAccent")),TextVariationsRuntype=(0,_runtypes.Union)(BaseVariationsRuntype,(0,_runtypes.Literal)("success"),(0,_runtypes.Literal)("info"),(0,_runtypes.Literal)("error"),(0,_runtypes.Literal)("warning")),TextWeightRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("normal"),(0,_runtypes.Literal)("bold"),(0,_runtypes.Literal)("black"),(0,_runtypes.Literal)("inherit")),IconsColorsRuntime=(0,_runtypes.Union)(BaseColorsRuntype,(0,_runtypes.Literal)("success"),(0,_runtypes.Literal)("info"),(0,_runtypes.Literal)("error"),(0,_runtypes.Literal)("warning"),(0,_runtypes.Literal)("inherit"),(0,_runtypes.Literal)("brand"),(0,_runtypes.Literal)("quaternary"),(0,_runtypes.Literal)("onAccent")),IconsVariationsRuntype=(0,_runtypes.Union)(BaseVariationsRuntype,(0,_runtypes.Literal)("success"),(0,_runtypes.Literal)("info"),(0,_runtypes.Literal)("error"),(0,_runtypes.Literal)("warning"),(0,_runtypes.Literal)("inherit"),(0,_runtypes.Literal)("brand")),IconSizesRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l")),CalloutVariationsRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("info"),(0,_runtypes.Literal)("success"),(0,_runtypes.Literal)("error"),(0,_runtypes.Literal)("warning"),(0,_runtypes.Literal)("neutral")),LinkVariationsRuntype=(0,_runtypes.Union)(BaseVariationsRuntype),LinkColorsRuntype=(0,_runtypes.Union)(BaseColorsRuntype),ButtonSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l")),HyphensRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("auto"),(0,_runtypes.Literal)("none"),(0,_runtypes.Literal)("manual")),OverflowWrapRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("normal"),(0,_runtypes.Literal)("break-word")),ToggleSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m")),ListSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l")),MediaItemSizeRuntype=(0,_runtypes.Union)((0,_runtypes.Literal)("xs"),(0,_runtypes.Literal)("s"),(0,_runtypes.Literal)("m"),(0,_runtypes.Literal)("l"),(0,_runtypes.Literal)("xl"));
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React from"react";import styled from"@emotion/styled";import TableCell from"./TableCell";import{Text}from"../Typography/Text/Text";import{LoadingSpinner}from"../LoadingSpinner/LoadingSpinner";import{useResponsiveStyles,useResponsiveValue}from"../../shared/mediaQueries";import{Checkbox}from"../Form/Checkbox/Checkbox";import{CHECKBOX_COLUMN}from"./constants";let StyledTr=styled("tr",{target:"e1eylxe60",label:"StyledTr"})(({theme,footer,isSelected,verticalAlign})=>({...!isSelected&&{"&:hover td":{backgroundColor:theme.values.color.background.secondary.default}},verticalAlign,...!footer&&{"&:last-of-type":{td:{borderBottom:"none"}}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["import type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AA0CiB"} */"),StyledText=styled(Text,{target:"e1eylxe61",label:"StyledText"})(({alignColumn})=>({...useResponsiveValue({textAlign:alignColumn})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["import type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AAkEmB"} */"),StyledTd=styled(TableCell,{target:"e1eylxe62",label:"StyledTd"})(({theme,bodyCellVerticalPadding})=>({...useResponsiveStyles({padding:[bodyCellVerticalPadding,{s:`${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,m:`${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,l:`${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`}]})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["import type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AA8EiB"} */"),StyledEmptyTd=styled("td",{target:"e1eylxe63",label:"StyledEmptyTd"})(({emptyTableContentHeight})=>({textAlign:"center",...useResponsiveValue({height:emptyTableContentHeight})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["import type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AAsGsB"} */");function EmptyTableBody({children,columns,emptyTableContentHeight}){return React.createElement("tbody",null,React.createElement("tr",null,React.createElement(StyledEmptyTd,{colSpan:columns.length,emptyTableContentHeight:emptyTableContentHeight},children)))}function CheckboxColumnCellContent({selectRowCheckboxAriaLabel,selectedRowIds,onRowSelectionChange,id,isSelectable}){return React.createElement(Checkbox,{size:"s","aria-label":selectRowCheckboxAriaLabel,checked:selectedRowIds.includes(id),disabled:!isSelectable,onChange:evt=>{evt.target.checked?onRowSelectionChange([...selectedRowIds,id]):onRowSelectionChange(selectedRowIds.filter(item=>item!==id))}})}export default React.memo(function({className,columns,rows=[],bodyCellVerticalPadding,isTableScrolledToLeft,isTableScrolledToRight,isLoading,loadingStateScreenReaderText="Loading",isEmpty,emptyTableContentHeight,footer,children,isFirstColumnSticky,isLastColumnSticky,emptyCellContent,selectedRowIds,onRowSelectionChange,getRowProps}){if(isLoading){let bodyContent=children&&React.createElement(React.Fragment,null,children)||React.createElement(LoadingSpinner,{screenReaderText:loadingStateScreenReaderText});return React.createElement(EmptyTableBody,{columns:columns,emptyTableContentHeight:emptyTableContentHeight},bodyContent)}if(isEmpty)return React.createElement(EmptyTableBody,{columns:columns,emptyTableContentHeight:emptyTableContentHeight},React.createElement(React.Fragment,null,children));let rowElements=rows.map(row=>{let{verticalAlign=null,isActive=!1,isSelectable=!0,selectRowCheckboxAriaLabel=`Select row${row.id}`}=getRowProps?.(row)||{},isSelected=selectedRowIds.includes(row.id);return React.createElement(StyledTr,{key:row.id,footer:footer,verticalAlign:verticalAlign,isSelected:isSelected},columns.map(({name,align="left",renderCell})=>{let content;return content=renderCell?renderCell(row):name===CHECKBOX_COLUMN.name?React.createElement(CheckboxColumnCellContent,{selectRowCheckboxAriaLabel:selectRowCheckboxAriaLabel,selectedRowIds:selectedRowIds,onRowSelectionChange:onRowSelectionChange,id:row.id,isSelectable:isSelectable}):React.createElement(StyledText,{size:"s",alignColumn:align},row[name]?row[name]:emptyCellContent),React.createElement(StyledTd,{key:name,alignColumn:align,isFirstColumnSticky:isFirstColumnSticky,isLastColumnSticky:isLastColumnSticky,bodyCellVerticalPadding:bodyCellVerticalPadding,isTableScrolledToLeft:isTableScrolledToLeft,isTableScrolledToRight:isTableScrolledToRight,isActive:isActive,isSelected:isSelected},content)}))});return React.createElement("tbody",{className:className},rowElements)});
|
|
1
|
+
import React from"react";import styled from"@emotion/styled";import TableCell from"./TableCell";import{Text}from"../Typography/Text/Text";import{LoadingSpinner}from"../LoadingSpinner/LoadingSpinner";import{useResponsiveStyles,useResponsiveValue}from"../../shared/mediaQueries";import{Checkbox}from"../Form/Checkbox/Checkbox";import{CHECKBOX_COLUMN}from"./constants";let StyledTr=styled("tr",{target:"e6k8n1r0",label:"StyledTr"})(({theme,footer,isSelected,verticalAlign})=>({...!isSelected&&{"&:hover td":{backgroundColor:theme.values.color.background.secondary.default}},verticalAlign,...!footer&&{"&:last-of-type":{td:{borderBottom:"none"}}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["/* eslint-disable react/jsx-no-useless-fragment */\nimport type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AA2CiB"} */"),StyledText=styled(Text,{target:"e6k8n1r1",label:"StyledText"})(({alignColumn})=>({...useResponsiveValue({textAlign:alignColumn})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["/* eslint-disable react/jsx-no-useless-fragment */\nimport type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AAmEmB"} */"),StyledTd=styled(TableCell,{target:"e6k8n1r2",label:"StyledTd"})(({theme,bodyCellVerticalPadding})=>({...useResponsiveStyles({padding:[bodyCellVerticalPadding,{s:`${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,m:`${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,l:`${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`}]})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["/* eslint-disable react/jsx-no-useless-fragment */\nimport type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AA+EiB"} */"),StyledEmptyTd=styled("td",{target:"e6k8n1r3",label:"StyledEmptyTd"})(({emptyTableContentHeight})=>({textAlign:"center",...useResponsiveValue({height:emptyTableContentHeight})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/DataTable/TableBody.tsx","sources":["src/components/DataTable/TableBody.tsx"],"sourcesContent":["/* eslint-disable react/jsx-no-useless-fragment */\nimport type { ChangeEventHandler } from \"react\";\nimport React from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { DataTableColumn, DataTableRow, DataTableRowProps } from \"./types\";\nimport type { TableCellProps } from \"./TableCell\";\nimport TableCell from \"./TableCell\";\nimport { Text } from \"../Typography/Text/Text\";\nimport type { DataTableProps } from \"./DataTable\";\nimport { LoadingSpinner } from \"../LoadingSpinner/LoadingSpinner\";\nimport {\n  useResponsiveStyles,\n  useResponsiveValue,\n} from \"../../shared/mediaQueries\";\nimport { Checkbox } from \"../Form/Checkbox/Checkbox\";\nimport { CHECKBOX_COLUMN } from \"./constants\";\n\nexport type TableBodyProps = {\n  className?: string;\n} & Pick<\n  DataTableProps,\n  | \"columns\"\n  | \"rows\"\n  | \"isLoading\"\n  | \"loadingStateScreenReaderText\"\n  | \"isEmpty\"\n  | \"emptyTableContentHeight\"\n  | \"bodyCellVerticalPadding\"\n  | \"footer\"\n  | \"isFirstColumnSticky\"\n  | \"isLastColumnSticky\"\n  | \"emptyCellContent\"\n  | \"selectedRowIds\"\n  | \"onRowSelectionChange\"\n  | \"getRowProps\"\n> &\n  Pick<TableCellProps, \"isTableScrolledToLeft\" | \"isTableScrolledToRight\">;\n\ntype StyledTrProps = {\n  isSelected?: boolean;\n} & Pick<TableBodyProps, \"footer\"> &\n  Pick<DataTableRowProps, \"verticalAlign\">;\n\nconst StyledTr = styled.tr<StyledTrProps>(\n  ({ theme, footer, isSelected, verticalAlign }) => ({\n    ...(!isSelected && {\n      \"&:hover td\": {\n        backgroundColor: theme.values.color.background.secondary.default,\n      },\n    }),\n\n    verticalAlign,\n\n    ...(!footer && {\n      \"&:last-of-type\": {\n        td: {\n          borderBottom: \"none\",\n        },\n      },\n    }),\n  })\n);\n\ntype StyledTextProps = {\n  alignColumn?: DataTableColumn[\"align\"];\n};\n\nconst StyledText = styled(Text)<StyledTextProps>(({ alignColumn }) => ({\n  ...useResponsiveValue({\n    textAlign: alignColumn,\n  }),\n}));\n\ntype StyledTdProps = TableCellProps &\n  Pick<\n    TableBodyProps,\n    \"isFirstColumnSticky\" | \"isLastColumnSticky\" | \"bodyCellVerticalPadding\"\n  >;\n\nconst StyledTd = styled(TableCell)<StyledTdProps>(\n  ({ theme, bodyCellVerticalPadding }) => ({\n    ...useResponsiveStyles({\n      padding: [\n        bodyCellVerticalPadding,\n        {\n          s: `${theme.variables.size.spacing.s} ${theme.variables.size.spacing.m}`,\n          m: `${theme.variables.size.spacing.m} ${theme.variables.size.spacing.m}`,\n          l: `${theme.variables.size.spacing.l} ${theme.variables.size.spacing.m}`,\n        } as any,\n      ],\n    }),\n  })\n);\n\ntype EmptyTableBodyProps = Pick<\n  TableBodyProps,\n  \"columns\" | \"emptyTableContentHeight\"\n> & {\n  children: React.ReactElement;\n};\n\ntype StyledEmptyTdProps = Pick<EmptyTableBodyProps, \"emptyTableContentHeight\">;\n\nconst StyledEmptyTd = styled.td<StyledEmptyTdProps>(\n  ({ emptyTableContentHeight }) => ({\n    textAlign: \"center\",\n    ...useResponsiveValue({\n      height: emptyTableContentHeight,\n    }),\n  })\n);\n\nfunction EmptyTableBody({\n  children,\n  columns,\n  emptyTableContentHeight,\n}: EmptyTableBodyProps): React.ReactElement {\n  return (\n    <tbody>\n      <tr>\n        <StyledEmptyTd\n          colSpan={columns.length}\n          emptyTableContentHeight={emptyTableContentHeight}\n        >\n          {children}\n        </StyledEmptyTd>\n      </tr>\n    </tbody>\n  );\n}\n\ntype CheckboxColumnCellContentProps = Pick<DataTableRow, \"id\"> &\n  Pick<TableBodyProps, \"selectedRowIds\" | \"onRowSelectionChange\"> &\n  Pick<DataTableRowProps, \"isSelectable\" | \"selectRowCheckboxAriaLabel\">;\n\nfunction CheckboxColumnCellContent({\n  selectRowCheckboxAriaLabel,\n  selectedRowIds,\n  onRowSelectionChange,\n  id,\n  isSelectable,\n}: CheckboxColumnCellContentProps): React.ReactElement {\n  const toggleRowSelection: ChangeEventHandler<HTMLInputElement> = (evt) => {\n    if (evt.target.checked) {\n      onRowSelectionChange([...selectedRowIds, id]);\n    } else {\n      onRowSelectionChange(selectedRowIds.filter((item) => item !== id));\n    }\n  };\n\n  return (\n    <Checkbox\n      size=\"s\"\n      aria-label={selectRowCheckboxAriaLabel}\n      checked={selectedRowIds.includes(id)}\n      disabled={!isSelectable}\n      onChange={toggleRowSelection}\n    />\n  );\n}\n\nfunction TableBody({\n  className,\n  columns,\n  rows = [],\n  bodyCellVerticalPadding,\n  isTableScrolledToLeft,\n  isTableScrolledToRight,\n  isLoading,\n  loadingStateScreenReaderText = \"Loading\",\n  isEmpty,\n  emptyTableContentHeight,\n  footer,\n  children,\n  isFirstColumnSticky,\n  isLastColumnSticky,\n  emptyCellContent,\n  selectedRowIds,\n  onRowSelectionChange,\n  getRowProps,\n}: React.PropsWithChildren<TableBodyProps>): React.ReactElement {\n  if (isLoading) {\n    const customLoadingContent = children && <>{children}</>;\n    const bodyContent = customLoadingContent || (\n      <LoadingSpinner screenReaderText={loadingStateScreenReaderText} />\n    );\n\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        {bodyContent}\n      </EmptyTableBody>\n    );\n  }\n\n  if (isEmpty) {\n    return (\n      <EmptyTableBody\n        columns={columns}\n        emptyTableContentHeight={emptyTableContentHeight}\n      >\n        <>{children}</>\n      </EmptyTableBody>\n    );\n  }\n\n  const rowElements = rows.map((row) => {\n    const {\n      verticalAlign = null,\n      isActive = false,\n      isSelectable = true,\n      selectRowCheckboxAriaLabel = `Select row${row.id}`,\n    } = getRowProps?.(row) || {};\n    const isSelected = selectedRowIds.includes(row.id);\n\n    return (\n      <StyledTr\n        key={row.id}\n        footer={footer}\n        verticalAlign={verticalAlign}\n        isSelected={isSelected}\n      >\n        {columns.map(({ name, align = \"left\", renderCell }) => {\n          let content;\n          if (renderCell) {\n            content = renderCell(row);\n          } else if (name === CHECKBOX_COLUMN.name) {\n            content = (\n              <CheckboxColumnCellContent\n                selectRowCheckboxAriaLabel={selectRowCheckboxAriaLabel}\n                selectedRowIds={selectedRowIds}\n                onRowSelectionChange={onRowSelectionChange}\n                id={row.id}\n                isSelectable={isSelectable}\n              />\n            );\n          } else {\n            content = (\n              <StyledText size=\"s\" alignColumn={align}>\n                {row[name] ? row[name] : emptyCellContent}\n              </StyledText>\n            );\n          }\n\n          return (\n            <StyledTd\n              key={name}\n              alignColumn={align}\n              isFirstColumnSticky={isFirstColumnSticky}\n              isLastColumnSticky={isLastColumnSticky}\n              bodyCellVerticalPadding={bodyCellVerticalPadding}\n              isTableScrolledToLeft={isTableScrolledToLeft}\n              isTableScrolledToRight={isTableScrolledToRight}\n              isActive={isActive}\n              isSelected={isSelected}\n            >\n              {content}\n            </StyledTd>\n          );\n        })}\n      </StyledTr>\n    );\n  });\n\n  return <tbody className={className}>{rowElements}</tbody>;\n}\n\nexport default React.memo(TableBody);\n"],"names":[],"mappings":"AAuGsB"} */");function EmptyTableBody({children,columns,emptyTableContentHeight}){return React.createElement("tbody",null,React.createElement("tr",null,React.createElement(StyledEmptyTd,{colSpan:columns.length,emptyTableContentHeight:emptyTableContentHeight},children)))}function CheckboxColumnCellContent({selectRowCheckboxAriaLabel,selectedRowIds,onRowSelectionChange,id,isSelectable}){return React.createElement(Checkbox,{size:"s","aria-label":selectRowCheckboxAriaLabel,checked:selectedRowIds.includes(id),disabled:!isSelectable,onChange:evt=>{evt.target.checked?onRowSelectionChange([...selectedRowIds,id]):onRowSelectionChange(selectedRowIds.filter(item=>item!==id))}})}export default React.memo(function({className,columns,rows=[],bodyCellVerticalPadding,isTableScrolledToLeft,isTableScrolledToRight,isLoading,loadingStateScreenReaderText="Loading",isEmpty,emptyTableContentHeight,footer,children,isFirstColumnSticky,isLastColumnSticky,emptyCellContent,selectedRowIds,onRowSelectionChange,getRowProps}){if(isLoading){let bodyContent=children&&React.createElement(React.Fragment,null,children)||React.createElement(LoadingSpinner,{screenReaderText:loadingStateScreenReaderText});return React.createElement(EmptyTableBody,{columns:columns,emptyTableContentHeight:emptyTableContentHeight},bodyContent)}if(isEmpty)return React.createElement(EmptyTableBody,{columns:columns,emptyTableContentHeight:emptyTableContentHeight},React.createElement(React.Fragment,null,children));let rowElements=rows.map(row=>{let{verticalAlign=null,isActive=!1,isSelectable=!0,selectRowCheckboxAriaLabel=`Select row${row.id}`}=getRowProps?.(row)||{},isSelected=selectedRowIds.includes(row.id);return React.createElement(StyledTr,{key:row.id,footer:footer,verticalAlign:verticalAlign,isSelected:isSelected},columns.map(({name,align="left",renderCell})=>{let content;return content=renderCell?renderCell(row):name===CHECKBOX_COLUMN.name?React.createElement(CheckboxColumnCellContent,{selectRowCheckboxAriaLabel:selectRowCheckboxAriaLabel,selectedRowIds:selectedRowIds,onRowSelectionChange:onRowSelectionChange,id:row.id,isSelectable:isSelectable}):React.createElement(StyledText,{size:"s",alignColumn:align},row[name]?row[name]:emptyCellContent),React.createElement(StyledTd,{key:name,alignColumn:align,isFirstColumnSticky:isFirstColumnSticky,isLastColumnSticky:isLastColumnSticky,bodyCellVerticalPadding:bodyCellVerticalPadding,isTableScrolledToLeft:isTableScrolledToLeft,isTableScrolledToRight:isTableScrolledToRight,isActive:isActive,isSelected:isSelected},content)}))});return React.createElement("tbody",{className:className},rowElements)});
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
import React from "react";
|
|
2
2
|
import type { EntityListProps } from "./types";
|
|
3
|
-
export declare function BaseEntityList({ data, size, onClick, isSelectable, selectedIds, onSelectionChange, renderRightContent, getLeftIconProps, hideBorder, filterFn, isVirtualized, maxHeight, "data-e2e-test-id": dataE2eTestId, }: EntityListProps): React.ReactElement;
|
|
3
|
+
export declare function BaseEntityList({ data, size, onClick, isSelectable, selectedIds, onSelectionChange, onSelectionToggle, renderRightContent, getLeftIconProps, hideBorder, filterFn, isVirtualized, maxHeight, "data-e2e-test-id": dataE2eTestId, }: EntityListProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useRef,useState,useEffect,useMemo,useCallback}from"react";import styled from"@emotion/styled";import{StyledList}from"./styled-components";import{useKeyboard}from"../../shared/useKeyboard";import{Icon}from"../Icon/Icon";import{EntityListItem}from"./EntityListItem";import{VirtualizedEntityList}from"./VirtualizedEntityList";let StyledMinHeightCell=styled("div",{target:"ek7zfhr0",label:"StyledMinHeightCell"})(({theme,textSize})=>({minHeight:"s"===textSize?`calc(${theme.variables.size.spacing.m} + ${theme.variables.size.spacing.xxs})`:theme.variables.size.spacing.l,display:"flex",alignItems:"center"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/EntityList/BaseEntityList.tsx","sources":["src/components/EntityList/BaseEntityList.tsx"],"sourcesContent":["import React, {\n  useRef,\n  useState,\n  useEffect,\n  useMemo,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { StyledList } from \"./styled-components\";\nimport type { EntityListProps, ListNode, TextSizeProps } from \"./types\";\nimport { useKeyboard } from \"../../shared/useKeyboard\";\nimport { Icon } from \"../Icon/Icon\";\nimport { EntityListItem } from \"./EntityListItem\";\nimport { VirtualizedEntityList } from \"./VirtualizedEntityList\";\n\n// This minHeight is needed for align small icons in the middle of 1 row text,\n// but it should stay at the top if there is multiline text\nconst StyledMinHeightCell = styled.div<TextSizeProps>(\n  ({ theme, textSize }) => ({\n    minHeight:\n      textSize === \"s\"\n        ? `calc(${theme.variables.size.spacing.m} + ${theme.variables.size.spacing.xxs})`\n        : theme.variables.size.spacing.l,\n    display: \"flex\",\n    alignItems: \"center\",\n  })\n);\n\nconst findEnabledItemIndex = (\n  items: ListNode[],\n  startIndex: number,\n  direction = \"next\"\n): number => {\n  if (direction === \"next\") {\n    for (let i = startIndex + 1; i < items.length; i += 1) {\n      if (!items[i].isDisabled) {\n        return i;\n      }\n    }\n  } else if (direction === \"previous\") {\n    for (let i = startIndex - 1; i >= 0; i -= 1) {\n      if (!items[i].isDisabled) {\n        return i;\n      }\n    }\n  }\n  return startIndex;\n};\n\nexport function BaseEntityList({\n  data,\n  size = \"m\",\n  onClick,\n  isSelectable,\n  selectedIds,\n  onSelectionChange,\n  renderRightContent,\n  getLeftIconProps,\n  hideBorder,\n  filterFn,\n  isVirtualized,\n  maxHeight,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: EntityListProps): React.ReactElement {\n  const dataShown = useMemo(() => {\n    if (!data.length) return data;\n\n    return filterFn ? data.filter(filterFn) : data;\n  }, [data, filterFn]);\n\n  const selectedIdsSet = useMemo(() => new Set(selectedIds), [selectedIds]);\n\n  const [isKeyboardFocus, setIsKeyboardFocus] = useState<boolean>(false);\n  const [selectedIndex, setSelectedIndex] = useState<number>(() =>\n    isSelectable || onClick ? findEnabledItemIndex(dataShown, -1) : -1\n  );\n  const refList = useRef<HTMLUListElement>(null);\n  useKeyboard(\n    {\n      ArrowDown: () => {\n        setIsKeyboardFocus(true);\n        const nextIndex = findEnabledItemIndex(dataShown, selectedIndex);\n        setSelectedIndex(nextIndex);\n      },\n      ArrowUp: () => {\n        setIsKeyboardFocus(true);\n        const prevIndex = findEnabledItemIndex(\n          dataShown,\n          selectedIndex,\n          \"previous\"\n        );\n        setSelectedIndex(prevIndex);\n      },\n    },\n    refList,\n    selectedIndex !== -1\n  );\n\n  useEffect(() => {\n    if (refList && refList.current && isKeyboardFocus && selectedIndex >= 0) {\n      const buttons = refList.current.querySelectorAll(\n        'div[data-id=\"list-item\"]'\n      );\n\n      (buttons[Math.abs(selectedIndex) % buttons.length] as HTMLElement).focus({\n        preventScroll: true,\n      });\n    }\n  }, [selectedIndex, isKeyboardFocus]);\n\n  const handleOnClick = useCallback(\n    (item: ListNode) => {\n      const { id } = item;\n\n      // Handle checkbox change\n      if (isSelectable) {\n        if (selectedIdsSet.has(id)) {\n          selectedIdsSet.delete(id);\n        } else {\n          selectedIdsSet.add(id);\n        }\n        onSelectionChange?.(Array.from(selectedIdsSet));\n      }\n\n      onClick?.(item);\n    },\n    [selectedIdsSet, onSelectionChange, onClick, isSelectable]\n  );\n\n  const handleBlur = useCallback(() => {\n    // Use setTimeout to wait for focus to fully propagate\n    setTimeout(() => {\n      if (\n        refList.current &&\n        !refList.current.contains(document.activeElement)\n      ) {\n        setIsKeyboardFocus(false);\n        setSelectedIndex(() => findEnabledItemIndex(dataShown, -1));\n      }\n    }, 0);\n  }, [refList, setSelectedIndex, dataShown]);\n\n  const renderEntityItem = (index: number): React.ReactNode => {\n    const item = dataShown[index];\n    if (!item) return null;\n\n    const isLastItem = index === dataShown.length - 1;\n    const tabIndex = !item.isDisabled && index === selectedIndex ? 0 : -1;\n    const isSelected = selectedIdsSet.has(item.id);\n\n    return (\n      <EntityListItem\n        key={item.id}\n        aria-label={item[\"aria-label\"] || item.label}\n        size={size}\n        hideBorder={hideBorder || isLastItem}\n        isActive={item.isActive || (isKeyboardFocus && selectedIndex === index)}\n        isDisabled={item.isDisabled}\n        isClickable={Boolean(onClick || selectedIds)}\n        description={item.description}\n        onKeyDown={() => handleOnClick(item)}\n        tabIndex={tabIndex}\n        onFocus={() => {\n          if (onClick || isSelectable) {\n            setSelectedIndex(index);\n            setIsKeyboardFocus(true);\n          }\n        }}\n        onClick={() => {\n          setIsKeyboardFocus(false);\n          handleOnClick(item);\n        }}\n        checkboxProps={\n          isSelectable && {\n            name: \"list-checkbox\",\n            size: \"s\",\n            onChange: () => handleOnClick(item),\n            checked: isSelected,\n          }\n        }\n        renderLabel={() => item.label}\n        renderLeft={\n          getLeftIconProps\n            ? ({ textSize }) => (\n                <Icon {...getLeftIconProps(item)} size={textSize} />\n              )\n            : null\n        }\n        renderRight={({ actionSize, textSize }) =>\n          renderRightContent ? (\n            <StyledMinHeightCell textSize={textSize}>\n              {renderRightContent({ textSize, actionSize, ...item })}\n            </StyledMinHeightCell>\n          ) : null\n        }\n      />\n    );\n  };\n\n  return isVirtualized && maxHeight ? (\n    <VirtualizedEntityList\n      id={dataE2eTestId}\n      maxHeight={maxHeight}\n      size={size}\n      dataShown={dataShown}\n      selectedIndex={selectedIndex}\n      itemTemplate={renderEntityItem}\n      data-e2e-test-id={dataE2eTestId}\n    />\n  ) : (\n    <StyledList\n      ref={refList}\n      size={size}\n      data-e2e-test-id={dataE2eTestId}\n      onBlur={handleBlur}\n      data-ds-id=\"EntityList\"\n      role=\"list\"\n    >\n      {dataShown.map((_item, i) => renderEntityItem(i))}\n    </StyledList>\n  );\n}\n"],"names":[],"mappings":"AAiB4B"} */"),findEnabledItemIndex=(items,startIndex,direction="next")=>{if("next"===direction){for(let i=startIndex+1;i<items.length;i+=1)if(!items[i].isDisabled)return i}else if("previous"===direction){for(let i=startIndex-1;i>=0;i-=1)if(!items[i].isDisabled)return i}return startIndex};export function BaseEntityList({data,size="m",onClick,isSelectable,selectedIds,onSelectionChange,renderRightContent,getLeftIconProps,hideBorder,filterFn,isVirtualized,maxHeight,"data-e2e-test-id":dataE2eTestId}){let dataShown=useMemo(()=>data.length&&filterFn?data.filter(filterFn):data,[data,filterFn]),selectedIdsSet=useMemo(()=>new Set(selectedIds),[selectedIds]),[isKeyboardFocus,setIsKeyboardFocus]=useState(!1),[selectedIndex,setSelectedIndex]=useState(()=>isSelectable||onClick?findEnabledItemIndex(dataShown,-1):-1),refList=useRef(null);useKeyboard({ArrowDown:()=>{setIsKeyboardFocus(!0),setSelectedIndex(findEnabledItemIndex(dataShown,selectedIndex))},ArrowUp:()=>{setIsKeyboardFocus(!0),setSelectedIndex(findEnabledItemIndex(dataShown,selectedIndex,"previous"))}},refList,-1!==selectedIndex),useEffect(()=>{if(refList&&refList.current&&isKeyboardFocus&&selectedIndex>=0){let buttons=refList.current.querySelectorAll('div[data-id="list-item"]');buttons[Math.abs(selectedIndex)%buttons.length].focus({preventScroll:!0})}},[selectedIndex,isKeyboardFocus]);let handleOnClick=useCallback(item=>{let{id}=item;isSelectable&&(selectedIdsSet.has(id)?selectedIdsSet.delete(id):selectedIdsSet.add(id),onSelectionChange?.(Array.from(selectedIdsSet))),onClick?.(item)},[selectedIdsSet,onSelectionChange,onClick,isSelectable]),handleBlur=useCallback(()=>{setTimeout(()=>{refList.current&&!refList.current.contains(document.activeElement)&&(setIsKeyboardFocus(!1),setSelectedIndex(()=>findEnabledItemIndex(dataShown,-1)))},0)},[refList,setSelectedIndex,dataShown]),renderEntityItem=index=>{let item=dataShown[index];if(!item)return null;let isLastItem=index===dataShown.length-1,tabIndex=item.isDisabled||index!==selectedIndex?-1:0,isSelected=selectedIdsSet.has(item.id);return React.createElement(EntityListItem,{key:item.id,"aria-label":item["aria-label"]||item.label,size:size,hideBorder:hideBorder||isLastItem,isActive:item.isActive||isKeyboardFocus&&selectedIndex===index,isDisabled:item.isDisabled,isClickable:!!(onClick||selectedIds),description:item.description,onKeyDown:()=>handleOnClick(item),tabIndex:tabIndex,onFocus:()=>{(onClick||isSelectable)&&(setSelectedIndex(index),setIsKeyboardFocus(!0))},onClick:()=>{setIsKeyboardFocus(!1),handleOnClick(item)},checkboxProps:isSelectable&&{name:"list-checkbox",size:"s",onChange:()=>handleOnClick(item),checked:isSelected},renderLabel:()=>item.label,renderLeft:getLeftIconProps?({textSize})=>React.createElement(Icon,{...getLeftIconProps(item),size:textSize}):null,renderRight:({actionSize,textSize})=>renderRightContent?React.createElement(StyledMinHeightCell,{textSize:textSize},renderRightContent({textSize,actionSize,...item})):null})};return isVirtualized&&maxHeight?React.createElement(VirtualizedEntityList,{id:dataE2eTestId,maxHeight:maxHeight,size:size,dataShown:dataShown,selectedIndex:selectedIndex,itemTemplate:renderEntityItem,"data-e2e-test-id":dataE2eTestId}):React.createElement(StyledList,{ref:refList,size:size,"data-e2e-test-id":dataE2eTestId,onBlur:handleBlur,"data-ds-id":"EntityList",role:"list"},dataShown.map((_item,i)=>renderEntityItem(i)))}
|
|
1
|
+
import React,{useRef,useState,useEffect,useMemo,useCallback}from"react";import styled from"@emotion/styled";import{StyledList}from"./styled-components";import{useKeyboard}from"../../shared/useKeyboard";import{Icon}from"../Icon/Icon";import{EntityListItem}from"./EntityListItem";import{VirtualizedEntityList}from"./VirtualizedEntityList";let StyledMinHeightCell=styled("div",{target:"e1lukbw40",label:"StyledMinHeightCell"})(({theme,textSize})=>({minHeight:"s"===textSize?`calc(${theme.variables.size.spacing.m} + ${theme.variables.size.spacing.xxs})`:theme.variables.size.spacing.l,display:"flex",alignItems:"center"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/EntityList/BaseEntityList.tsx","sources":["src/components/EntityList/BaseEntityList.tsx"],"sourcesContent":["import React, {\n  useRef,\n  useState,\n  useEffect,\n  useMemo,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { StyledList } from \"./styled-components\";\nimport type { EntityListProps, ListNode, TextSizeProps } from \"./types\";\nimport { useKeyboard } from \"../../shared/useKeyboard\";\nimport { Icon } from \"../Icon/Icon\";\nimport { EntityListItem } from \"./EntityListItem\";\nimport { VirtualizedEntityList } from \"./VirtualizedEntityList\";\n\n// This minHeight is needed for align small icons in the middle of 1 row text,\n// but it should stay at the top if there is multiline text\nconst StyledMinHeightCell = styled.div<TextSizeProps>(\n  ({ theme, textSize }) => ({\n    minHeight:\n      textSize === \"s\"\n        ? `calc(${theme.variables.size.spacing.m} + ${theme.variables.size.spacing.xxs})`\n        : theme.variables.size.spacing.l,\n    display: \"flex\",\n    alignItems: \"center\",\n  })\n);\n\nconst findEnabledItemIndex = (\n  items: ListNode[],\n  startIndex: number,\n  direction = \"next\"\n): number => {\n  if (direction === \"next\") {\n    for (let i = startIndex + 1; i < items.length; i += 1) {\n      if (!items[i].isDisabled) {\n        return i;\n      }\n    }\n  } else if (direction === \"previous\") {\n    for (let i = startIndex - 1; i >= 0; i -= 1) {\n      if (!items[i].isDisabled) {\n        return i;\n      }\n    }\n  }\n  return startIndex;\n};\n\nexport function BaseEntityList({\n  data,\n  size = \"m\",\n  onClick,\n  isSelectable,\n  selectedIds,\n  onSelectionChange,\n  onSelectionToggle,\n  renderRightContent,\n  getLeftIconProps,\n  hideBorder,\n  filterFn,\n  isVirtualized,\n  maxHeight,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: EntityListProps): React.ReactElement {\n  const dataShown = useMemo(() => {\n    if (!data.length) return data;\n\n    return filterFn ? data.filter(filterFn) : data;\n  }, [data, filterFn]);\n\n  const selectedIdsSet = useMemo(() => new Set(selectedIds), [selectedIds]);\n\n  const [isKeyboardFocus, setIsKeyboardFocus] = useState<boolean>(false);\n  const [selectedIndex, setSelectedIndex] = useState<number>(() =>\n    isSelectable || onClick ? findEnabledItemIndex(dataShown, -1) : -1\n  );\n  const refList = useRef<HTMLUListElement>(null);\n  useKeyboard(\n    {\n      ArrowDown: () => {\n        setIsKeyboardFocus(true);\n        const nextIndex = findEnabledItemIndex(dataShown, selectedIndex);\n        setSelectedIndex(nextIndex);\n      },\n      ArrowUp: () => {\n        setIsKeyboardFocus(true);\n        const prevIndex = findEnabledItemIndex(\n          dataShown,\n          selectedIndex,\n          \"previous\"\n        );\n        setSelectedIndex(prevIndex);\n      },\n    },\n    refList,\n    selectedIndex !== -1\n  );\n\n  useEffect(() => {\n    if (refList && refList.current && isKeyboardFocus && selectedIndex >= 0) {\n      const buttons = refList.current.querySelectorAll(\n        'div[data-id=\"list-item\"]'\n      );\n\n      (buttons[Math.abs(selectedIndex) % buttons.length] as HTMLElement).focus({\n        preventScroll: true,\n      });\n    }\n  }, [selectedIndex, isKeyboardFocus]);\n\n  const handleOnClick = useCallback(\n    (item: ListNode) => {\n      const { id } = item;\n\n      if (isSelectable) {\n        const willBeSelected = !selectedIdsSet.has(id);\n        const newSelectedIdsSet = new Set(selectedIdsSet);\n\n        if (willBeSelected) {\n          newSelectedIdsSet.add(id);\n        } else {\n          newSelectedIdsSet.delete(id);\n        }\n\n        onSelectionToggle?.(id, willBeSelected);\n        onSelectionChange?.(Array.from(newSelectedIdsSet));\n      }\n\n      onClick?.(item);\n    },\n    [\n      selectedIdsSet,\n      onSelectionChange,\n      onSelectionToggle,\n      onClick,\n      isSelectable,\n    ]\n  );\n\n  const handleBlur = useCallback(() => {\n    // Use setTimeout to wait for focus to fully propagate\n    setTimeout(() => {\n      if (\n        refList.current &&\n        !refList.current.contains(document.activeElement)\n      ) {\n        setIsKeyboardFocus(false);\n        setSelectedIndex(() => findEnabledItemIndex(dataShown, -1));\n      }\n    }, 0);\n  }, [refList, setSelectedIndex, dataShown]);\n\n  const renderEntityItem = (index: number): React.ReactNode => {\n    const item = dataShown[index];\n    if (!item) return null;\n\n    const isLastItem = index === dataShown.length - 1;\n    const tabIndex = !item.isDisabled && index === selectedIndex ? 0 : -1;\n    const isSelected = selectedIdsSet.has(item.id);\n\n    return (\n      <EntityListItem\n        key={item.id}\n        aria-label={item[\"aria-label\"] || item.label}\n        size={size}\n        hideBorder={hideBorder || isLastItem}\n        isActive={item.isActive || (isKeyboardFocus && selectedIndex === index)}\n        isDisabled={item.isDisabled}\n        isClickable={Boolean(onClick || selectedIds)}\n        description={item.description}\n        onKeyDown={() => handleOnClick(item)}\n        tabIndex={tabIndex}\n        onFocus={() => {\n          if (onClick || isSelectable) {\n            setSelectedIndex(index);\n            setIsKeyboardFocus(true);\n          }\n        }}\n        onClick={() => {\n          setIsKeyboardFocus(false);\n          handleOnClick(item);\n        }}\n        checkboxProps={\n          isSelectable && {\n            name: \"list-checkbox\",\n            size: \"s\",\n            checked: isSelected,\n            onClick: (e: React.MouseEvent<HTMLInputElement>) => {\n              e.stopPropagation();\n              e.preventDefault();\n              handleOnClick(item);\n            },\n          }\n        }\n        renderLabel={() => item.label}\n        renderLeft={\n          getLeftIconProps\n            ? ({ textSize }) => (\n                <Icon {...getLeftIconProps(item)} size={textSize} />\n              )\n            : null\n        }\n        renderRight={({ actionSize, textSize }) =>\n          renderRightContent ? (\n            <StyledMinHeightCell textSize={textSize}>\n              {renderRightContent({ textSize, actionSize, ...item })}\n            </StyledMinHeightCell>\n          ) : null\n        }\n      />\n    );\n  };\n\n  return isVirtualized && maxHeight ? (\n    <VirtualizedEntityList\n      id={dataE2eTestId}\n      maxHeight={maxHeight}\n      size={size}\n      dataShown={dataShown}\n      selectedIndex={selectedIndex}\n      itemTemplate={renderEntityItem}\n      data-e2e-test-id={dataE2eTestId}\n    />\n  ) : (\n    <StyledList\n      ref={refList}\n      size={size}\n      data-e2e-test-id={dataE2eTestId}\n      onBlur={handleBlur}\n      data-ds-id=\"EntityList\"\n      role=\"list\"\n    >\n      {dataShown.map((_item, i) => renderEntityItem(i))}\n    </StyledList>\n  );\n}\n"],"names":[],"mappings":"AAiB4B"} */"),findEnabledItemIndex=(items,startIndex,direction="next")=>{if("next"===direction){for(let i=startIndex+1;i<items.length;i+=1)if(!items[i].isDisabled)return i}else if("previous"===direction){for(let i=startIndex-1;i>=0;i-=1)if(!items[i].isDisabled)return i}return startIndex};export function BaseEntityList({data,size="m",onClick,isSelectable,selectedIds,onSelectionChange,onSelectionToggle,renderRightContent,getLeftIconProps,hideBorder,filterFn,isVirtualized,maxHeight,"data-e2e-test-id":dataE2eTestId}){let dataShown=useMemo(()=>data.length&&filterFn?data.filter(filterFn):data,[data,filterFn]),selectedIdsSet=useMemo(()=>new Set(selectedIds),[selectedIds]),[isKeyboardFocus,setIsKeyboardFocus]=useState(!1),[selectedIndex,setSelectedIndex]=useState(()=>isSelectable||onClick?findEnabledItemIndex(dataShown,-1):-1),refList=useRef(null);useKeyboard({ArrowDown:()=>{setIsKeyboardFocus(!0),setSelectedIndex(findEnabledItemIndex(dataShown,selectedIndex))},ArrowUp:()=>{setIsKeyboardFocus(!0),setSelectedIndex(findEnabledItemIndex(dataShown,selectedIndex,"previous"))}},refList,-1!==selectedIndex),useEffect(()=>{if(refList&&refList.current&&isKeyboardFocus&&selectedIndex>=0){let buttons=refList.current.querySelectorAll('div[data-id="list-item"]');buttons[Math.abs(selectedIndex)%buttons.length].focus({preventScroll:!0})}},[selectedIndex,isKeyboardFocus]);let handleOnClick=useCallback(item=>{let{id}=item;if(isSelectable){let willBeSelected=!selectedIdsSet.has(id),newSelectedIdsSet=new Set(selectedIdsSet);willBeSelected?newSelectedIdsSet.add(id):newSelectedIdsSet.delete(id),onSelectionToggle?.(id,willBeSelected),onSelectionChange?.(Array.from(newSelectedIdsSet))}onClick?.(item)},[selectedIdsSet,onSelectionChange,onSelectionToggle,onClick,isSelectable]),handleBlur=useCallback(()=>{setTimeout(()=>{refList.current&&!refList.current.contains(document.activeElement)&&(setIsKeyboardFocus(!1),setSelectedIndex(()=>findEnabledItemIndex(dataShown,-1)))},0)},[refList,setSelectedIndex,dataShown]),renderEntityItem=index=>{let item=dataShown[index];if(!item)return null;let isLastItem=index===dataShown.length-1,tabIndex=item.isDisabled||index!==selectedIndex?-1:0,isSelected=selectedIdsSet.has(item.id);return React.createElement(EntityListItem,{key:item.id,"aria-label":item["aria-label"]||item.label,size:size,hideBorder:hideBorder||isLastItem,isActive:item.isActive||isKeyboardFocus&&selectedIndex===index,isDisabled:item.isDisabled,isClickable:!!(onClick||selectedIds),description:item.description,onKeyDown:()=>handleOnClick(item),tabIndex:tabIndex,onFocus:()=>{(onClick||isSelectable)&&(setSelectedIndex(index),setIsKeyboardFocus(!0))},onClick:()=>{setIsKeyboardFocus(!1),handleOnClick(item)},checkboxProps:isSelectable&&{name:"list-checkbox",size:"s",checked:isSelected,onClick:e=>{e.stopPropagation(),e.preventDefault(),handleOnClick(item)}},renderLabel:()=>item.label,renderLeft:getLeftIconProps?({textSize})=>React.createElement(Icon,{...getLeftIconProps(item),size:textSize}):null,renderRight:({actionSize,textSize})=>renderRightContent?React.createElement(StyledMinHeightCell,{textSize:textSize},renderRightContent({textSize,actionSize,...item})):null})};return isVirtualized&&maxHeight?React.createElement(VirtualizedEntityList,{id:dataE2eTestId,maxHeight:maxHeight,size:size,dataShown:dataShown,selectedIndex:selectedIndex,itemTemplate:renderEntityItem,"data-e2e-test-id":dataE2eTestId}):React.createElement(StyledList,{ref:refList,size:size,"data-e2e-test-id":dataE2eTestId,onBlur:handleBlur,"data-ds-id":"EntityList",role:"list"},dataShown.map((_item,i)=>renderEntityItem(i)))}
|