@amboss/design-system 3.32.0 → 3.32.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/Form/DateInput/DateInput.js +1 -1
- package/build/cjs/components/Form/Datepicker/DatePickerInput.d.ts +20 -0
- package/build/cjs/components/Form/Datepicker/DatePickerInput.js +1 -0
- package/build/cjs/components/Form/Datepicker/Datepicker.d.ts +21 -0
- package/build/cjs/components/Form/Datepicker/Datepicker.js +1 -0
- package/build/cjs/components/Form/Datepicker/Datepicker.types.d.ts +40 -0
- package/build/cjs/components/Form/Datepicker/Datepicker.types.js +1 -0
- package/build/cjs/components/Form/Datepicker/DatepickerButton.d.ts +11 -0
- package/build/cjs/components/Form/Datepicker/DatepickerButton.js +1 -0
- package/build/cjs/components/Form/Datepicker/index.d.ts +3 -0
- package/build/cjs/components/Form/Datepicker/index.js +1 -0
- package/build/cjs/components/Form/PasswordInput/PasswordInput.js +1 -1
- package/build/cjs/components/Form/PasswordInput/PasswordInputButton.js +1 -1
- package/build/cjs/components/Form/dateMask.d.ts +12 -0
- package/build/cjs/components/Form/dateMask.js +1 -0
- package/build/cjs/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.js +1 -1
- package/build/cjs/components/Tabs/-types.d.ts +39 -0
- package/build/cjs/components/Tabs/-types.js +1 -0
- package/build/cjs/components/Tabs/Tabs.d.ts +3 -42
- package/build/cjs/components/Tabs/Tabs.js +1 -1
- package/build/cjs/components/Tabs/useTabFocus.d.ts +13 -0
- package/build/cjs/components/Tabs/useTabFocus.js +1 -0
- package/build/esm/components/Form/DateInput/DateInput.js +1 -1
- package/build/esm/components/Form/Datepicker/DatePickerInput.d.ts +20 -0
- package/build/esm/components/Form/Datepicker/DatePickerInput.js +1 -0
- package/build/esm/components/Form/Datepicker/Datepicker.d.ts +21 -0
- package/build/esm/components/Form/Datepicker/Datepicker.js +1 -0
- package/build/esm/components/Form/Datepicker/Datepicker.types.d.ts +40 -0
- package/build/esm/components/Form/Datepicker/Datepicker.types.js +1 -0
- package/build/esm/components/Form/Datepicker/DatepickerButton.d.ts +11 -0
- package/build/esm/components/Form/Datepicker/DatepickerButton.js +1 -0
- package/build/esm/components/Form/Datepicker/index.d.ts +3 -0
- package/build/esm/components/Form/Datepicker/index.js +1 -0
- package/build/esm/components/Form/PasswordInput/PasswordInput.js +1 -1
- package/build/esm/components/Form/PasswordInput/PasswordInputButton.js +1 -1
- package/build/esm/components/Form/dateMask.d.ts +12 -0
- package/build/esm/components/Form/dateMask.js +1 -0
- package/build/esm/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.js +1 -1
- package/build/esm/components/Tabs/-types.d.ts +39 -0
- package/build/esm/components/Tabs/-types.js +1 -0
- package/build/esm/components/Tabs/Tabs.d.ts +3 -42
- package/build/esm/components/Tabs/Tabs.js +1 -1
- package/build/esm/components/Tabs/useTabFocus.d.ts +13 -0
- package/build/esm/components/Tabs/useTabFocus.js +1 -0
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useRef,useEffect}from"react";import styled from"@emotion/styled";import{Button}from"../../Button/Button";import{Icon}from"../../Icon/Icon";import{Inline}from"../../Inline/Inline";import{useIsScrollable}from"../useIsScrollable";import{CarouselThumbnail}from"../CarouselThumbnail/CarouselThumbnail";let StyledWrapper=styled("div",{target:"e1k5s29y0",label:"StyledWrapper"})(({showNavButtons})=>({width:"100%",...showNavButtons?{padding:"0 40px"}:{padding:"0 8px"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                idx\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={idx}\n                  isActive={idx === currentIndex}\n                  ref={idx === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAWsB"} */"),StyledButtonWrapper=styled("div",{target:"e1k5s29y1",label:"StyledButtonWrapper"})(({alignRight=!1,isScrollable,theme})=>({position:"absolute",top:"0",bottom:"0",left:alignRight?"auto":"0",right:alignRight?"0":"auto",height:"100%",width:40,display:"flex",boxShadow:isScrollable?theme.values.elevation[1]:"none",zIndex:1,button:{width:40}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                idx\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={idx}\n                  isActive={idx === currentIndex}\n                  ref={idx === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAwB4B"} */"),StyledOuterContainer=styled("div",{target:"e1k5s29y2",label:"StyledOuterContainer"})(({theme})=>({position:"relative",display:"flex",flexWrap:"nowrap",justifyContent:"center",overflowX:"auto",overflowY:"hidden",paddingTop:theme.variables.size.spacing.xxs,scrollBehavior:"smooth",width:"100%",scrollbarWidth:"auto",scrollbarColor:`${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,"::-webkit-scrollbar":{height:"8px"},"::-webkit-scrollbar-thumb":{background:theme.values.color.tag.background.gray},"::-webkit-scrollbar-track":{background:theme.values.color.background.transparent.active}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                idx\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={idx}\n                  isActive={idx === currentIndex}\n                  ref={idx === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AA2C6B"} */"),StyledInnerContainer=styled("div",{target:"e1k5s29y3",label:"StyledInnerContainer"})({display:"flex",flexWrap:"nowrap",maxWidth:"100%"},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                idx\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={idx}\n                  isActive={idx === currentIndex}\n                  ref={idx === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAkE6B"} */");export const MediaCarousel=({activeItemRef,currentIndex,mediaAssets,nextBtnAriaLabel,onClickNext,onClickPrevious,onClickThumbnail,prevBtnAriaLabel,setIsReady,showNavButtons,skipArrowKeysListener,...ariaAttributes})=>{let thumbnailCount=mediaAssets.length,scrollContainerRef=useRef(null),innerContainerRef=useRef(null),isScrollable=useIsScrollable(scrollContainerRef,innerContainerRef),currentIndexRef=useRef(currentIndex);useEffect(()=>{currentIndexRef.current=currentIndex},[currentIndex]),useEffect(()=>{activeItemRef.current&&scrollContainerRef.current&&innerContainerRef.current&&setIsReady(()=>!0)},[activeItemRef,scrollContainerRef,innerContainerRef,setIsReady]);let handleClickNext=()=>{if(currentIndexRef.current!==thumbnailCount-1&&(onClickNext(),isScrollable&&scrollContainerRef.current&&activeItemRef.current)){let itemWidth=activeItemRef.current.offsetWidth+8;scrollContainerRef.current?.scrollBy?.({left:itemWidth,behavior:"smooth"})}},handleClickPrevious=()=>{if(!(currentIndexRef.current<=0)&&(onClickPrevious(),isScrollable&&scrollContainerRef.current&&activeItemRef.current)){let itemWidth=activeItemRef.current.offsetWidth+8;scrollContainerRef.current?.scrollBy?.({left:-itemWidth,behavior:"smooth"})}};return useEffect(()=>{if(skipArrowKeysListener)return;let currentRef=scrollContainerRef.current;if(currentRef)return currentRef.addEventListener("keydown",listener),()=>{currentRef.removeEventListener("keydown",listener)};function listener(event){"ArrowRight"===event.key&&handleClickNext(),"ArrowLeft"===event.key&&handleClickPrevious()}},[scrollContainerRef]),React.createElement(StyledWrapper,{showNavButtons:showNavButtons},showNavButtons&&React.createElement(StyledButtonWrapper,{isScrollable:isScrollable},React.createElement(Button,{variant:"tertiary",onClick:handleClickPrevious,disabled:0===currentIndex,size:"s","aria-label":prevBtnAriaLabel},React.createElement(Icon,{name:"chevron-left",color:"secondary"}))),React.createElement(StyledOuterContainer,{ref:scrollContainerRef,"aria-label":"thumbnails scrollable container",...ariaAttributes},React.createElement(StyledInnerContainer,{ref:innerContainerRef},React.createElement(Inline,{noWrap:!0,space:"xxxs"},mediaAssets.map(({src,title,bottomLeftIconName,topLeftIconName,xid},idx)=>React.createElement(CarouselThumbnail,{key:`carousel-item-${xid}`,assetIndex:idx,isActive:idx===currentIndex,ref:idx===currentIndex?activeItemRef:null,src:src,title:title,topLeftIconName:topLeftIconName,bottomLeftIconName:bottomLeftIconName,onClickThumbnail:onClickThumbnail}))))),showNavButtons&&React.createElement(StyledButtonWrapper,{alignRight:!0,isScrollable:isScrollable},React.createElement(Button,{variant:"tertiary",onClick:handleClickNext,disabled:currentIndex===thumbnailCount-1,size:"s","aria-label":nextBtnAriaLabel},React.createElement(Icon,{name:"chevron-right",color:"secondary"}))))};
|
|
1
|
+
import React,{useRef,useEffect}from"react";import styled from"@emotion/styled";import{Button}from"../../Button/Button";import{Icon}from"../../Icon/Icon";import{Inline}from"../../Inline/Inline";import{useIsScrollable}from"../useIsScrollable";import{CarouselThumbnail}from"../CarouselThumbnail/CarouselThumbnail";let StyledWrapper=styled("div",{target:"e6wa5tu0",label:"StyledWrapper"})(({showNavButtons})=>({width:"100%",...showNavButtons?{padding:"0 40px"}:{padding:"0 8px"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                index\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={index}\n                  isActive={index === currentIndex}\n                  ref={index === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAWsB"} */"),StyledButtonWrapper=styled("div",{target:"e6wa5tu1",label:"StyledButtonWrapper"})(({alignRight=!1,isScrollable,theme})=>({position:"absolute",top:"0",bottom:"0",left:alignRight?"auto":"0",right:alignRight?"0":"auto",height:"100%",width:40,display:"flex",boxShadow:isScrollable?theme.values.elevation[1]:"none",zIndex:1,button:{width:40}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                index\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={index}\n                  isActive={index === currentIndex}\n                  ref={index === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAwB4B"} */"),StyledOuterContainer=styled("div",{target:"e6wa5tu2",label:"StyledOuterContainer"})(({theme})=>({position:"relative",display:"flex",flexWrap:"nowrap",justifyContent:"center",overflowX:"auto",overflowY:"hidden",paddingTop:theme.variables.size.spacing.xxs,scrollBehavior:"smooth",width:"100%",scrollbarWidth:"auto",scrollbarColor:`${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,"::-webkit-scrollbar":{height:"8px"},"::-webkit-scrollbar-thumb":{background:theme.values.color.tag.background.gray},"::-webkit-scrollbar-track":{background:theme.values.color.background.transparent.active}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                index\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={index}\n                  isActive={index === currentIndex}\n                  ref={index === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AA2C6B"} */"),StyledInnerContainer=styled("div",{target:"e6wa5tu3",label:"StyledInnerContainer"})({display:"flex",flexWrap:"nowrap",maxWidth:"100%"},"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx","sources":["src/components/MediaViewerCarousel/MediaCarousel/MediaCarousel.tsx"],"sourcesContent":["/* eslint-disable consistent-return */\nimport React, { useRef, useEffect } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Button } from \"../../Button/Button\";\nimport { Icon, type IconProps } from \"../../Icon/Icon\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { useIsScrollable } from \"../useIsScrollable\";\nimport { CarouselThumbnail } from \"../CarouselThumbnail/CarouselThumbnail\";\n\nconst BUTTON_WIDTH = 40;\n\nconst StyledWrapper = styled.div<{\n  showNavButtons: boolean;\n}>(({ showNavButtons }) => ({\n  width: \"100%\",\n  ...(showNavButtons\n    ? {\n        padding: `0 ${BUTTON_WIDTH}px`,\n      }\n    : {\n        padding: \"0 8px\",\n      }),\n}));\n\nconst StyledButtonWrapper = styled.div<{\n  alignRight?: boolean;\n  isScrollable: boolean;\n}>(({ alignRight = false, isScrollable, theme }) => ({\n  position: \"absolute\",\n  top: \"0\",\n  bottom: \"0\",\n  left: alignRight ? \"auto\" : \"0\",\n  right: alignRight ? \"0\" : \"auto\",\n  height: \"100%\",\n  width: BUTTON_WIDTH,\n  display: \"flex\",\n  boxShadow: isScrollable ? theme.values.elevation[1] : \"none\",\n  zIndex: 1,\n  button: {\n    width: BUTTON_WIDTH,\n  },\n}));\n\nconst StyledOuterContainer = styled.div(({ theme }) => ({\n  position: \"relative\",\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  justifyContent: \"center\",\n  overflowX: \"auto\",\n  overflowY: \"hidden\",\n  paddingTop: theme.variables.size.spacing.xxs,\n  scrollBehavior: \"smooth\",\n  width: \"100%\",\n  scrollbarWidth: \"auto\",\n  scrollbarColor: `${theme.values.color.tag.background.gray} ${theme.values.color.background.transparent.active}`,\n  \"::-webkit-scrollbar\": {\n    height: \"8px\",\n  },\n  \"::-webkit-scrollbar-thumb\": {\n    background: theme.values.color.tag.background.gray,\n  },\n  \"::-webkit-scrollbar-track\": {\n    background: theme.values.color.background.transparent.active,\n  },\n}));\n\nconst StyledInnerContainer = styled.div({\n  display: \"flex\",\n  flexWrap: \"nowrap\",\n  maxWidth: \"100%\",\n});\n\ntype MediaAsset = {\n  bottomLeftIconName?: IconProps[\"name\"];\n  src: string;\n  title: string;\n  topLeftIconName?: IconProps[\"name\"];\n  xid: string;\n};\n\nexport type MediaCarouselProps = {\n  activeItemRef?: React.MutableRefObject<HTMLButtonElement | null>;\n  currentIndex: number;\n  mediaAssets: MediaAsset[];\n  nextBtnAriaLabel: string;\n  onClickNext: VoidFunction;\n  onClickPrevious: VoidFunction;\n  onClickThumbnail: (newIndex: number) => void;\n  prevBtnAriaLabel: string;\n  setIsReady: React.Dispatch<React.SetStateAction<boolean>>;\n  showNavButtons: boolean;\n  skipArrowKeysListener: boolean;\n};\n\nexport const MediaCarousel = ({\n  activeItemRef,\n  currentIndex,\n  mediaAssets,\n  nextBtnAriaLabel,\n  onClickNext,\n  onClickPrevious,\n  onClickThumbnail,\n  prevBtnAriaLabel,\n  setIsReady,\n  showNavButtons,\n  skipArrowKeysListener,\n  ...ariaAttributes\n}: MediaCarouselProps): React.ReactNode => {\n  const thumbnailCount = mediaAssets.length;\n  const scrollContainerRef = useRef<HTMLDivElement>(null);\n  const innerContainerRef = useRef<HTMLDivElement>(null);\n  const isScrollable = useIsScrollable(scrollContainerRef, innerContainerRef);\n  const currentIndexRef = useRef(currentIndex);\n\n  useEffect(() => {\n    currentIndexRef.current = currentIndex;\n  }, [currentIndex]);\n\n  useEffect(() => {\n    if (\n      !(\n        activeItemRef.current &&\n        scrollContainerRef.current &&\n        innerContainerRef.current\n      )\n    )\n      return;\n\n    // ensures that we scroll to the active item on first mount\n    setIsReady(() => true);\n  }, [activeItemRef, scrollContainerRef, innerContainerRef, setIsReady]);\n\n  const handleClickNext = () => {\n    if (currentIndexRef.current === thumbnailCount - 1) return;\n    onClickNext();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  const handleClickPrevious = () => {\n    if (currentIndexRef.current <= 0) return;\n    onClickPrevious();\n    if (isScrollable && scrollContainerRef.current && activeItemRef.current) {\n      const itemWidth = activeItemRef.current.offsetWidth + 8; // 8px for margins (4px on each side)\n      scrollContainerRef.current?.scrollBy?.({\n        left: -itemWidth,\n        behavior: \"smooth\",\n      });\n    }\n  };\n\n  useEffect(() => {\n    if (skipArrowKeysListener) return;\n    const currentRef = scrollContainerRef.current;\n    if (!currentRef) return;\n\n    function listener(event: KeyboardEvent) {\n      if (event.key === \"ArrowRight\") {\n        handleClickNext();\n      }\n      if (event.key === \"ArrowLeft\") {\n        handleClickPrevious();\n      }\n    }\n\n    currentRef.addEventListener(\"keydown\", listener);\n\n    return () => {\n      currentRef.removeEventListener(\"keydown\", listener);\n    };\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [scrollContainerRef]);\n\n  return (\n    <StyledWrapper showNavButtons={showNavButtons}>\n      {showNavButtons && (\n        <StyledButtonWrapper isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickPrevious}\n            disabled={currentIndex === 0}\n            size=\"s\"\n            aria-label={prevBtnAriaLabel}\n          >\n            <Icon name=\"chevron-left\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n      <StyledOuterContainer\n        ref={scrollContainerRef}\n        aria-label=\"thumbnails scrollable container\"\n        {...ariaAttributes}\n      >\n        <StyledInnerContainer ref={innerContainerRef}>\n          <Inline noWrap space=\"xxxs\">\n            {mediaAssets.map(\n              (\n                { src, title, bottomLeftIconName, topLeftIconName, xid },\n                index\n              ) => (\n                <CarouselThumbnail\n                  key={`carousel-item-${xid}`}\n                  assetIndex={index}\n                  isActive={index === currentIndex}\n                  ref={index === currentIndex ? activeItemRef : null}\n                  src={src}\n                  title={title}\n                  topLeftIconName={topLeftIconName}\n                  bottomLeftIconName={bottomLeftIconName}\n                  onClickThumbnail={onClickThumbnail}\n                />\n              )\n            )}\n          </Inline>\n        </StyledInnerContainer>\n      </StyledOuterContainer>\n      {showNavButtons && (\n        <StyledButtonWrapper alignRight isScrollable={isScrollable}>\n          <Button\n            variant=\"tertiary\"\n            onClick={handleClickNext}\n            disabled={currentIndex === thumbnailCount - 1}\n            size=\"s\"\n            aria-label={nextBtnAriaLabel}\n          >\n            <Icon name=\"chevron-right\" color=\"secondary\" />\n          </Button>\n        </StyledButtonWrapper>\n      )}\n    </StyledWrapper>\n  );\n};\n"],"names":[],"mappings":"AAkE6B"} */");export const MediaCarousel=({activeItemRef,currentIndex,mediaAssets,nextBtnAriaLabel,onClickNext,onClickPrevious,onClickThumbnail,prevBtnAriaLabel,setIsReady,showNavButtons,skipArrowKeysListener,...ariaAttributes})=>{let thumbnailCount=mediaAssets.length,scrollContainerRef=useRef(null),innerContainerRef=useRef(null),isScrollable=useIsScrollable(scrollContainerRef,innerContainerRef),currentIndexRef=useRef(currentIndex);useEffect(()=>{currentIndexRef.current=currentIndex},[currentIndex]),useEffect(()=>{activeItemRef.current&&scrollContainerRef.current&&innerContainerRef.current&&setIsReady(()=>!0)},[activeItemRef,scrollContainerRef,innerContainerRef,setIsReady]);let handleClickNext=()=>{if(currentIndexRef.current!==thumbnailCount-1&&(onClickNext(),isScrollable&&scrollContainerRef.current&&activeItemRef.current)){let itemWidth=activeItemRef.current.offsetWidth+8;scrollContainerRef.current?.scrollBy?.({left:itemWidth,behavior:"smooth"})}},handleClickPrevious=()=>{if(!(currentIndexRef.current<=0)&&(onClickPrevious(),isScrollable&&scrollContainerRef.current&&activeItemRef.current)){let itemWidth=activeItemRef.current.offsetWidth+8;scrollContainerRef.current?.scrollBy?.({left:-itemWidth,behavior:"smooth"})}};return useEffect(()=>{if(skipArrowKeysListener)return;let currentRef=scrollContainerRef.current;if(currentRef)return currentRef.addEventListener("keydown",listener),()=>{currentRef.removeEventListener("keydown",listener)};function listener(event){"ArrowRight"===event.key&&handleClickNext(),"ArrowLeft"===event.key&&handleClickPrevious()}},[scrollContainerRef]),React.createElement(StyledWrapper,{showNavButtons:showNavButtons},showNavButtons&&React.createElement(StyledButtonWrapper,{isScrollable:isScrollable},React.createElement(Button,{variant:"tertiary",onClick:handleClickPrevious,disabled:0===currentIndex,size:"s","aria-label":prevBtnAriaLabel},React.createElement(Icon,{name:"chevron-left",color:"secondary"}))),React.createElement(StyledOuterContainer,{ref:scrollContainerRef,"aria-label":"thumbnails scrollable container",...ariaAttributes},React.createElement(StyledInnerContainer,{ref:innerContainerRef},React.createElement(Inline,{noWrap:!0,space:"xxxs"},mediaAssets.map(({src,title,bottomLeftIconName,topLeftIconName,xid},index)=>React.createElement(CarouselThumbnail,{key:`carousel-item-${xid}`,assetIndex:index,isActive:index===currentIndex,ref:index===currentIndex?activeItemRef:null,src:src,title:title,topLeftIconName:topLeftIconName,bottomLeftIconName:bottomLeftIconName,onClickThumbnail:onClickThumbnail}))))),showNavButtons&&React.createElement(StyledButtonWrapper,{alignRight:!0,isScrollable:isScrollable},React.createElement(Button,{variant:"tertiary",onClick:handleClickNext,disabled:currentIndex===thumbnailCount-1,size:"s","aria-label":nextBtnAriaLabel},React.createElement(Icon,{name:"chevron-right",color:"secondary"}))))};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import type React from "react";
|
|
2
|
+
import type { MQ, SpaceSizes } from "../../types";
|
|
3
|
+
import type { IconName } from "../Icon/Icon";
|
|
4
|
+
export type IconProp = {
|
|
5
|
+
name: IconName;
|
|
6
|
+
position: "left" | "right";
|
|
7
|
+
};
|
|
8
|
+
/**
|
|
9
|
+
* @param label - The text to display in the tab
|
|
10
|
+
* @param sublabel - A subtext, encased in parentheses
|
|
11
|
+
* @param icon - An icon to display. Can be either an icon name (defaults to left position) or an object with name and position
|
|
12
|
+
* @param as - The element to render the tab as, defaults to `button`
|
|
13
|
+
* @param disabled - Whether the tab is disabled
|
|
14
|
+
* @param tooltipContent - The content of the tooltip to display when the tab is disabled
|
|
15
|
+
*/
|
|
16
|
+
export type Tab = {
|
|
17
|
+
label: string;
|
|
18
|
+
sublabel?: string;
|
|
19
|
+
icon?: IconName | IconProp;
|
|
20
|
+
as?: "a" | "button" | React.ElementType;
|
|
21
|
+
disabled?: boolean;
|
|
22
|
+
active?: boolean;
|
|
23
|
+
tooltipContent?: string;
|
|
24
|
+
} & Partial<React.ComponentProps<"button">> & Partial<React.ComponentProps<"a">>;
|
|
25
|
+
export type TabsProps = {
|
|
26
|
+
/** An array of tabs, contains `header`, `subtitle`, `iconLeft`, `as` (defining the tab as a button or a link) */
|
|
27
|
+
tabs: Tab[];
|
|
28
|
+
children?: React.ReactElement;
|
|
29
|
+
/** The index of the active tab */
|
|
30
|
+
activeTab?: number | null;
|
|
31
|
+
onTabSelect?: (selectedTab: number) => void;
|
|
32
|
+
activeTabClickable?: boolean;
|
|
33
|
+
tabPanelId?: string;
|
|
34
|
+
/** Horizontal padding space for the tab container, suited for tabs within a Card element */
|
|
35
|
+
horizontalPadding?: SpaceSizes | MQ<SpaceSizes>;
|
|
36
|
+
/** Whether to show a bottom border between tabs and content */
|
|
37
|
+
hideBottomBorder?: boolean;
|
|
38
|
+
"data-e2e-test-id"?: string;
|
|
39
|
+
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export{};
|
|
@@ -1,43 +1,4 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import type {
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
name: IconName;
|
|
6
|
-
position: "left" | "right";
|
|
7
|
-
};
|
|
8
|
-
/**
|
|
9
|
-
* @param label - The text to display in the tab
|
|
10
|
-
* @param sublabel - A subtext, encased in parentheses
|
|
11
|
-
* @param icon - An icon to display. Can be either an icon name (defaults to left position) or an object with name and position
|
|
12
|
-
* @param as - The element to render the tab as, defaults to `button`
|
|
13
|
-
* @param disabled - Whether the tab is disabled
|
|
14
|
-
* @param tooltipContent - The content of the tooltip to display when the tab is disabled
|
|
15
|
-
*/
|
|
16
|
-
type Tab = {
|
|
17
|
-
id?: string;
|
|
18
|
-
label: string;
|
|
19
|
-
sublabel?: string;
|
|
20
|
-
icon?: IconName | IconProp;
|
|
21
|
-
as?: "a" | "button" | React.ElementType;
|
|
22
|
-
disabled?: boolean;
|
|
23
|
-
active?: boolean;
|
|
24
|
-
tooltipContent?: string;
|
|
25
|
-
} & Partial<React.ComponentProps<"button">> & Partial<React.ComponentProps<"a">>;
|
|
26
|
-
export type TabsProps = {
|
|
27
|
-
/** An array of tabs, contains `header`, `subtitle`, `iconLeft`, `as` (defining the tab as a button or a link) */
|
|
28
|
-
tabs: Tab[];
|
|
29
|
-
children?: React.ReactElement;
|
|
30
|
-
/** The index of the active tab */
|
|
31
|
-
activeTab?: number | null;
|
|
32
|
-
onTabSelect?: (selectedTab: number) => void;
|
|
33
|
-
activeTabClickable?: boolean;
|
|
34
|
-
tabPanelId?: string;
|
|
35
|
-
/** Horizontal padding space for the tab container, suited for tabs within a Card element */
|
|
36
|
-
horizontalPadding?: SpaceSizes | MQ<SpaceSizes>;
|
|
37
|
-
/** Whether to show a bottom border between tabs and content */
|
|
38
|
-
hideBottomBorder?: boolean;
|
|
39
|
-
"aria-label"?: string;
|
|
40
|
-
"data-e2e-test-id"?: string;
|
|
41
|
-
};
|
|
42
|
-
export declare function Tabs({ tabs, horizontalPadding, activeTab, children, onTabSelect, activeTabClickable, tabPanelId, "aria-label": ariaLabel, "data-e2e-test-id": dataE2eTestId, hideBottomBorder, ...ariaAttributes }: TabsProps): React.ReactElement;
|
|
43
|
-
export {};
|
|
2
|
+
import type { TabsProps } from "./-types";
|
|
3
|
+
export type { TabsProps };
|
|
4
|
+
export declare function Tabs({ tabs, horizontalPadding, activeTab, children, onTabSelect, activeTabClickable, tabPanelId: tabPanelIdProp, "data-e2e-test-id": dataE2eTestId, hideBottomBorder, ...ariaAttributes }: TabsProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import styled from"@emotion/styled";import isPropValid from"@emotion/is-prop-valid";import React from"react";import{useResponsiveStyles}from"../../shared/mediaQueries";import{Icon}from"../Icon/Icon";import{Stack}from"../Stack/Stack";import{Tooltip}from"../Tooltip/Tooltip";let StyledContainer=styled("div",{shouldForwardProp:prop=>isPropValid(prop),target:"ea907xn0",label:"StyledContainer"})(({theme,horizontalPadding,hideBottomBorder})=>({display:"flex",flexDirection:"row",...useResponsiveStyles({gap:[["m","l","l"],theme.variables.size.spacing],paddingInline:[horizontalPadding,theme.variables.size.spacing]}),...!hideBottomBorder&&{borderBottom:`1px solid ${theme.values.color.border.secondary.default}`},whiteSpace:"nowrap",overflow:"auto",scrollbarWidth:"none",msOverflowStyle:"none","::-webkit-scrollbar":{display:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tabs/Tabs.tsx","sources":["src/components/Tabs/Tabs.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport styled from \"@emotion/styled\";\nimport isPropValid from \"@emotion/is-prop-valid\";\nimport React from \"react\";\nimport { useResponsiveStyles } from \"../../shared/mediaQueries\";\nimport type { MQ, SpaceSizes } from \"../../types\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Stack } from \"../Stack/Stack\";\nimport { Tooltip } from \"../Tooltip/Tooltip\";\n\nconst CONTAINER_BORDER_WIDTH = 1;\nconst TAB_BORDER_WIDTH = 4;\n\nconst StyledContainer = styled(\"div\", {\n  shouldForwardProp: (prop) => isPropValid(prop),\n})<Pick<TabsProps, \"horizontalPadding\" | \"hideBottomBorder\">>(\n  ({ theme, horizontalPadding, hideBottomBorder }) => ({\n    display: \"flex\",\n    flexDirection: \"row\",\n    ...useResponsiveStyles({\n      gap: [[\"m\", \"l\", \"l\"], theme.variables.size.spacing],\n      paddingInline: [horizontalPadding, theme.variables.size.spacing],\n    }),\n\n    ...(!hideBottomBorder && {\n      borderBottom: `${CONTAINER_BORDER_WIDTH}px solid ${theme.values.color.border.secondary.default}`,\n    }),\n\n    whiteSpace: \"nowrap\",\n\n    // Scrollbar\n    overflow: \"auto\",\n    scrollbarWidth: \"none\", // Firefox\n    msOverflowStyle: \"none\", // IE 10+\n    \"::-webkit-scrollbar\": {\n      display: \"none\", // Chrome\n    },\n  })\n);\n\nconst StyledTab = styled(\"button\", {\n  shouldForwardProp: (prop) => isPropValid(prop) && prop !== \"active\",\n})<\n  Pick<TabsProps, \"hideBottomBorder\" | \"activeTabClickable\"> &\n    Pick<Tab, \"disabled\" | \"active\">\n>(({ theme, active, disabled, hideBottomBorder, activeTabClickable }) => ({\n  display: \"flex\",\n  gap: theme.variables.size.spacing.xxs,\n  alignItems: \"center\",\n\n  background: \"none\",\n  border: 0,\n  boxSizing: \"border-box\",\n  color: theme.values.color.text.secondary.default,\n  padding: 0,\n  fontFamily: theme.variables.fontFamily.lato,\n  fontWeight: theme.variables.weight.bold,\n  fontSize: theme.variables.size.font.s,\n  lineHeight: theme.variables.size.lineHeight.xs,\n  textDecoration: \"none\",\n\n  borderBottom: `${TAB_BORDER_WIDTH}px solid transparent`,\n  // since we have a bottom border on the container,\n  // we need to add padding to the top of the tab to maintain consistent height\n  paddingTop: hideBottomBorder\n    ? theme.variables.size.spacing.m\n    : `calc(${theme.variables.size.spacing.m} - ${CONTAINER_BORDER_WIDTH}px)`,\n  paddingBottom: `calc(${theme.variables.size.spacing.m} - ${TAB_BORDER_WIDTH}px)`,\n\n  ...(active &&\n    !disabled && {\n      color: theme.values.color.text.accent.default,\n      borderBottomColor: theme.values.color.border.accent.default,\n    }),\n  ...(!active &&\n    !disabled && {\n      cursor: \"pointer\",\n      \"&:hover\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.default,\n      },\n      \"&:active\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.active,\n      },\n    }),\n  ...(activeTabClickable && {\n    cursor: \"pointer\",\n  }),\n  ...(disabled && {\n    opacity: theme.variables.opacity.disabled,\n    cursor: \"not-allowed\",\n  }),\n}));\n\nconst addSubtitle = (text: string) => (text ? `(${text})` : ``);\ntype IconProp = { name: IconName; position: \"left\" | \"right\" };\n\n/**\n * @param label - The text to display in the tab\n * @param sublabel - A subtext, encased in parentheses\n * @param icon - An icon to display. Can be either an icon name (defaults to left position) or an object with name and position\n * @param as - The element to render the tab as, defaults to `button`\n * @param disabled - Whether the tab is disabled\n * @param tooltipContent - The content of the tooltip to display when the tab is disabled\n */\ntype Tab = {\n  id?: string;\n  label: string;\n  sublabel?: string;\n  icon?: IconName | IconProp;\n  as?: \"a\" | \"button\" | React.ElementType;\n  disabled?: boolean;\n  active?: boolean;\n  tooltipContent?: string;\n} & Partial<React.ComponentProps<\"button\">> &\n  Partial<React.ComponentProps<\"a\">>;\n\nexport type TabsProps = {\n  /** An array of tabs, contains `header`, `subtitle`, `iconLeft`, `as` (defining the tab as a button or a link) */\n  tabs: Tab[];\n  children?: React.ReactElement;\n  /** The index of the active tab */\n  activeTab?: number | null;\n  onTabSelect?: (selectedTab: number) => void;\n  activeTabClickable?: boolean;\n  tabPanelId?: string;\n  /** Horizontal padding space for the tab container, suited for tabs within a Card element */\n  horizontalPadding?: SpaceSizes | MQ<SpaceSizes>; // 'm' | ['m', 'l', 'l']\n  /** Whether to show a bottom border between tabs and content */\n  hideBottomBorder?: boolean;\n  \"aria-label\"?: string;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport function Tabs({\n  tabs,\n  horizontalPadding,\n  activeTab = 0,\n  children,\n  onTabSelect,\n  activeTabClickable,\n  tabPanelId = \"tabPanel\",\n  \"aria-label\": ariaLabel,\n  \"data-e2e-test-id\": dataE2eTestId,\n  hideBottomBorder,\n  ...ariaAttributes\n}: TabsProps): React.ReactElement {\n  return (\n    <Stack data-e2e-test-id={dataE2eTestId} data-ds-id=\"Tabs\">\n      <StyledContainer\n        hideBottomBorder={hideBottomBorder}\n        horizontalPadding={horizontalPadding}\n        {...ariaAttributes}\n        role=\"tablist\"\n        aria-label={ariaLabel}\n      >\n        {tabs.map(\n          (\n            {\n              id: tabId,\n              label,\n              sublabel,\n              icon,\n              as = \"button\",\n              disabled,\n              tooltipContent,\n              ...rest\n            },\n            i\n          ) => {\n            // Parse icon configuration\n            let iconConfig: IconProp;\n\n            if (icon) {\n              if (typeof icon === \"string\") {\n                iconConfig = { name: icon, position: \"left\" };\n              } else {\n                iconConfig = icon;\n              }\n            }\n\n            const Tab = (\n              <StyledTab\n                id={tabId || `tab-${i}`}\n                activeTabClickable={activeTabClickable}\n                hideBottomBorder={hideBottomBorder}\n                disabled={disabled}\n                as={as}\n                key={label}\n                active={activeTab === i}\n                onClick={() => onTabSelect && onTabSelect(i)}\n                {...rest}\n                type={as !== \"button\" ? undefined : \"button\"}\n                role=\"tab\"\n                aria-selected={activeTab === i}\n                aria-controls={children ? tabPanelId : undefined}\n              >\n                {iconConfig && iconConfig.position === \"left\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n                {label} {addSubtitle(sublabel)}\n                {iconConfig && iconConfig.position === \"right\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n              </StyledTab>\n            );\n\n            if (disabled && tooltipContent) {\n              return <Tooltip content={tooltipContent}>{Tab}</Tooltip>;\n            }\n\n            return Tab;\n          }\n        )}\n      </StyledContainer>\n      {children && (\n        <div role=\"tabpanel\" id={tabPanelId}>\n          {children}\n        </div>\n      )}\n    </Stack>\n  );\n}\n"],"names":[],"mappings":"AAewB"} */"),StyledTab=styled("button",{shouldForwardProp:prop=>isPropValid(prop)&&"active"!==prop,target:"ea907xn1",label:"StyledTab"})(({theme,active,disabled,hideBottomBorder,activeTabClickable})=>({display:"flex",gap:theme.variables.size.spacing.xxs,alignItems:"center",background:"none",border:0,boxSizing:"border-box",color:theme.values.color.text.secondary.default,padding:0,fontFamily:theme.variables.fontFamily.lato,fontWeight:theme.variables.weight.bold,fontSize:theme.variables.size.font.s,lineHeight:theme.variables.size.lineHeight.xs,textDecoration:"none",borderBottom:"4px solid transparent",paddingTop:hideBottomBorder?theme.variables.size.spacing.m:`calc(${theme.variables.size.spacing.m} - 1px)`,paddingBottom:`calc(${theme.variables.size.spacing.m} - 4px)`,...active&&!disabled&&{color:theme.values.color.text.accent.default,borderBottomColor:theme.values.color.border.accent.default},...!active&&!disabled&&{cursor:"pointer","&:hover":{color:theme.values.color.text.primary.default,borderBottomColor:theme.values.color.border.secondary.default},"&:active":{color:theme.values.color.text.primary.default,borderBottomColor:theme.values.color.border.secondary.active}},...activeTabClickable&&{cursor:"pointer"},...disabled&&{opacity:theme.variables.opacity.disabled,cursor:"not-allowed"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tabs/Tabs.tsx","sources":["src/components/Tabs/Tabs.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport styled from \"@emotion/styled\";\nimport isPropValid from \"@emotion/is-prop-valid\";\nimport React from \"react\";\nimport { useResponsiveStyles } from \"../../shared/mediaQueries\";\nimport type { MQ, SpaceSizes } from \"../../types\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Stack } from \"../Stack/Stack\";\nimport { Tooltip } from \"../Tooltip/Tooltip\";\n\nconst CONTAINER_BORDER_WIDTH = 1;\nconst TAB_BORDER_WIDTH = 4;\n\nconst StyledContainer = styled(\"div\", {\n  shouldForwardProp: (prop) => isPropValid(prop),\n})<Pick<TabsProps, \"horizontalPadding\" | \"hideBottomBorder\">>(\n  ({ theme, horizontalPadding, hideBottomBorder }) => ({\n    display: \"flex\",\n    flexDirection: \"row\",\n    ...useResponsiveStyles({\n      gap: [[\"m\", \"l\", \"l\"], theme.variables.size.spacing],\n      paddingInline: [horizontalPadding, theme.variables.size.spacing],\n    }),\n\n    ...(!hideBottomBorder && {\n      borderBottom: `${CONTAINER_BORDER_WIDTH}px solid ${theme.values.color.border.secondary.default}`,\n    }),\n\n    whiteSpace: \"nowrap\",\n\n    // Scrollbar\n    overflow: \"auto\",\n    scrollbarWidth: \"none\", // Firefox\n    msOverflowStyle: \"none\", // IE 10+\n    \"::-webkit-scrollbar\": {\n      display: \"none\", // Chrome\n    },\n  })\n);\n\nconst StyledTab = styled(\"button\", {\n  shouldForwardProp: (prop) => isPropValid(prop) && prop !== \"active\",\n})<\n  Pick<TabsProps, \"hideBottomBorder\" | \"activeTabClickable\"> &\n    Pick<Tab, \"disabled\" | \"active\">\n>(({ theme, active, disabled, hideBottomBorder, activeTabClickable }) => ({\n  display: \"flex\",\n  gap: theme.variables.size.spacing.xxs,\n  alignItems: \"center\",\n\n  background: \"none\",\n  border: 0,\n  boxSizing: \"border-box\",\n  color: theme.values.color.text.secondary.default,\n  padding: 0,\n  fontFamily: theme.variables.fontFamily.lato,\n  fontWeight: theme.variables.weight.bold,\n  fontSize: theme.variables.size.font.s,\n  lineHeight: theme.variables.size.lineHeight.xs,\n  textDecoration: \"none\",\n\n  borderBottom: `${TAB_BORDER_WIDTH}px solid transparent`,\n  // since we have a bottom border on the container,\n  // we need to add padding to the top of the tab to maintain consistent height\n  paddingTop: hideBottomBorder\n    ? theme.variables.size.spacing.m\n    : `calc(${theme.variables.size.spacing.m} - ${CONTAINER_BORDER_WIDTH}px)`,\n  paddingBottom: `calc(${theme.variables.size.spacing.m} - ${TAB_BORDER_WIDTH}px)`,\n\n  ...(active &&\n    !disabled && {\n      color: theme.values.color.text.accent.default,\n      borderBottomColor: theme.values.color.border.accent.default,\n    }),\n  ...(!active &&\n    !disabled && {\n      cursor: \"pointer\",\n      \"&:hover\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.default,\n      },\n      \"&:active\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.active,\n      },\n    }),\n  ...(activeTabClickable && {\n    cursor: \"pointer\",\n  }),\n  ...(disabled && {\n    opacity: theme.variables.opacity.disabled,\n    cursor: \"not-allowed\",\n  }),\n}));\n\nconst addSubtitle = (text: string) => (text ? `(${text})` : ``);\ntype IconProp = { name: IconName; position: \"left\" | \"right\" };\n\n/**\n * @param label - The text to display in the tab\n * @param sublabel - A subtext, encased in parentheses\n * @param icon - An icon to display. Can be either an icon name (defaults to left position) or an object with name and position\n * @param as - The element to render the tab as, defaults to `button`\n * @param disabled - Whether the tab is disabled\n * @param tooltipContent - The content of the tooltip to display when the tab is disabled\n */\ntype Tab = {\n  id?: string;\n  label: string;\n  sublabel?: string;\n  icon?: IconName | IconProp;\n  as?: \"a\" | \"button\" | React.ElementType;\n  disabled?: boolean;\n  active?: boolean;\n  tooltipContent?: string;\n} & Partial<React.ComponentProps<\"button\">> &\n  Partial<React.ComponentProps<\"a\">>;\n\nexport type TabsProps = {\n  /** An array of tabs, contains `header`, `subtitle`, `iconLeft`, `as` (defining the tab as a button or a link) */\n  tabs: Tab[];\n  children?: React.ReactElement;\n  /** The index of the active tab */\n  activeTab?: number | null;\n  onTabSelect?: (selectedTab: number) => void;\n  activeTabClickable?: boolean;\n  tabPanelId?: string;\n  /** Horizontal padding space for the tab container, suited for tabs within a Card element */\n  horizontalPadding?: SpaceSizes | MQ<SpaceSizes>; // 'm' | ['m', 'l', 'l']\n  /** Whether to show a bottom border between tabs and content */\n  hideBottomBorder?: boolean;\n  \"aria-label\"?: string;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport function Tabs({\n  tabs,\n  horizontalPadding,\n  activeTab = 0,\n  children,\n  onTabSelect,\n  activeTabClickable,\n  tabPanelId = \"tabPanel\",\n  \"aria-label\": ariaLabel,\n  \"data-e2e-test-id\": dataE2eTestId,\n  hideBottomBorder,\n  ...ariaAttributes\n}: TabsProps): React.ReactElement {\n  return (\n    <Stack data-e2e-test-id={dataE2eTestId} data-ds-id=\"Tabs\">\n      <StyledContainer\n        hideBottomBorder={hideBottomBorder}\n        horizontalPadding={horizontalPadding}\n        {...ariaAttributes}\n        role=\"tablist\"\n        aria-label={ariaLabel}\n      >\n        {tabs.map(\n          (\n            {\n              id: tabId,\n              label,\n              sublabel,\n              icon,\n              as = \"button\",\n              disabled,\n              tooltipContent,\n              ...rest\n            },\n            i\n          ) => {\n            // Parse icon configuration\n            let iconConfig: IconProp;\n\n            if (icon) {\n              if (typeof icon === \"string\") {\n                iconConfig = { name: icon, position: \"left\" };\n              } else {\n                iconConfig = icon;\n              }\n            }\n\n            const Tab = (\n              <StyledTab\n                id={tabId || `tab-${i}`}\n                activeTabClickable={activeTabClickable}\n                hideBottomBorder={hideBottomBorder}\n                disabled={disabled}\n                as={as}\n                key={label}\n                active={activeTab === i}\n                onClick={() => onTabSelect && onTabSelect(i)}\n                {...rest}\n                type={as !== \"button\" ? undefined : \"button\"}\n                role=\"tab\"\n                aria-selected={activeTab === i}\n                aria-controls={children ? tabPanelId : undefined}\n              >\n                {iconConfig && iconConfig.position === \"left\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n                {label} {addSubtitle(sublabel)}\n                {iconConfig && iconConfig.position === \"right\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n              </StyledTab>\n            );\n\n            if (disabled && tooltipContent) {\n              return <Tooltip content={tooltipContent}>{Tab}</Tooltip>;\n            }\n\n            return Tab;\n          }\n        )}\n      </StyledContainer>\n      {children && (\n        <div role=\"tabpanel\" id={tabPanelId}>\n          {children}\n        </div>\n      )}\n    </Stack>\n  );\n}\n"],"names":[],"mappings":"AA0CkB"} */"),addSubtitle=text=>text?`(${text})`:"";export function Tabs({tabs,horizontalPadding,activeTab=0,children,onTabSelect,activeTabClickable,tabPanelId="tabPanel","aria-label":ariaLabel,"data-e2e-test-id":dataE2eTestId,hideBottomBorder,...ariaAttributes}){return React.createElement(Stack,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"Tabs"},React.createElement(StyledContainer,{hideBottomBorder:hideBottomBorder,horizontalPadding:horizontalPadding,...ariaAttributes,role:"tablist","aria-label":ariaLabel},tabs.map(({id:tabId,label,sublabel,icon,as="button",disabled,tooltipContent,...rest},i)=>{let iconConfig;icon&&(iconConfig="string"==typeof icon?{name:icon,position:"left"}:icon);let Tab=React.createElement(StyledTab,{id:tabId||`tab-${i}`,activeTabClickable:activeTabClickable,hideBottomBorder:hideBottomBorder,disabled:disabled,as:as,key:label,active:activeTab===i,onClick:()=>onTabSelect&&onTabSelect(i),...rest,type:"button"!==as?void 0:"button",role:"tab","aria-selected":activeTab===i,"aria-controls":children?tabPanelId:void 0},iconConfig&&"left"===iconConfig.position&&React.createElement(Icon,{size:"s",name:iconConfig.name}),label," ",addSubtitle(sublabel),iconConfig&&"right"===iconConfig.position&&React.createElement(Icon,{size:"s",name:iconConfig.name}));return disabled&&tooltipContent?React.createElement(Tooltip,{content:tooltipContent},Tab):Tab})),children&&React.createElement("div",{role:"tabpanel",id:tabPanelId},children))}
|
|
1
|
+
import styled from"@emotion/styled";import isPropValid from"@emotion/is-prop-valid";import React,{useId,useRef}from"react";import{useResponsiveStyles}from"../../shared/mediaQueries";import{useKeyboard}from"../../shared/useKeyboard";import{Icon}from"../Icon/Icon";import{Stack}from"../Stack/Stack";import{Tooltip}from"../Tooltip/Tooltip";import{useTabFocus}from"./useTabFocus";import{getButtonAppearanceReset}from"../../shared/mixins/Button/getButtonAppearanceReset";let StyledContainer=styled("div",{shouldForwardProp:prop=>isPropValid(prop),target:"e2kj0g00",label:"StyledContainer"})(({theme,horizontalPadding,hideBottomBorder})=>({display:"flex",flexDirection:"row",...useResponsiveStyles({gap:[["m","l","l"],theme.variables.size.spacing],paddingInline:[horizontalPadding,theme.variables.size.spacing]}),...!hideBottomBorder&&{borderBottom:`1px solid ${theme.values.color.border.secondary.default}`},whiteSpace:"nowrap",overflow:"auto",scrollbarWidth:"none",msOverflowStyle:"none","::-webkit-scrollbar":{display:"none"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tabs/Tabs.tsx","sources":["src/components/Tabs/Tabs.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport styled from \"@emotion/styled\";\nimport isPropValid from \"@emotion/is-prop-valid\";\nimport React, { useId, useRef } from \"react\";\nimport { useResponsiveStyles } from \"../../shared/mediaQueries\";\nimport { useKeyboard } from \"../../shared/useKeyboard\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Stack } from \"../Stack/Stack\";\nimport { Tooltip } from \"../Tooltip/Tooltip\";\nimport type { IconProp, Tab, TabsProps } from \"./-types\";\nimport { useTabFocus } from \"./useTabFocus\";\nimport { getButtonAppearanceReset } from \"../../shared/mixins/Button/getButtonAppearanceReset\";\n\nexport type { TabsProps };\n\nconst TAB_INDEX_ATTR = \"data-tab-index\";\n\nconst CONTAINER_BORDER_WIDTH = 1;\nconst TAB_BORDER_WIDTH = 4;\n\nconst StyledContainer = styled(\"div\", {\n  shouldForwardProp: (prop) => isPropValid(prop),\n})<Pick<TabsProps, \"horizontalPadding\" | \"hideBottomBorder\">>(\n  ({ theme, horizontalPadding, hideBottomBorder }) => ({\n    display: \"flex\",\n    flexDirection: \"row\",\n    ...useResponsiveStyles({\n      gap: [[\"m\", \"l\", \"l\"], theme.variables.size.spacing],\n      paddingInline: [horizontalPadding, theme.variables.size.spacing],\n    }),\n\n    ...(!hideBottomBorder && {\n      borderBottom: `${CONTAINER_BORDER_WIDTH}px solid ${theme.values.color.border.secondary.default}`,\n    }),\n\n    whiteSpace: \"nowrap\",\n\n    // Scrollbar\n    overflow: \"auto\",\n    scrollbarWidth: \"none\", // Firefox\n    msOverflowStyle: \"none\", // IE 10+\n    \"::-webkit-scrollbar\": {\n      display: \"none\", // Chrome\n    },\n  })\n);\n\nconst StyledTab = styled(\"button\", {\n  shouldForwardProp: (prop) =>\n    isPropValid(prop) && prop !== \"active\" && prop !== \"disabled\",\n})<\n  Pick<TabsProps, \"hideBottomBorder\" | \"activeTabClickable\"> &\n    Pick<Tab, \"disabled\" | \"active\">\n>(({ theme, active, disabled, hideBottomBorder, activeTabClickable }) => ({\n  ...getButtonAppearanceReset(),\n  padding: 0,\n  display: \"flex\",\n  gap: theme.variables.size.spacing.xxs,\n  alignItems: \"center\",\n\n  boxSizing: \"border-box\",\n  color: theme.values.color.text.secondary.default,\n  fontFamily: theme.variables.fontFamily.lato,\n  fontWeight: theme.variables.weight.bold,\n  fontSize: theme.variables.size.font.s,\n  lineHeight: theme.variables.size.lineHeight.xs,\n\n  borderBottom: `${TAB_BORDER_WIDTH}px solid transparent`,\n  // since we have a bottom border on the container,\n  // we need to add padding to the top of the tab to maintain consistent height\n  paddingTop: hideBottomBorder\n    ? theme.variables.size.spacing.m\n    : `calc(${theme.variables.size.spacing.m} - ${CONTAINER_BORDER_WIDTH}px)`,\n  paddingBottom: `calc(${theme.variables.size.spacing.m} - ${TAB_BORDER_WIDTH}px)`,\n\n  ...(active &&\n    !disabled && {\n      color: theme.values.color.text.accent.default,\n      borderBottomColor: theme.values.color.border.accent.default,\n    }),\n  ...(!active &&\n    !disabled && {\n      cursor: \"pointer\",\n      \"&:hover\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.default,\n      },\n      \"&:active\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.active,\n      },\n    }),\n  ...(activeTabClickable && {\n    cursor: \"pointer\",\n  }),\n  ...(disabled && {\n    opacity: theme.variables.opacity.disabled,\n    cursor: \"not-allowed\",\n  }),\n\n  // TODO: The focus outline is clipped by the parent's `overflow: auto` in the scrollable variant.\n  // Adding padding/margin to prevent clipping would need a design review.\n  \"&:focus-visible\": {\n    outlineWidth: 2,\n    outlineOffset: -1,\n  },\n}));\n\nconst addSubtitle = (text: string) => (text ? `(${text})` : ``);\n\nexport function Tabs({\n  tabs,\n  horizontalPadding,\n  activeTab = 0,\n  children,\n  onTabSelect,\n  activeTabClickable,\n  tabPanelId: tabPanelIdProp,\n  \"data-e2e-test-id\": dataE2eTestId,\n  hideBottomBorder,\n  ...ariaAttributes\n}: TabsProps): React.ReactElement {\n  const tablistRef = useRef<HTMLDivElement>(null);\n  const instanceId = useId();\n  const tabPanelId = tabPanelIdProp || `${instanceId}-panel`;\n\n  const { onBlur, onTabFocus, getTabIndex, focusNextTab } = useTabFocus({\n    tabs,\n    activeTab,\n    tablistRef,\n    tabAttribute: TAB_INDEX_ATTR,\n  });\n\n  useKeyboard(\n    {\n      ArrowRight: () => focusNextTab(1),\n      ArrowLeft: () => focusNextTab(-1),\n    },\n    tablistRef,\n    true\n  );\n\n  const getTabId = (index: number) => `${instanceId}-tab-${index}`;\n\n  return (\n    <Stack data-e2e-test-id={dataE2eTestId} data-ds-id=\"Tabs\">\n      <StyledContainer\n        {...ariaAttributes}\n        hideBottomBorder={hideBottomBorder}\n        horizontalPadding={horizontalPadding}\n        onBlur={onBlur}\n        ref={tablistRef}\n        aria-orientation=\"horizontal\"\n        role=\"tablist\"\n      >\n        {tabs.map(\n          (\n            {\n              label,\n              sublabel,\n              icon,\n              as = \"button\",\n              disabled,\n              tooltipContent,\n              ...rest\n            },\n            i\n          ) => {\n            // Parse icon configuration\n            let iconConfig: IconProp;\n\n            if (icon) {\n              if (typeof icon === \"string\") {\n                iconConfig = { name: icon, position: \"left\" };\n              } else {\n                iconConfig = icon;\n              }\n            }\n\n            const Tab = (\n              <StyledTab\n                id={getTabId(i)}\n                activeTabClickable={activeTabClickable}\n                hideBottomBorder={hideBottomBorder}\n                disabled={disabled}\n                as={as}\n                key={label}\n                active={activeTab === i}\n                {...rest}\n                onFocus={(e) => {\n                  onTabFocus(i);\n                  rest.onFocus?.(e);\n                }}\n                onClick={(e) => {\n                  if (disabled) return;\n                  onTabSelect?.(i);\n                  rest.onClick?.(e);\n                }}\n                tabIndex={getTabIndex(i)}\n                data-tab-index={i}\n                type={as !== \"button\" ? undefined : \"button\"}\n                role=\"tab\"\n                aria-selected={activeTab === i}\n                aria-disabled={disabled || undefined}\n                aria-controls={children ? tabPanelId : undefined}\n              >\n                {iconConfig && iconConfig.position === \"left\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n                {label} {addSubtitle(sublabel)}\n                {iconConfig && iconConfig.position === \"right\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n              </StyledTab>\n            );\n\n            if (disabled && tooltipContent) {\n              return (\n                <Tooltip key={label} content={tooltipContent}>\n                  {Tab}\n                </Tooltip>\n              );\n            }\n\n            return Tab;\n          }\n        )}\n      </StyledContainer>\n      {children && activeTab != null && (\n        <div\n          role=\"tabpanel\"\n          id={tabPanelId}\n          aria-labelledby={getTabId(activeTab)}\n        >\n          {children}\n        </div>\n      )}\n    </Stack>\n  );\n}\n"],"names":[],"mappings":"AAqBwB"} */"),StyledTab=styled("button",{shouldForwardProp:prop=>isPropValid(prop)&&"active"!==prop&&"disabled"!==prop,target:"e2kj0g01",label:"StyledTab"})(({theme,active,disabled,hideBottomBorder,activeTabClickable})=>({...getButtonAppearanceReset(),padding:0,display:"flex",gap:theme.variables.size.spacing.xxs,alignItems:"center",boxSizing:"border-box",color:theme.values.color.text.secondary.default,fontFamily:theme.variables.fontFamily.lato,fontWeight:theme.variables.weight.bold,fontSize:theme.variables.size.font.s,lineHeight:theme.variables.size.lineHeight.xs,borderBottom:"4px solid transparent",paddingTop:hideBottomBorder?theme.variables.size.spacing.m:`calc(${theme.variables.size.spacing.m} - 1px)`,paddingBottom:`calc(${theme.variables.size.spacing.m} - 4px)`,...active&&!disabled&&{color:theme.values.color.text.accent.default,borderBottomColor:theme.values.color.border.accent.default},...!active&&!disabled&&{cursor:"pointer","&:hover":{color:theme.values.color.text.primary.default,borderBottomColor:theme.values.color.border.secondary.default},"&:active":{color:theme.values.color.text.primary.default,borderBottomColor:theme.values.color.border.secondary.active}},...activeTabClickable&&{cursor:"pointer"},...disabled&&{opacity:theme.variables.opacity.disabled,cursor:"not-allowed"},"&:focus-visible":{outlineWidth:2,outlineOffset:-1}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Tabs/Tabs.tsx","sources":["src/components/Tabs/Tabs.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport styled from \"@emotion/styled\";\nimport isPropValid from \"@emotion/is-prop-valid\";\nimport React, { useId, useRef } from \"react\";\nimport { useResponsiveStyles } from \"../../shared/mediaQueries\";\nimport { useKeyboard } from \"../../shared/useKeyboard\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Stack } from \"../Stack/Stack\";\nimport { Tooltip } from \"../Tooltip/Tooltip\";\nimport type { IconProp, Tab, TabsProps } from \"./-types\";\nimport { useTabFocus } from \"./useTabFocus\";\nimport { getButtonAppearanceReset } from \"../../shared/mixins/Button/getButtonAppearanceReset\";\n\nexport type { TabsProps };\n\nconst TAB_INDEX_ATTR = \"data-tab-index\";\n\nconst CONTAINER_BORDER_WIDTH = 1;\nconst TAB_BORDER_WIDTH = 4;\n\nconst StyledContainer = styled(\"div\", {\n  shouldForwardProp: (prop) => isPropValid(prop),\n})<Pick<TabsProps, \"horizontalPadding\" | \"hideBottomBorder\">>(\n  ({ theme, horizontalPadding, hideBottomBorder }) => ({\n    display: \"flex\",\n    flexDirection: \"row\",\n    ...useResponsiveStyles({\n      gap: [[\"m\", \"l\", \"l\"], theme.variables.size.spacing],\n      paddingInline: [horizontalPadding, theme.variables.size.spacing],\n    }),\n\n    ...(!hideBottomBorder && {\n      borderBottom: `${CONTAINER_BORDER_WIDTH}px solid ${theme.values.color.border.secondary.default}`,\n    }),\n\n    whiteSpace: \"nowrap\",\n\n    // Scrollbar\n    overflow: \"auto\",\n    scrollbarWidth: \"none\", // Firefox\n    msOverflowStyle: \"none\", // IE 10+\n    \"::-webkit-scrollbar\": {\n      display: \"none\", // Chrome\n    },\n  })\n);\n\nconst StyledTab = styled(\"button\", {\n  shouldForwardProp: (prop) =>\n    isPropValid(prop) && prop !== \"active\" && prop !== \"disabled\",\n})<\n  Pick<TabsProps, \"hideBottomBorder\" | \"activeTabClickable\"> &\n    Pick<Tab, \"disabled\" | \"active\">\n>(({ theme, active, disabled, hideBottomBorder, activeTabClickable }) => ({\n  ...getButtonAppearanceReset(),\n  padding: 0,\n  display: \"flex\",\n  gap: theme.variables.size.spacing.xxs,\n  alignItems: \"center\",\n\n  boxSizing: \"border-box\",\n  color: theme.values.color.text.secondary.default,\n  fontFamily: theme.variables.fontFamily.lato,\n  fontWeight: theme.variables.weight.bold,\n  fontSize: theme.variables.size.font.s,\n  lineHeight: theme.variables.size.lineHeight.xs,\n\n  borderBottom: `${TAB_BORDER_WIDTH}px solid transparent`,\n  // since we have a bottom border on the container,\n  // we need to add padding to the top of the tab to maintain consistent height\n  paddingTop: hideBottomBorder\n    ? theme.variables.size.spacing.m\n    : `calc(${theme.variables.size.spacing.m} - ${CONTAINER_BORDER_WIDTH}px)`,\n  paddingBottom: `calc(${theme.variables.size.spacing.m} - ${TAB_BORDER_WIDTH}px)`,\n\n  ...(active &&\n    !disabled && {\n      color: theme.values.color.text.accent.default,\n      borderBottomColor: theme.values.color.border.accent.default,\n    }),\n  ...(!active &&\n    !disabled && {\n      cursor: \"pointer\",\n      \"&:hover\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.default,\n      },\n      \"&:active\": {\n        color: theme.values.color.text.primary.default,\n        borderBottomColor: theme.values.color.border.secondary.active,\n      },\n    }),\n  ...(activeTabClickable && {\n    cursor: \"pointer\",\n  }),\n  ...(disabled && {\n    opacity: theme.variables.opacity.disabled,\n    cursor: \"not-allowed\",\n  }),\n\n  // TODO: The focus outline is clipped by the parent's `overflow: auto` in the scrollable variant.\n  // Adding padding/margin to prevent clipping would need a design review.\n  \"&:focus-visible\": {\n    outlineWidth: 2,\n    outlineOffset: -1,\n  },\n}));\n\nconst addSubtitle = (text: string) => (text ? `(${text})` : ``);\n\nexport function Tabs({\n  tabs,\n  horizontalPadding,\n  activeTab = 0,\n  children,\n  onTabSelect,\n  activeTabClickable,\n  tabPanelId: tabPanelIdProp,\n  \"data-e2e-test-id\": dataE2eTestId,\n  hideBottomBorder,\n  ...ariaAttributes\n}: TabsProps): React.ReactElement {\n  const tablistRef = useRef<HTMLDivElement>(null);\n  const instanceId = useId();\n  const tabPanelId = tabPanelIdProp || `${instanceId}-panel`;\n\n  const { onBlur, onTabFocus, getTabIndex, focusNextTab } = useTabFocus({\n    tabs,\n    activeTab,\n    tablistRef,\n    tabAttribute: TAB_INDEX_ATTR,\n  });\n\n  useKeyboard(\n    {\n      ArrowRight: () => focusNextTab(1),\n      ArrowLeft: () => focusNextTab(-1),\n    },\n    tablistRef,\n    true\n  );\n\n  const getTabId = (index: number) => `${instanceId}-tab-${index}`;\n\n  return (\n    <Stack data-e2e-test-id={dataE2eTestId} data-ds-id=\"Tabs\">\n      <StyledContainer\n        {...ariaAttributes}\n        hideBottomBorder={hideBottomBorder}\n        horizontalPadding={horizontalPadding}\n        onBlur={onBlur}\n        ref={tablistRef}\n        aria-orientation=\"horizontal\"\n        role=\"tablist\"\n      >\n        {tabs.map(\n          (\n            {\n              label,\n              sublabel,\n              icon,\n              as = \"button\",\n              disabled,\n              tooltipContent,\n              ...rest\n            },\n            i\n          ) => {\n            // Parse icon configuration\n            let iconConfig: IconProp;\n\n            if (icon) {\n              if (typeof icon === \"string\") {\n                iconConfig = { name: icon, position: \"left\" };\n              } else {\n                iconConfig = icon;\n              }\n            }\n\n            const Tab = (\n              <StyledTab\n                id={getTabId(i)}\n                activeTabClickable={activeTabClickable}\n                hideBottomBorder={hideBottomBorder}\n                disabled={disabled}\n                as={as}\n                key={label}\n                active={activeTab === i}\n                {...rest}\n                onFocus={(e) => {\n                  onTabFocus(i);\n                  rest.onFocus?.(e);\n                }}\n                onClick={(e) => {\n                  if (disabled) return;\n                  onTabSelect?.(i);\n                  rest.onClick?.(e);\n                }}\n                tabIndex={getTabIndex(i)}\n                data-tab-index={i}\n                type={as !== \"button\" ? undefined : \"button\"}\n                role=\"tab\"\n                aria-selected={activeTab === i}\n                aria-disabled={disabled || undefined}\n                aria-controls={children ? tabPanelId : undefined}\n              >\n                {iconConfig && iconConfig.position === \"left\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n                {label} {addSubtitle(sublabel)}\n                {iconConfig && iconConfig.position === \"right\" && (\n                  <Icon size=\"s\" name={iconConfig.name} />\n                )}\n              </StyledTab>\n            );\n\n            if (disabled && tooltipContent) {\n              return (\n                <Tooltip key={label} content={tooltipContent}>\n                  {Tab}\n                </Tooltip>\n              );\n            }\n\n            return Tab;\n          }\n        )}\n      </StyledContainer>\n      {children && activeTab != null && (\n        <div\n          role=\"tabpanel\"\n          id={tabPanelId}\n          aria-labelledby={getTabId(activeTab)}\n        >\n          {children}\n        </div>\n      )}\n    </Stack>\n  );\n}\n"],"names":[],"mappings":"AAgDkB"} */"),addSubtitle=text=>text?`(${text})`:"";export function Tabs({tabs,horizontalPadding,activeTab=0,children,onTabSelect,activeTabClickable,tabPanelId:tabPanelIdProp,"data-e2e-test-id":dataE2eTestId,hideBottomBorder,...ariaAttributes}){let tablistRef=useRef(null),instanceId=useId(),tabPanelId=tabPanelIdProp||`${instanceId}-panel`,{onBlur,onTabFocus,getTabIndex,focusNextTab}=useTabFocus({tabs,activeTab,tablistRef,tabAttribute:"data-tab-index"});useKeyboard({ArrowRight:()=>focusNextTab(1),ArrowLeft:()=>focusNextTab(-1)},tablistRef,!0);let getTabId=index=>`${instanceId}-tab-${index}`;return React.createElement(Stack,{"data-e2e-test-id":dataE2eTestId,"data-ds-id":"Tabs"},React.createElement(StyledContainer,{...ariaAttributes,hideBottomBorder:hideBottomBorder,horizontalPadding:horizontalPadding,onBlur:onBlur,ref:tablistRef,"aria-orientation":"horizontal",role:"tablist"},tabs.map(({label,sublabel,icon,as="button",disabled,tooltipContent,...rest},i)=>{let iconConfig;icon&&(iconConfig="string"==typeof icon?{name:icon,position:"left"}:icon);let Tab=React.createElement(StyledTab,{id:getTabId(i),activeTabClickable:activeTabClickable,hideBottomBorder:hideBottomBorder,disabled:disabled,as:as,key:label,active:activeTab===i,...rest,onFocus:e=>{onTabFocus(i),rest.onFocus?.(e)},onClick:e=>{disabled||(onTabSelect?.(i),rest.onClick?.(e))},tabIndex:getTabIndex(i),"data-tab-index":i,type:"button"!==as?void 0:"button",role:"tab","aria-selected":activeTab===i,"aria-disabled":disabled||void 0,"aria-controls":children?tabPanelId:void 0},iconConfig&&"left"===iconConfig.position&&React.createElement(Icon,{size:"s",name:iconConfig.name}),label," ",addSubtitle(sublabel),iconConfig&&"right"===iconConfig.position&&React.createElement(Icon,{size:"s",name:iconConfig.name}));return disabled&&tooltipContent?React.createElement(Tooltip,{key:label,content:tooltipContent},Tab):Tab})),children&&null!=activeTab&&React.createElement("div",{role:"tabpanel",id:tabPanelId,"aria-labelledby":getTabId(activeTab)},children))}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { RefObject } from "react";
|
|
2
|
+
import type { TabsProps } from "./-types";
|
|
3
|
+
type UseTabFocusOptions = Pick<TabsProps, "tabs" | "activeTab"> & {
|
|
4
|
+
tablistRef: RefObject<HTMLElement>;
|
|
5
|
+
tabAttribute: string;
|
|
6
|
+
};
|
|
7
|
+
export declare const useTabFocus: ({ tabs, activeTab, tablistRef, tabAttribute, }: UseTabFocusOptions) => {
|
|
8
|
+
onBlur: (e: React.FocusEvent) => void;
|
|
9
|
+
onTabFocus: (index: number) => void;
|
|
10
|
+
getTabIndex: (index: number) => 0 | -1;
|
|
11
|
+
focusNextTab: (direction: 1 | -1) => void;
|
|
12
|
+
};
|
|
13
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{useState}from"react";let findNextTab=(tabs,from,direction)=>{let tabsLen=tabs.length,start=from;return from<0&&(start=1===direction?-1:tabsLen),(start+direction+tabsLen)%tabsLen};export const useTabFocus=({tabs,activeTab,tablistRef,tabAttribute})=>{let[focusedIndex,setFocusedIndex]=useState(null),focusTab=index=>{tablistRef.current?.querySelector(`[${tabAttribute}="${index}"]`)?.focus()},currentIndex=focusedIndex??activeTab??-1,focusableIndex=focusedIndex??(null!=activeTab?activeTab:0);return{onBlur:e=>{tablistRef.current?.contains(e.relatedTarget)||setFocusedIndex(null)},onTabFocus:index=>setFocusedIndex(index),getTabIndex:index=>index===focusableIndex?0:-1,focusNextTab:direction=>{focusTab(findNextTab(tabs,currentIndex,direction))}}};
|