@amboss/design-system 3.9.6 → 3.9.7
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/Combobox/OptionsList.js +1 -1
- package/build/cjs/components/VirtualScrollList/VirtualScrollList.d.ts +2 -1
- package/build/cjs/components/VirtualScrollList/VirtualScrollList.js +1 -1
- package/build/esm/components/Form/Combobox/OptionsList.js +1 -1
- package/build/esm/components/VirtualScrollList/VirtualScrollList.d.ts +2 -1
- package/build/esm/components/VirtualScrollList/VirtualScrollList.js +1 -1
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={OptionsList:function(){return OptionsList},getOptionId:function(){return getOptionId},getOptionsListId:function(){return getOptionsListId}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:all[name]});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_useKeyboard=require("../../../shared/useKeyboard"),_usePopupMenu=require("../../../shared/popupMenu/usePopupMenu"),_VirtualScrollList=require("../../VirtualScrollList/VirtualScrollList"),_Text=require("../../Typography/Text/Text"),_Box=require("../../Box/Box"),_Inline=require("../../Inline/Inline"),_Icon=require("../../Icon/Icon"),_StyledSelectComponents=require("./StyledSelectComponents"),_constants=require("./constants"),_Stack=require("../../Stack/Stack"),_Checkbox=require("../Checkbox/Checkbox"),_Portal=require("../../Portal/Portal"),_ScreenReaderText=require("../../Utilities/ScreenReaderText/ScreenReaderText");function getOptionId(selectName,optionValue){return`${selectName}Option${optionValue}`}function getOptionsListId(selectName){return`${selectName}OptionsList`}function scrollToItem(list,itemIndex){list[itemIndex]?.scrollIntoView?.({behavior:"smooth",block:"nearest"})}const CustomOption=(0,_styled.default)(_StyledSelectComponents.StyledOption,{target:"e1d5yhis0",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA6CqB"} */"),StyledListContainer=(0,_styled.default)("div",{target:"e1d5yhis1",label:"StyledListContainer"})(({theme,maxHeight})=>({maxHeight,boxSizing:"border-box",overflow:"auto",padding:`${theme.variables.size.spacing.xxs} 0`}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA0D4B"} */");function OptionsList({isOpen,isMultiSelect=!1,value,disabled,options,portalContainer,triggerRef,triggerWrapperRef,maxHeight,emptyStateMessage="No options available",selectedIndex,optionsListWidth,isVirtualized=!0,name,onCloseDropdown,onSelectedIndexChange,forceChangeFakeSelect}){let optionsListRef=(0,_react.useRef)(null),listContainerRef=(0,_react.useRef)(null),listItemsRef=(0,_react.useRef)(null),refForPositioning=triggerWrapperRef||triggerRef,activeItemIndex=(0,_react.useMemo)(()=>options.findIndex(option=>isMultiSelect?value.includes(option.value):option.value===value),[options,isMultiSelect,value]),{menuStyle:optionsListStyle,calculateStyle}=(0,_usePopupMenu.usePopupMenu)({triggerRef:refForPositioning,menuRef:optionsListRef,isOpen});(0,_react.useEffect)(()=>{if(isOpen&&listContainerRef.current&&!isVirtualized&&optionsListStyle.top){let listElm=listContainerRef.current;listElm.clientHeight<listElm.scrollHeight&&(listItemsRef.current=listElm.querySelectorAll("[data-list-item]"),scrollToItem(listItemsRef.current,activeItemIndex))}else listItemsRef.current=null},[isOpen,isVirtualized,listContainerRef,options,maxHeight,optionsListStyle,activeItemIndex]),(0,_react.useEffect)(()=>{listItemsRef.current&&scrollToItem(listItemsRef.current,selectedIndex)},[listItemsRef,selectedIndex]),(0,_useKeyboard.useKeyboard)({Escape:()=>onCloseDropdown(!0),Enter:()=>{let preselectedOption=options[selectedIndex];if(!preselectedOption?.disabled){if(preselectedOption){forceChangeFakeSelect(preselectedOption),isMultiSelect||onCloseDropdown(!0);return}isMultiSelect||onCloseDropdown()}},ArrowUp:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex>0&&(newIndex-=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)},ArrowDown:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex<options.length-1&&(newIndex+=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)}},triggerRef,isOpen&&!disabled);let renderItem=index=>{let option=options[index],id=getOptionId(name,option.value);if(isMultiSelect){let isActive=value.includes(option.value);return _react.default.createElement(CustomOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||forceChangeFakeSelect(option)},onClick:evt=>evt.stopPropagation()},_react.default.createElement(_Checkbox.CheckboxRaw,{name:"",disabled:option.disabled,checked:isActive,size:"s",label:option.label,onChange:()=>null,labelHint:option?.description}))}let isActive=value===option.value,optionText=_react.default.createElement(_Stack.Stack,{space:"zero"},_react.default.createElement(_Text.Text,{color:"primary",size:"m"},option.label),option.description&&_react.default.createElement(_Text.Text,{color:"tertiary",size:"s"},option.description));return _react.default.createElement(_StyledSelectComponents.StyledOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||(forceChangeFakeSelect(option),onCloseDropdown(!0))}},isActive?_react.default.createElement(_Inline.Inline,{noWrap:!0,alignItems:"spaceBetween"},optionText,_react.default.createElement(_Box.Box,{space:"zero",tSpace:"xxxs"},_react.default.createElement(_Icon.Icon,{name:"check",size:"s",color:"accent"}))):optionText)},listStyle={...optionsListStyle,width:optionsListWidth||refForPositioning.current?.getBoundingClientRect().width},listElm=_react.default.createElement(_StyledSelectComponents.StyledDropdown,{style:listStyle,ref:optionsListRef,onMouseDown:e=>e.preventDefault(),"data-e2e-test-id":"optionsList"},isVirtualized?_react.default.createElement(_VirtualScrollList.VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:options.length,emptyState:()=>null,containerStyle:_constants.dropdownContainerStyle,itemInView:selectedIndex,itemTemplate:renderItem,onContentHeightChange:calculateStyle}):_react.default.createElement(StyledListContainer,{maxHeight:maxHeight,role:"listbox",id:getOptionsListId(name),ref:listContainerRef},options.map((_option,index)=>renderItem(index))));if(!isOpen)return null;let listPortal=options.length?_react.default.createElement(_Portal.Portal,{portalContainer:portalContainer},listElm):null;return _react.default.createElement(_react.default.Fragment,null,_react.default.createElement(_ScreenReaderText.ScreenReaderText,{role:"status","aria-live":"polite"},options.length?"":emptyStateMessage),listPortal)}
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0});var target=exports,all={OptionsList:function(){return OptionsList},getOptionId:function(){return getOptionId},getOptionsListId:function(){return getOptionsListId}};for(var name in all)Object.defineProperty(target,name,{enumerable:!0,get:all[name]});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_react=/*#__PURE__*/require("@swc/helpers/_/_interop_require_wildcard")._(require("react")),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_useKeyboard=require("../../../shared/useKeyboard"),_usePopupMenu=require("../../../shared/popupMenu/usePopupMenu"),_VirtualScrollList=require("../../VirtualScrollList/VirtualScrollList"),_Text=require("../../Typography/Text/Text"),_Box=require("../../Box/Box"),_Inline=require("../../Inline/Inline"),_Icon=require("../../Icon/Icon"),_StyledSelectComponents=require("./StyledSelectComponents"),_constants=require("./constants"),_Stack=require("../../Stack/Stack"),_Checkbox=require("../Checkbox/Checkbox"),_Portal=require("../../Portal/Portal"),_ScreenReaderText=require("../../Utilities/ScreenReaderText/ScreenReaderText");function getOptionId(selectName,optionValue){return`${selectName}Option${optionValue}`}function getOptionsListId(selectName){return`${selectName}OptionsList`}function scrollToItem(list,itemIndex){list[itemIndex]?.scrollIntoView?.({behavior:"smooth",block:"nearest"})}const CustomOption=(0,_styled.default)(_StyledSelectComponents.StyledOption,{target:"e159rw3y0",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          id={getOptionsListId(name)}\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA6CqB"} */"),StyledListContainer=(0,_styled.default)("div",{target:"e159rw3y1",label:"StyledListContainer"})(({theme,maxHeight})=>({maxHeight,boxSizing:"border-box",overflow:"auto",padding:`${theme.variables.size.spacing.xxs} 0`}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          id={getOptionsListId(name)}\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA0D4B"} */");function OptionsList({isOpen,isMultiSelect=!1,value,disabled,options,portalContainer,triggerRef,triggerWrapperRef,maxHeight,emptyStateMessage="No options available",selectedIndex,optionsListWidth,isVirtualized=!0,name,onCloseDropdown,onSelectedIndexChange,forceChangeFakeSelect}){let optionsListRef=(0,_react.useRef)(null),listContainerRef=(0,_react.useRef)(null),listItemsRef=(0,_react.useRef)(null),refForPositioning=triggerWrapperRef||triggerRef,activeItemIndex=(0,_react.useMemo)(()=>options.findIndex(option=>isMultiSelect?value.includes(option.value):option.value===value),[options,isMultiSelect,value]),{menuStyle:optionsListStyle,calculateStyle}=(0,_usePopupMenu.usePopupMenu)({triggerRef:refForPositioning,menuRef:optionsListRef,isOpen});(0,_react.useEffect)(()=>{if(isOpen&&listContainerRef.current&&!isVirtualized&&optionsListStyle.top){let listElm=listContainerRef.current;listElm.clientHeight<listElm.scrollHeight&&(listItemsRef.current=listElm.querySelectorAll("[data-list-item]"),scrollToItem(listItemsRef.current,activeItemIndex))}else listItemsRef.current=null},[isOpen,isVirtualized,listContainerRef,options,maxHeight,optionsListStyle,activeItemIndex]),(0,_react.useEffect)(()=>{listItemsRef.current&&scrollToItem(listItemsRef.current,selectedIndex)},[listItemsRef,selectedIndex]),(0,_useKeyboard.useKeyboard)({Escape:()=>onCloseDropdown(!0),Enter:()=>{let preselectedOption=options[selectedIndex];if(!preselectedOption?.disabled){if(preselectedOption){forceChangeFakeSelect(preselectedOption),isMultiSelect||onCloseDropdown(!0);return}isMultiSelect||onCloseDropdown()}},ArrowUp:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex>0&&(newIndex-=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)},ArrowDown:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex<options.length-1&&(newIndex+=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)}},triggerRef,isOpen&&!disabled);let renderItem=index=>{let option=options[index],id=getOptionId(name,option.value);if(isMultiSelect){let isActive=value.includes(option.value);return _react.default.createElement(CustomOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||forceChangeFakeSelect(option)},onClick:evt=>evt.stopPropagation()},_react.default.createElement(_Checkbox.CheckboxRaw,{name:"",disabled:option.disabled,checked:isActive,size:"s",label:option.label,onChange:()=>null,labelHint:option?.description}))}let isActive=value===option.value,optionText=_react.default.createElement(_Stack.Stack,{space:"zero"},_react.default.createElement(_Text.Text,{color:"primary",size:"m"},option.label),option.description&&_react.default.createElement(_Text.Text,{color:"tertiary",size:"s"},option.description));return _react.default.createElement(_StyledSelectComponents.StyledOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||(forceChangeFakeSelect(option),onCloseDropdown(!0))}},isActive?_react.default.createElement(_Inline.Inline,{noWrap:!0,alignItems:"spaceBetween"},optionText,_react.default.createElement(_Box.Box,{space:"zero",tSpace:"xxxs"},_react.default.createElement(_Icon.Icon,{name:"check",size:"s",color:"accent"}))):optionText)},listStyle={...optionsListStyle,width:optionsListWidth||refForPositioning.current?.getBoundingClientRect().width},listElm=_react.default.createElement(_StyledSelectComponents.StyledDropdown,{style:listStyle,ref:optionsListRef,onMouseDown:e=>e.preventDefault(),"data-e2e-test-id":"optionsList"},isVirtualized?_react.default.createElement(_VirtualScrollList.VirtualScrollList,{id:getOptionsListId(name),maxHeight:maxHeight,itemHeight:36,itemAmount:options.length,emptyState:()=>null,containerStyle:_constants.dropdownContainerStyle,itemInView:selectedIndex,itemTemplate:renderItem,onContentHeightChange:calculateStyle}):_react.default.createElement(StyledListContainer,{maxHeight:maxHeight,role:"listbox",id:getOptionsListId(name),ref:listContainerRef},options.map((_option,index)=>renderItem(index))));if(!isOpen)return null;let listPortal=options.length?_react.default.createElement(_Portal.Portal,{portalContainer:portalContainer},listElm):null;return _react.default.createElement(_react.default.Fragment,null,_react.default.createElement(_ScreenReaderText.ScreenReaderText,{role:"status","aria-live":"polite"},options.length?"":emptyStateMessage),listPortal)}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import React from "react";
|
|
2
2
|
export type VirtualScrollListProps = {
|
|
3
3
|
"data-e2e-test-id"?: string;
|
|
4
|
+
id: string;
|
|
4
5
|
maxHeight: number;
|
|
5
6
|
itemHeight: number;
|
|
6
7
|
itemAmount: number;
|
|
@@ -10,4 +11,4 @@ export type VirtualScrollListProps = {
|
|
|
10
11
|
containerStyle?: React.CSSProperties;
|
|
11
12
|
onContentHeightChange?: () => void;
|
|
12
13
|
};
|
|
13
|
-
export declare function VirtualScrollList({ maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, containerStyle, "data-e2e-test-id": dataE2eTestId, onContentHeightChange, }: VirtualScrollListProps): React.ReactElement;
|
|
14
|
+
export declare function VirtualScrollList({ id, maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, containerStyle, "data-e2e-test-id": dataE2eTestId, onContentHeightChange, }: VirtualScrollListProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),Object.defineProperty(exports,"VirtualScrollList",{enumerable:!0,get:function(){return VirtualScrollList}});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_interop_require_wildcard=require("@swc/helpers/_/_interop_require_wildcard"),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_react=/*#__PURE__*/_interop_require_wildcard._(require("react")),_VirtualScrollListReducer=require("./VirtualScrollListReducer"),StyledContainer=(0,_styled.default)("div",{target:"e19y1j3j0",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbiAgb25Db250ZW50SGVpZ2h0Q2hhbmdlLFxufTogVmlydHVhbFNjcm9sbExpc3RQcm9wcyk6IFJlYWN0LlJlYWN0RWxlbWVudCB7XG4gIGNvbnN0IHNjcm9sbGFibGVDb250YWluZXJSZWYgPSB1c2VSZWYobnVsbCk7XG4gIGNvbnN0IHZpZXdwb3J0UmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCBhZGp1c3RlZEZvclJlZmxvdyA9IHVzZVJlZihmYWxzZSk7XG5cbiAgY29uc3QgW1xuICAgIHtcbiAgICAgIHNjcm9sbGVkSXRlbUNvdW50LFxuICAgICAgYW1vdW50T2ZJdGVtc0luVmlldyxcbiAgICAgIHNjcm9sbGVkSW5QeCxcbiAgICAgIG1heENvbnRlbnRIZWlnaHQgPSBtYXhIZWlnaHQsXG4gICAgICByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uLFxuICAgIH0sXG4gICAgZGlzcGF0Y2gsXG4gIF0gPSB1c2VSZWR1Y2VyKFZpcnR1YWxTY3JvbGxSZWR1Y2VyLCB7fSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7IHR5cGU6IFwicmVzZXRcIiwgaXRlbUhlaWdodCwgbWF4SGVpZ2h0LCBvdmVyc2NhbiB9KTtcbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gMDtcbiAgfSwgW2l0ZW1IZWlnaHQsIGl0ZW1BbW91bnQsIG1heEhlaWdodF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goe1xuICAgICAgdHlwZTogXCJyZWNvbW1lbmRTY3JvbGxQb3NpdGlvblwiLFxuICAgICAgc2Nyb2xsVG9wOiBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wLFxuICAgICAgaXRlbVRvQmVJblZpZXc6IGl0ZW1JblZpZXcsXG4gICAgfSk7XG4gIH0sIFtpdGVtSW5WaWV3XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbiA9PT0gbnVsbCkgcmV0dXJuO1xuXG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb247XG4gIH0sIFtyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uXSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcblxuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgIHZpZXdwb3J0Tm9kZTogdmlld3BvcnRSZWYuY3VycmVudCxcbiAgICAgIGl0ZW1BbW91bnQsXG4gICAgfSk7XG4gIH0sIFtpdGVtQW1vdW50LCBzY3JvbGxlZEl0ZW1Db3VudF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgb25Db250ZW50SGVpZ2h0Q2hhbmdlPy4oKTtcbiAgfSwgW21heENvbnRlbnRIZWlnaHQsIG9uQ29udGVudEhlaWdodENoYW5nZV0pO1xuXG4gIGNvbnN0IGl0ZW1Db3VudFRvQmVSZW5kZXJlZCA9IE1hdGgubWluKFxuICAgIE1hdGgubWF4KDAsIGl0ZW1BbW91bnQgLSBzY3JvbGxlZEl0ZW1Db3VudCksXG4gICAgYW1vdW50T2ZJdGVtc0luVmlld1xuICApO1xuXG4gIC8qIFNvbWV0aW1lcywgb3B0aW9uIHRleHQgY2FuIHJlZmxvdyBkdWUgdGhlIGFwcGVhcmFuY2Ugb2YgdmVydGljYWwgc2Nyb2xsYmFyIHdoaWNoIGNhbiBjYXVzZSB0ZXh0IHRvIHdyYXAgdG8gbXVsdGlwbGUgbGluZXMgY2F1c2luZyB2aWV3cG9ydFJlZiBoZWlnaHQgdG8gY2hhbmdlLiBJZiBjb250YWluZXIgaGFzIHNjcm9sbCB0aGVuIHdlIGRpc3BhdGNoIGB1cGRhdGVWaWV3cG9ydGAgb25lIG1vcmUgdGltZSB0byBnZXQgdGhlIGNvcnJlY3QgaGVpZ2h0LCB3aXRoIGEgc2V0VGltZW91dCB0byBhbGxvdyB0aW1lIGZvciByZWZsb3cuXG4gICAqL1xuICBpZiAoXG4gICAgbWF4Q29udGVudEhlaWdodCA+IG1heEhlaWdodCAmJlxuICAgICFhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ICYmXG4gICAgdmlld3BvcnRSZWYuY3VycmVudFxuICApIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGRpc3BhdGNoKHtcbiAgICAgICAgdHlwZTogXCJ1cGRhdGVWaWV3cG9ydFwiLFxuICAgICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICAgIGl0ZW1BbW91bnQsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgICBhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ID0gdHJ1ZTtcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZENvbnRhaW5lclxuICAgICAgcmVmPXtzY3JvbGxhYmxlQ29udGFpbmVyUmVmfVxuICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHR9XG4gICAgICBzdHlsZT17e1xuICAgICAgICBtYXhIZWlnaHQsXG4gICAgICAgIC4uLmNvbnRhaW5lclN0eWxlLFxuICAgICAgfX1cbiAgICAgIGRhdGEtZTJlLXRlc3QtaWQ9e2RhdGFFMmVUZXN0SWR9XG4gICAgICBkYXRhLWRzLWlkPVwiVmlydHVhbFNjcm9sbExpc3RcIlxuICAgICAgb25TY3JvbGw9eyhlOiBSZWFjdC5VSUV2ZW50PEhUTUxFbGVtZW50PikgPT4ge1xuICAgICAgICBkaXNwYXRjaCh7XG4gICAgICAgICAgdHlwZTogXCJzY3JvbGxcIixcbiAgICAgICAgICBzY3JvbGxUb3A6IGUuY3VycmVudFRhcmdldC5zY3JvbGxUb3AsXG4gICAgICAgIH0pO1xuICAgICAgfX1cbiAgICA+XG4gICAgICB7aXRlbUFtb3VudCA9PT0gMCA/IChcbiAgICAgICAgZW1wdHlTdGF0ZSgpXG4gICAgICApIDogKFxuICAgICAgICA8U3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgbWF4Q29udGVudEhlaWdodD17bWF4Q29udGVudEhlaWdodH0+XG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgcmVmPXt2aWV3cG9ydFJlZn1cbiAgICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICAgIHRyYW5zZm9ybTogYHRyYW5zbGF0ZVkoJHtzY3JvbGxlZEluUHh9cHhgLFxuICAgICAgICAgICAgfX1cbiAgICAgICAgICAgIHJvbGU9XCJsaXN0Ym94XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7ISFpdGVtQ291bnRUb0JlUmVuZGVyZWQgJiZcbiAgICAgICAgICAgICAgbmV3IEFycmF5KGl0ZW1Db3VudFRvQmVSZW5kZXJlZClcbiAgICAgICAgICAgICAgICAuZmlsbCgwKVxuICAgICAgICAgICAgICAgIC5tYXAoKF8sIGluZGV4KSA9PiBpdGVtVGVtcGxhdGUoc2Nyb2xsZWRJdGVtQ291bnQgKyBpbmRleCkpfVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICA8L1N0eWxlZFNjcm9sbGFibGVDb250ZW50PlxuICAgICAgKX1cbiAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgKTtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLd0IifQ== */"),StyledScrollableContent=(0,_styled.default)("div",{target:"e19y1j3j1",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbiAgb25Db250ZW50SGVpZ2h0Q2hhbmdlLFxufTogVmlydHVhbFNjcm9sbExpc3RQcm9wcyk6IFJlYWN0LlJlYWN0RWxlbWVudCB7XG4gIGNvbnN0IHNjcm9sbGFibGVDb250YWluZXJSZWYgPSB1c2VSZWYobnVsbCk7XG4gIGNvbnN0IHZpZXdwb3J0UmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCBhZGp1c3RlZEZvclJlZmxvdyA9IHVzZVJlZihmYWxzZSk7XG5cbiAgY29uc3QgW1xuICAgIHtcbiAgICAgIHNjcm9sbGVkSXRlbUNvdW50LFxuICAgICAgYW1vdW50T2ZJdGVtc0luVmlldyxcbiAgICAgIHNjcm9sbGVkSW5QeCxcbiAgICAgIG1heENvbnRlbnRIZWlnaHQgPSBtYXhIZWlnaHQsXG4gICAgICByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uLFxuICAgIH0sXG4gICAgZGlzcGF0Y2gsXG4gIF0gPSB1c2VSZWR1Y2VyKFZpcnR1YWxTY3JvbGxSZWR1Y2VyLCB7fSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7IHR5cGU6IFwicmVzZXRcIiwgaXRlbUhlaWdodCwgbWF4SGVpZ2h0LCBvdmVyc2NhbiB9KTtcbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gMDtcbiAgfSwgW2l0ZW1IZWlnaHQsIGl0ZW1BbW91bnQsIG1heEhlaWdodF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goe1xuICAgICAgdHlwZTogXCJyZWNvbW1lbmRTY3JvbGxQb3NpdGlvblwiLFxuICAgICAgc2Nyb2xsVG9wOiBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wLFxuICAgICAgaXRlbVRvQmVJblZpZXc6IGl0ZW1JblZpZXcsXG4gICAgfSk7XG4gIH0sIFtpdGVtSW5WaWV3XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbiA9PT0gbnVsbCkgcmV0dXJuO1xuXG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb247XG4gIH0sIFtyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uXSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcblxuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgIHZpZXdwb3J0Tm9kZTogdmlld3BvcnRSZWYuY3VycmVudCxcbiAgICAgIGl0ZW1BbW91bnQsXG4gICAgfSk7XG4gIH0sIFtpdGVtQW1vdW50LCBzY3JvbGxlZEl0ZW1Db3VudF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgb25Db250ZW50SGVpZ2h0Q2hhbmdlPy4oKTtcbiAgfSwgW21heENvbnRlbnRIZWlnaHQsIG9uQ29udGVudEhlaWdodENoYW5nZV0pO1xuXG4gIGNvbnN0IGl0ZW1Db3VudFRvQmVSZW5kZXJlZCA9IE1hdGgubWluKFxuICAgIE1hdGgubWF4KDAsIGl0ZW1BbW91bnQgLSBzY3JvbGxlZEl0ZW1Db3VudCksXG4gICAgYW1vdW50T2ZJdGVtc0luVmlld1xuICApO1xuXG4gIC8qIFNvbWV0aW1lcywgb3B0aW9uIHRleHQgY2FuIHJlZmxvdyBkdWUgdGhlIGFwcGVhcmFuY2Ugb2YgdmVydGljYWwgc2Nyb2xsYmFyIHdoaWNoIGNhbiBjYXVzZSB0ZXh0IHRvIHdyYXAgdG8gbXVsdGlwbGUgbGluZXMgY2F1c2luZyB2aWV3cG9ydFJlZiBoZWlnaHQgdG8gY2hhbmdlLiBJZiBjb250YWluZXIgaGFzIHNjcm9sbCB0aGVuIHdlIGRpc3BhdGNoIGB1cGRhdGVWaWV3cG9ydGAgb25lIG1vcmUgdGltZSB0byBnZXQgdGhlIGNvcnJlY3QgaGVpZ2h0LCB3aXRoIGEgc2V0VGltZW91dCB0byBhbGxvdyB0aW1lIGZvciByZWZsb3cuXG4gICAqL1xuICBpZiAoXG4gICAgbWF4Q29udGVudEhlaWdodCA+IG1heEhlaWdodCAmJlxuICAgICFhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ICYmXG4gICAgdmlld3BvcnRSZWYuY3VycmVudFxuICApIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGRpc3BhdGNoKHtcbiAgICAgICAgdHlwZTogXCJ1cGRhdGVWaWV3cG9ydFwiLFxuICAgICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICAgIGl0ZW1BbW91bnQsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgICBhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ID0gdHJ1ZTtcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZENvbnRhaW5lclxuICAgICAgcmVmPXtzY3JvbGxhYmxlQ29udGFpbmVyUmVmfVxuICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHR9XG4gICAgICBzdHlsZT17e1xuICAgICAgICBtYXhIZWlnaHQsXG4gICAgICAgIC4uLmNvbnRhaW5lclN0eWxlLFxuICAgICAgfX1cbiAgICAgIGRhdGEtZTJlLXRlc3QtaWQ9e2RhdGFFMmVUZXN0SWR9XG4gICAgICBkYXRhLWRzLWlkPVwiVmlydHVhbFNjcm9sbExpc3RcIlxuICAgICAgb25TY3JvbGw9eyhlOiBSZWFjdC5VSUV2ZW50PEhUTUxFbGVtZW50PikgPT4ge1xuICAgICAgICBkaXNwYXRjaCh7XG4gICAgICAgICAgdHlwZTogXCJzY3JvbGxcIixcbiAgICAgICAgICBzY3JvbGxUb3A6IGUuY3VycmVudFRhcmdldC5zY3JvbGxUb3AsXG4gICAgICAgIH0pO1xuICAgICAgfX1cbiAgICA+XG4gICAgICB7aXRlbUFtb3VudCA9PT0gMCA/IChcbiAgICAgICAgZW1wdHlTdGF0ZSgpXG4gICAgICApIDogKFxuICAgICAgICA8U3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgbWF4Q29udGVudEhlaWdodD17bWF4Q29udGVudEhlaWdodH0+XG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgcmVmPXt2aWV3cG9ydFJlZn1cbiAgICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICAgIHRyYW5zZm9ybTogYHRyYW5zbGF0ZVkoJHtzY3JvbGxlZEluUHh9cHhgLFxuICAgICAgICAgICAgfX1cbiAgICAgICAgICAgIHJvbGU9XCJsaXN0Ym94XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7ISFpdGVtQ291bnRUb0JlUmVuZGVyZWQgJiZcbiAgICAgICAgICAgICAgbmV3IEFycmF5KGl0ZW1Db3VudFRvQmVSZW5kZXJlZClcbiAgICAgICAgICAgICAgICAuZmlsbCgwKVxuICAgICAgICAgICAgICAgIC5tYXAoKF8sIGluZGV4KSA9PiBpdGVtVGVtcGxhdGUoc2Nyb2xsZWRJdGVtQ291bnQgKyBpbmRleCkpfVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICA8L1N0eWxlZFNjcm9sbGFibGVDb250ZW50PlxuICAgICAgKX1cbiAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgKTtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFjZ0MifQ== */");function VirtualScrollList({maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,containerStyle,"data-e2e-test-id":dataE2eTestId,onContentHeightChange}){let scrollableContainerRef=(0,_react.useRef)(null),viewportRef=(0,_react.useRef)(null),adjustedForReflow=(0,_react.useRef)(!1),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=(0,_react.useReducer)(_VirtualScrollListReducer.VirtualScrollReducer,{});(0,_react.useLayoutEffect)(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),(0,_react.useEffect)(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),(0,_react.useEffect)(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),(0,_react.useLayoutEffect)(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]),(0,_react.useEffect)(()=>{onContentHeightChange?.()},[maxContentHeight,onContentHeightChange]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return maxContentHeight>maxHeight&&!adjustedForReflow.current&&viewportRef.current&&(setTimeout(()=>{dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})}),adjustedForReflow.current=!0),_react.default.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight,...containerStyle},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():_react.default.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},_react.default.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`},role:"listbox"},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|
|
1
|
+
"use strict";Object.defineProperty(exports,"__esModule",{value:!0}),Object.defineProperty(exports,"VirtualScrollList",{enumerable:!0,get:function(){return VirtualScrollList}});const _interop_require_default=require("@swc/helpers/_/_interop_require_default"),_interop_require_wildcard=require("@swc/helpers/_/_interop_require_wildcard"),_styled=/*#__PURE__*/_interop_require_default._(require("@emotion/styled")),_react=/*#__PURE__*/_interop_require_wildcard._(require("react")),_VirtualScrollListReducer=require("./VirtualScrollListReducer"),StyledContainer=(0,_styled.default)("div",{target:"e1hq8rdx0",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgaWQ6IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBpZCxcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBjb250YWluZXJTdHlsZSxcbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI6IGRhdGFFMmVUZXN0SWQsXG4gIG9uQ29udGVudEhlaWdodENoYW5nZSxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcbiAgY29uc3QgYWRqdXN0ZWRGb3JSZWZsb3cgPSB1c2VSZWYoZmFsc2UpO1xuXG4gIGNvbnN0IFtcbiAgICB7XG4gICAgICBzY3JvbGxlZEl0ZW1Db3VudCxcbiAgICAgIGFtb3VudE9mSXRlbXNJblZpZXcsXG4gICAgICBzY3JvbGxlZEluUHgsXG4gICAgICBtYXhDb250ZW50SGVpZ2h0ID0gbWF4SGVpZ2h0LFxuICAgICAgcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbixcbiAgICB9LFxuICAgIGRpc3BhdGNoLFxuICBdID0gdXNlUmVkdWNlcihWaXJ0dWFsU2Nyb2xsUmVkdWNlciwge30pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goeyB0eXBlOiBcInJlc2V0XCIsIGl0ZW1IZWlnaHQsIG1heEhlaWdodCwgb3ZlcnNjYW4gfSk7XG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IDA7XG4gIH0sIFtpdGVtSGVpZ2h0LCBpdGVtQW1vdW50LCBtYXhIZWlnaHRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwicmVjb21tZW5kU2Nyb2xsUG9zaXRpb25cIixcbiAgICAgIHNjcm9sbFRvcDogc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCxcbiAgICAgIGl0ZW1Ub0JlSW5WaWV3OiBpdGVtSW5WaWV3LFxuICAgIH0pO1xuICB9LCBbaXRlbUluVmlld10pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24gPT09IG51bGwpIHJldHVybjtcblxuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uO1xuICB9LCBbcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbl0pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCF2aWV3cG9ydFJlZi5jdXJyZW50KSByZXR1cm47XG5cbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIG9uQ29udGVudEhlaWdodENoYW5nZT8uKCk7XG4gIH0sIFttYXhDb250ZW50SGVpZ2h0LCBvbkNvbnRlbnRIZWlnaHRDaGFuZ2VdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICAvKiBTb21ldGltZXMsIG9wdGlvbiB0ZXh0IGNhbiByZWZsb3cgZHVlIHRoZSBhcHBlYXJhbmNlIG9mIHZlcnRpY2FsIHNjcm9sbGJhciB3aGljaCBjYW4gY2F1c2UgdGV4dCB0byB3cmFwIHRvIG11bHRpcGxlIGxpbmVzIGNhdXNpbmcgdmlld3BvcnRSZWYgaGVpZ2h0IHRvIGNoYW5nZS4gSWYgY29udGFpbmVyIGhhcyBzY3JvbGwgdGhlbiB3ZSBkaXNwYXRjaCBgdXBkYXRlVmlld3BvcnRgIG9uZSBtb3JlIHRpbWUgdG8gZ2V0IHRoZSBjb3JyZWN0IGhlaWdodCwgd2l0aCBhIHNldFRpbWVvdXQgdG8gYWxsb3cgdGltZSBmb3IgcmVmbG93LlxuICAgKi9cbiAgaWYgKFxuICAgIG1heENvbnRlbnRIZWlnaHQgPiBtYXhIZWlnaHQgJiZcbiAgICAhYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCAmJlxuICAgIHZpZXdwb3J0UmVmLmN1cnJlbnRcbiAgKSB7XG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICBkaXNwYXRjaCh7XG4gICAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgICAgdmlld3BvcnROb2RlOiB2aWV3cG9ydFJlZi5jdXJyZW50LFxuICAgICAgICBpdGVtQW1vdW50LFxuICAgICAgfSk7XG4gICAgfSk7XG4gICAgYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCA9IHRydWU7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgICByb2xlPVwibGlzdGJveFwiXG4gICAgICAgICAgICBpZD17aWR9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS3dCIn0= */"),StyledScrollableContent=(0,_styled.default)("div",{target:"e1hq8rdx1",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgaWQ6IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBpZCxcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBjb250YWluZXJTdHlsZSxcbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI6IGRhdGFFMmVUZXN0SWQsXG4gIG9uQ29udGVudEhlaWdodENoYW5nZSxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcbiAgY29uc3QgYWRqdXN0ZWRGb3JSZWZsb3cgPSB1c2VSZWYoZmFsc2UpO1xuXG4gIGNvbnN0IFtcbiAgICB7XG4gICAgICBzY3JvbGxlZEl0ZW1Db3VudCxcbiAgICAgIGFtb3VudE9mSXRlbXNJblZpZXcsXG4gICAgICBzY3JvbGxlZEluUHgsXG4gICAgICBtYXhDb250ZW50SGVpZ2h0ID0gbWF4SGVpZ2h0LFxuICAgICAgcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbixcbiAgICB9LFxuICAgIGRpc3BhdGNoLFxuICBdID0gdXNlUmVkdWNlcihWaXJ0dWFsU2Nyb2xsUmVkdWNlciwge30pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goeyB0eXBlOiBcInJlc2V0XCIsIGl0ZW1IZWlnaHQsIG1heEhlaWdodCwgb3ZlcnNjYW4gfSk7XG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IDA7XG4gIH0sIFtpdGVtSGVpZ2h0LCBpdGVtQW1vdW50LCBtYXhIZWlnaHRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwicmVjb21tZW5kU2Nyb2xsUG9zaXRpb25cIixcbiAgICAgIHNjcm9sbFRvcDogc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCxcbiAgICAgIGl0ZW1Ub0JlSW5WaWV3OiBpdGVtSW5WaWV3LFxuICAgIH0pO1xuICB9LCBbaXRlbUluVmlld10pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24gPT09IG51bGwpIHJldHVybjtcblxuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uO1xuICB9LCBbcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbl0pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCF2aWV3cG9ydFJlZi5jdXJyZW50KSByZXR1cm47XG5cbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIG9uQ29udGVudEhlaWdodENoYW5nZT8uKCk7XG4gIH0sIFttYXhDb250ZW50SGVpZ2h0LCBvbkNvbnRlbnRIZWlnaHRDaGFuZ2VdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICAvKiBTb21ldGltZXMsIG9wdGlvbiB0ZXh0IGNhbiByZWZsb3cgZHVlIHRoZSBhcHBlYXJhbmNlIG9mIHZlcnRpY2FsIHNjcm9sbGJhciB3aGljaCBjYW4gY2F1c2UgdGV4dCB0byB3cmFwIHRvIG11bHRpcGxlIGxpbmVzIGNhdXNpbmcgdmlld3BvcnRSZWYgaGVpZ2h0IHRvIGNoYW5nZS4gSWYgY29udGFpbmVyIGhhcyBzY3JvbGwgdGhlbiB3ZSBkaXNwYXRjaCBgdXBkYXRlVmlld3BvcnRgIG9uZSBtb3JlIHRpbWUgdG8gZ2V0IHRoZSBjb3JyZWN0IGhlaWdodCwgd2l0aCBhIHNldFRpbWVvdXQgdG8gYWxsb3cgdGltZSBmb3IgcmVmbG93LlxuICAgKi9cbiAgaWYgKFxuICAgIG1heENvbnRlbnRIZWlnaHQgPiBtYXhIZWlnaHQgJiZcbiAgICAhYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCAmJlxuICAgIHZpZXdwb3J0UmVmLmN1cnJlbnRcbiAgKSB7XG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICBkaXNwYXRjaCh7XG4gICAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgICAgdmlld3BvcnROb2RlOiB2aWV3cG9ydFJlZi5jdXJyZW50LFxuICAgICAgICBpdGVtQW1vdW50LFxuICAgICAgfSk7XG4gICAgfSk7XG4gICAgYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCA9IHRydWU7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgICByb2xlPVwibGlzdGJveFwiXG4gICAgICAgICAgICBpZD17aWR9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY2dDIn0= */");function VirtualScrollList({id,maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,containerStyle,"data-e2e-test-id":dataE2eTestId,onContentHeightChange}){let scrollableContainerRef=(0,_react.useRef)(null),viewportRef=(0,_react.useRef)(null),adjustedForReflow=(0,_react.useRef)(!1),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=(0,_react.useReducer)(_VirtualScrollListReducer.VirtualScrollReducer,{});(0,_react.useLayoutEffect)(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),(0,_react.useEffect)(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),(0,_react.useEffect)(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),(0,_react.useLayoutEffect)(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]),(0,_react.useEffect)(()=>{onContentHeightChange?.()},[maxContentHeight,onContentHeightChange]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return maxContentHeight>maxHeight&&!adjustedForReflow.current&&viewportRef.current&&(setTimeout(()=>{dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})}),adjustedForReflow.current=!0),_react.default.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight,...containerStyle},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():_react.default.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},_react.default.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`},role:"listbox",id:id},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useRef,useEffect,useMemo}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{usePopupMenu}from"../../../shared/popupMenu/usePopupMenu";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";import{Inline}from"../../Inline/Inline";import{Icon}from"../../Icon/Icon";import{StyledDropdown,StyledOption}from"./StyledSelectComponents";import{dropdownContainerStyle}from"./constants";import{Stack}from"../../Stack/Stack";import{CheckboxRaw}from"../Checkbox/Checkbox";import{Portal}from"../../Portal/Portal";import{ScreenReaderText}from"../../Utilities/ScreenReaderText/ScreenReaderText";export function getOptionId(selectName,optionValue){return`${selectName}Option${optionValue}`}export function getOptionsListId(selectName){return`${selectName}OptionsList`}function scrollToItem(list,itemIndex){list[itemIndex]?.scrollIntoView?.({behavior:"smooth",block:"nearest"})}let CustomOption=styled(StyledOption,{target:"e1d5yhis0",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA6CqB"} */"),StyledListContainer=styled("div",{target:"e1d5yhis1",label:"StyledListContainer"})(({theme,maxHeight})=>({maxHeight,boxSizing:"border-box",overflow:"auto",padding:`${theme.variables.size.spacing.xxs} 0`}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA0D4B"} */");export function OptionsList({isOpen,isMultiSelect=!1,value,disabled,options,portalContainer,triggerRef,triggerWrapperRef,maxHeight,emptyStateMessage="No options available",selectedIndex,optionsListWidth,isVirtualized=!0,name,onCloseDropdown,onSelectedIndexChange,forceChangeFakeSelect}){let optionsListRef=useRef(null),listContainerRef=useRef(null),listItemsRef=useRef(null),refForPositioning=triggerWrapperRef||triggerRef,activeItemIndex=useMemo(()=>options.findIndex(option=>isMultiSelect?value.includes(option.value):option.value===value),[options,isMultiSelect,value]),{menuStyle:optionsListStyle,calculateStyle}=usePopupMenu({triggerRef:refForPositioning,menuRef:optionsListRef,isOpen});useEffect(()=>{if(isOpen&&listContainerRef.current&&!isVirtualized&&optionsListStyle.top){let listElm=listContainerRef.current;listElm.clientHeight<listElm.scrollHeight&&(listItemsRef.current=listElm.querySelectorAll("[data-list-item]"),scrollToItem(listItemsRef.current,activeItemIndex))}else listItemsRef.current=null},[isOpen,isVirtualized,listContainerRef,options,maxHeight,optionsListStyle,activeItemIndex]),useEffect(()=>{listItemsRef.current&&scrollToItem(listItemsRef.current,selectedIndex)},[listItemsRef,selectedIndex]),useKeyboard({Escape:()=>onCloseDropdown(!0),Enter:()=>{let preselectedOption=options[selectedIndex];if(!preselectedOption?.disabled){if(preselectedOption){forceChangeFakeSelect(preselectedOption),isMultiSelect||onCloseDropdown(!0);return}isMultiSelect||onCloseDropdown()}},ArrowUp:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex>0&&(newIndex-=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)},ArrowDown:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex<options.length-1&&(newIndex+=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)}},triggerRef,isOpen&&!disabled);let renderItem=index=>{let option=options[index],id=getOptionId(name,option.value);if(isMultiSelect){let isActive=value.includes(option.value);return React.createElement(CustomOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||forceChangeFakeSelect(option)},onClick:evt=>evt.stopPropagation()},React.createElement(CheckboxRaw,{name:"",disabled:option.disabled,checked:isActive,size:"s",label:option.label,onChange:()=>null,labelHint:option?.description}))}let isActive=value===option.value,optionText=React.createElement(Stack,{space:"zero"},React.createElement(Text,{color:"primary",size:"m"},option.label),option.description&&React.createElement(Text,{color:"tertiary",size:"s"},option.description));return React.createElement(StyledOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||(forceChangeFakeSelect(option),onCloseDropdown(!0))}},isActive?React.createElement(Inline,{noWrap:!0,alignItems:"spaceBetween"},optionText,React.createElement(Box,{space:"zero",tSpace:"xxxs"},React.createElement(Icon,{name:"check",size:"s",color:"accent"}))):optionText)},listStyle={...optionsListStyle,width:optionsListWidth||refForPositioning.current?.getBoundingClientRect().width},listElm=React.createElement(StyledDropdown,{style:listStyle,ref:optionsListRef,onMouseDown:e=>e.preventDefault(),"data-e2e-test-id":"optionsList"},isVirtualized?React.createElement(VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:options.length,emptyState:()=>null,containerStyle:dropdownContainerStyle,itemInView:selectedIndex,itemTemplate:renderItem,onContentHeightChange:calculateStyle}):React.createElement(StyledListContainer,{maxHeight:maxHeight,role:"listbox",id:getOptionsListId(name),ref:listContainerRef},options.map((_option,index)=>renderItem(index))));if(!isOpen)return null;let listPortal=options.length?React.createElement(Portal,{portalContainer:portalContainer},listElm):null;return React.createElement(React.Fragment,null,React.createElement(ScreenReaderText,{role:"status","aria-live":"polite"},options.length?"":emptyStateMessage),listPortal)}
|
|
1
|
+
import React,{useRef,useEffect,useMemo}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{usePopupMenu}from"../../../shared/popupMenu/usePopupMenu";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";import{Inline}from"../../Inline/Inline";import{Icon}from"../../Icon/Icon";import{StyledDropdown,StyledOption}from"./StyledSelectComponents";import{dropdownContainerStyle}from"./constants";import{Stack}from"../../Stack/Stack";import{CheckboxRaw}from"../Checkbox/Checkbox";import{Portal}from"../../Portal/Portal";import{ScreenReaderText}from"../../Utilities/ScreenReaderText/ScreenReaderText";export function getOptionId(selectName,optionValue){return`${selectName}Option${optionValue}`}export function getOptionsListId(selectName){return`${selectName}OptionsList`}function scrollToItem(list,itemIndex){list[itemIndex]?.scrollIntoView?.({behavior:"smooth",block:"nearest"})}let CustomOption=styled(StyledOption,{target:"e159rw3y0",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          id={getOptionsListId(name)}\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA6CqB"} */"),StyledListContainer=styled("div",{target:"e159rw3y1",label:"StyledListContainer"})(({theme,maxHeight})=>({maxHeight,boxSizing:"border-box",overflow:"auto",padding:`${theme.variables.size.spacing.xxs} 0`}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Combobox/OptionsList.tsx","sources":["src/components/Form/Combobox/OptionsList.tsx"],"sourcesContent":["import type { RefObject } from \"react\";\nimport React, { useRef, useEffect, useMemo } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { CommonSelectProps, SelectOption } from \"./Combobox\";\nimport { usePopupMenu } from \"../../../shared/popupMenu/usePopupMenu\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport { Icon } from \"../../Icon/Icon\";\nimport { StyledDropdown, StyledOption } from \"./StyledSelectComponents\";\nimport { dropdownContainerStyle } from \"./constants\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { ScreenReaderText } from \"../../Utilities/ScreenReaderText/ScreenReaderText\";\n\ntype OptionsListProps = {\n  isOpen: boolean;\n  isMultiSelect?: boolean;\n  triggerRef: RefObject<HTMLInputElement>;\n  triggerWrapperRef?: RefObject<HTMLDivElement>;\n  onCloseDropdown: (noSelect?: boolean) => void;\n  selectedIndex: number;\n  onSelectedIndexChange: (index: number) => void;\n  forceChangeFakeSelect: (selectedOption: SelectOption) => void;\n  isVirtualized?: boolean;\n} & Partial<CommonSelectProps>;\n\nexport function getOptionId(selectName: string, optionValue: string): string {\n  return `${selectName}Option${optionValue}`;\n}\n\nexport function getOptionsListId(selectName: string): string {\n  return `${selectName}OptionsList`;\n}\n\nfunction scrollToItem(list: HTMLDivElement[], itemIndex: number) {\n  list[itemIndex]?.scrollIntoView?.({\n    behavior: \"smooth\",\n    block: \"nearest\",\n  });\n}\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst StyledListContainer = styled.div<Pick<OptionsListProps, \"maxHeight\">>(\n  ({ theme, maxHeight }) => ({\n    maxHeight,\n    boxSizing: \"border-box\",\n    overflow: \"auto\",\n    padding: `${theme.variables.size.spacing.xxs} 0`,\n  })\n);\n\nexport function OptionsList({\n  isOpen,\n  isMultiSelect = false,\n  value,\n  disabled,\n  options,\n  portalContainer,\n  triggerRef,\n  triggerWrapperRef,\n  maxHeight,\n  emptyStateMessage = \"No options available\",\n  selectedIndex,\n  optionsListWidth,\n  isVirtualized = true,\n  name,\n  onCloseDropdown,\n  onSelectedIndexChange,\n  forceChangeFakeSelect,\n}: OptionsListProps): React.ReactElement {\n  const optionsListRef = useRef(null);\n  const listContainerRef = useRef(null);\n  const listItemsRef = useRef(null);\n  const refForPositioning = triggerWrapperRef || triggerRef;\n  const activeItemIndex = useMemo(\n    () =>\n      options.findIndex((option) =>\n        isMultiSelect ? value.includes(option.value) : option.value === value\n      ),\n    [options, isMultiSelect, value]\n  );\n\n  const { menuStyle: optionsListStyle, calculateStyle } = usePopupMenu({\n    triggerRef: refForPositioning,\n    menuRef: optionsListRef,\n    isOpen,\n  });\n\n  useEffect(() => {\n    if (\n      isOpen &&\n      listContainerRef.current &&\n      !isVirtualized &&\n      optionsListStyle.top\n    ) {\n      const listElm = listContainerRef.current;\n      // determine overflow\n      if (listElm.clientHeight < listElm.scrollHeight) {\n        listItemsRef.current = listElm.querySelectorAll(\"[data-list-item]\");\n        scrollToItem(listItemsRef.current, activeItemIndex);\n      }\n    } else {\n      listItemsRef.current = null;\n    }\n  }, [\n    isOpen,\n    isVirtualized,\n    listContainerRef,\n    options,\n    maxHeight,\n    optionsListStyle,\n    activeItemIndex,\n  ]);\n\n  useEffect(() => {\n    if (listItemsRef.current) {\n      scrollToItem(listItemsRef.current, selectedIndex);\n    }\n  }, [listItemsRef, selectedIndex]);\n\n  useKeyboard(\n    {\n      Escape: () => onCloseDropdown(true),\n      Enter: () => {\n        const preselectedOption = options[selectedIndex];\n\n        if (!preselectedOption?.disabled) {\n          if (preselectedOption) {\n            forceChangeFakeSelect(preselectedOption);\n            if (!isMultiSelect) {\n              onCloseDropdown(true);\n            }\n            return;\n          }\n          if (!isMultiSelect) {\n            onCloseDropdown();\n          }\n        }\n      },\n      ArrowUp: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex > 0) {\n          newIndex -= 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n      ArrowDown: () => {\n        let newIndex = selectedIndex === -1 ? activeItemIndex : selectedIndex;\n        while (newIndex < options.length - 1) {\n          newIndex += 1;\n          if (!options[newIndex]?.disabled) {\n            break;\n          }\n        }\n        // Only update the selection if the new item is not disabled\n        if (!options[newIndex]?.disabled) {\n          onSelectedIndexChange(newIndex);\n        }\n      },\n    },\n    triggerRef,\n    isOpen && !disabled\n  );\n\n  const renderItem = (index: number) => {\n    const option: SelectOption = options[index];\n    // Generate unique id for the option\n    const id = getOptionId(name, option.value);\n\n    if (isMultiSelect) {\n      const isActive = value.includes(option.value);\n\n      return (\n        <CustomOption\n          role=\"option\"\n          aria-selected={isActive}\n          aria-disabled={option.disabled}\n          id={id}\n          key={option.value}\n          active={isActive}\n          disabled={option.disabled}\n          preSelected={selectedIndex === index}\n          data-list-item\n          onMouseDown={() => {\n            if (!option.disabled) {\n              forceChangeFakeSelect(option);\n            }\n          }}\n          // stop event propagation to not close modal on selection if select is within a Modal\n          onClick={(evt) => evt.stopPropagation()}\n        >\n          <CheckboxRaw\n            name=\"\"\n            disabled={option.disabled}\n            checked={isActive}\n            size=\"s\"\n            label={option.label}\n            onChange={() => null}\n            labelHint={option?.description}\n          />\n        </CustomOption>\n      );\n    }\n\n    // else render single select item\n    const isActive = value === option.value;\n    const optionText = (\n      <Stack space=\"zero\">\n        <Text color=\"primary\" size=\"m\">\n          {option.label}\n        </Text>\n        {option.description && (\n          <Text color=\"tertiary\" size=\"s\">\n            {option.description}\n          </Text>\n        )}\n      </Stack>\n    );\n\n    return (\n      <StyledOption\n        role=\"option\"\n        aria-selected={isActive}\n        aria-disabled={option.disabled}\n        id={id}\n        key={option.value}\n        active={isActive}\n        disabled={option.disabled}\n        preSelected={selectedIndex === index}\n        data-list-item\n        onMouseDown={() => {\n          if (!option.disabled) {\n            forceChangeFakeSelect(option);\n            onCloseDropdown(true);\n          }\n        }}\n      >\n        {isActive ? (\n          <Inline noWrap alignItems=\"spaceBetween\">\n            {optionText}\n            <Box space=\"zero\" tSpace=\"xxxs\">\n              <Icon name=\"check\" size=\"s\" color=\"accent\" />\n            </Box>\n          </Inline>\n        ) : (\n          optionText\n        )}\n      </StyledOption>\n    );\n  };\n\n  const listStyle = {\n    ...optionsListStyle,\n    width:\n      optionsListWidth ||\n      refForPositioning.current?.getBoundingClientRect().width,\n  };\n\n  const listElm = (\n    <StyledDropdown\n      style={listStyle}\n      ref={optionsListRef}\n      // this is to prevent known bug of Chrome when element\n      // loses focus on click on the scrollbar\n      onMouseDown={(e) => e.preventDefault()}\n      data-e2e-test-id=\"optionsList\"\n    >\n      {isVirtualized ? (\n        <VirtualScrollList\n          id={getOptionsListId(name)}\n          maxHeight={maxHeight}\n          itemHeight={36}\n          itemAmount={options.length}\n          emptyState={() => null}\n          containerStyle={dropdownContainerStyle}\n          itemInView={selectedIndex}\n          itemTemplate={renderItem}\n          onContentHeightChange={calculateStyle}\n        />\n      ) : (\n        <StyledListContainer\n          maxHeight={maxHeight}\n          role=\"listbox\"\n          id={getOptionsListId(name)}\n          ref={listContainerRef}\n        >\n          {options.map((_option, index) => renderItem(index))}\n        </StyledListContainer>\n      )}\n    </StyledDropdown>\n  );\n\n  if (!isOpen) {\n    return null;\n  }\n\n  const listPortal = options.length ? (\n    <Portal portalContainer={portalContainer}>{listElm}</Portal>\n  ) : null;\n\n  return (\n    <>\n      <ScreenReaderText role=\"status\" aria-live=\"polite\">\n        {!options.length ? emptyStateMessage : \"\"}\n      </ScreenReaderText>\n      {listPortal}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA0D4B"} */");export function OptionsList({isOpen,isMultiSelect=!1,value,disabled,options,portalContainer,triggerRef,triggerWrapperRef,maxHeight,emptyStateMessage="No options available",selectedIndex,optionsListWidth,isVirtualized=!0,name,onCloseDropdown,onSelectedIndexChange,forceChangeFakeSelect}){let optionsListRef=useRef(null),listContainerRef=useRef(null),listItemsRef=useRef(null),refForPositioning=triggerWrapperRef||triggerRef,activeItemIndex=useMemo(()=>options.findIndex(option=>isMultiSelect?value.includes(option.value):option.value===value),[options,isMultiSelect,value]),{menuStyle:optionsListStyle,calculateStyle}=usePopupMenu({triggerRef:refForPositioning,menuRef:optionsListRef,isOpen});useEffect(()=>{if(isOpen&&listContainerRef.current&&!isVirtualized&&optionsListStyle.top){let listElm=listContainerRef.current;listElm.clientHeight<listElm.scrollHeight&&(listItemsRef.current=listElm.querySelectorAll("[data-list-item]"),scrollToItem(listItemsRef.current,activeItemIndex))}else listItemsRef.current=null},[isOpen,isVirtualized,listContainerRef,options,maxHeight,optionsListStyle,activeItemIndex]),useEffect(()=>{listItemsRef.current&&scrollToItem(listItemsRef.current,selectedIndex)},[listItemsRef,selectedIndex]),useKeyboard({Escape:()=>onCloseDropdown(!0),Enter:()=>{let preselectedOption=options[selectedIndex];if(!preselectedOption?.disabled){if(preselectedOption){forceChangeFakeSelect(preselectedOption),isMultiSelect||onCloseDropdown(!0);return}isMultiSelect||onCloseDropdown()}},ArrowUp:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex>0&&(newIndex-=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)},ArrowDown:()=>{let newIndex=-1===selectedIndex?activeItemIndex:selectedIndex;for(;newIndex<options.length-1&&(newIndex+=1,options[newIndex]?.disabled););options[newIndex]?.disabled||onSelectedIndexChange(newIndex)}},triggerRef,isOpen&&!disabled);let renderItem=index=>{let option=options[index],id=getOptionId(name,option.value);if(isMultiSelect){let isActive=value.includes(option.value);return React.createElement(CustomOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||forceChangeFakeSelect(option)},onClick:evt=>evt.stopPropagation()},React.createElement(CheckboxRaw,{name:"",disabled:option.disabled,checked:isActive,size:"s",label:option.label,onChange:()=>null,labelHint:option?.description}))}let isActive=value===option.value,optionText=React.createElement(Stack,{space:"zero"},React.createElement(Text,{color:"primary",size:"m"},option.label),option.description&&React.createElement(Text,{color:"tertiary",size:"s"},option.description));return React.createElement(StyledOption,{role:"option","aria-selected":isActive,"aria-disabled":option.disabled,id:id,key:option.value,active:isActive,disabled:option.disabled,preSelected:selectedIndex===index,"data-list-item":!0,onMouseDown:()=>{option.disabled||(forceChangeFakeSelect(option),onCloseDropdown(!0))}},isActive?React.createElement(Inline,{noWrap:!0,alignItems:"spaceBetween"},optionText,React.createElement(Box,{space:"zero",tSpace:"xxxs"},React.createElement(Icon,{name:"check",size:"s",color:"accent"}))):optionText)},listStyle={...optionsListStyle,width:optionsListWidth||refForPositioning.current?.getBoundingClientRect().width},listElm=React.createElement(StyledDropdown,{style:listStyle,ref:optionsListRef,onMouseDown:e=>e.preventDefault(),"data-e2e-test-id":"optionsList"},isVirtualized?React.createElement(VirtualScrollList,{id:getOptionsListId(name),maxHeight:maxHeight,itemHeight:36,itemAmount:options.length,emptyState:()=>null,containerStyle:dropdownContainerStyle,itemInView:selectedIndex,itemTemplate:renderItem,onContentHeightChange:calculateStyle}):React.createElement(StyledListContainer,{maxHeight:maxHeight,role:"listbox",id:getOptionsListId(name),ref:listContainerRef},options.map((_option,index)=>renderItem(index))));if(!isOpen)return null;let listPortal=options.length?React.createElement(Portal,{portalContainer:portalContainer},listElm):null;return React.createElement(React.Fragment,null,React.createElement(ScreenReaderText,{role:"status","aria-live":"polite"},options.length?"":emptyStateMessage),listPortal)}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import React from "react";
|
|
2
2
|
export type VirtualScrollListProps = {
|
|
3
3
|
"data-e2e-test-id"?: string;
|
|
4
|
+
id: string;
|
|
4
5
|
maxHeight: number;
|
|
5
6
|
itemHeight: number;
|
|
6
7
|
itemAmount: number;
|
|
@@ -10,4 +11,4 @@ export type VirtualScrollListProps = {
|
|
|
10
11
|
containerStyle?: React.CSSProperties;
|
|
11
12
|
onContentHeightChange?: () => void;
|
|
12
13
|
};
|
|
13
|
-
export declare function VirtualScrollList({ maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, containerStyle, "data-e2e-test-id": dataE2eTestId, onContentHeightChange, }: VirtualScrollListProps): React.ReactElement;
|
|
14
|
+
export declare function VirtualScrollList({ id, maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, containerStyle, "data-e2e-test-id": dataE2eTestId, onContentHeightChange, }: VirtualScrollListProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import styled from"@emotion/styled";import React,{useRef,useReducer,useLayoutEffect,useEffect}from"react";import{VirtualScrollReducer}from"./VirtualScrollListReducer";let StyledContainer=styled("div",{target:"e19y1j3j0",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbiAgb25Db250ZW50SGVpZ2h0Q2hhbmdlLFxufTogVmlydHVhbFNjcm9sbExpc3RQcm9wcyk6IFJlYWN0LlJlYWN0RWxlbWVudCB7XG4gIGNvbnN0IHNjcm9sbGFibGVDb250YWluZXJSZWYgPSB1c2VSZWYobnVsbCk7XG4gIGNvbnN0IHZpZXdwb3J0UmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCBhZGp1c3RlZEZvclJlZmxvdyA9IHVzZVJlZihmYWxzZSk7XG5cbiAgY29uc3QgW1xuICAgIHtcbiAgICAgIHNjcm9sbGVkSXRlbUNvdW50LFxuICAgICAgYW1vdW50T2ZJdGVtc0luVmlldyxcbiAgICAgIHNjcm9sbGVkSW5QeCxcbiAgICAgIG1heENvbnRlbnRIZWlnaHQgPSBtYXhIZWlnaHQsXG4gICAgICByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uLFxuICAgIH0sXG4gICAgZGlzcGF0Y2gsXG4gIF0gPSB1c2VSZWR1Y2VyKFZpcnR1YWxTY3JvbGxSZWR1Y2VyLCB7fSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7IHR5cGU6IFwicmVzZXRcIiwgaXRlbUhlaWdodCwgbWF4SGVpZ2h0LCBvdmVyc2NhbiB9KTtcbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gMDtcbiAgfSwgW2l0ZW1IZWlnaHQsIGl0ZW1BbW91bnQsIG1heEhlaWdodF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goe1xuICAgICAgdHlwZTogXCJyZWNvbW1lbmRTY3JvbGxQb3NpdGlvblwiLFxuICAgICAgc2Nyb2xsVG9wOiBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wLFxuICAgICAgaXRlbVRvQmVJblZpZXc6IGl0ZW1JblZpZXcsXG4gICAgfSk7XG4gIH0sIFtpdGVtSW5WaWV3XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbiA9PT0gbnVsbCkgcmV0dXJuO1xuXG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb247XG4gIH0sIFtyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uXSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcblxuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgIHZpZXdwb3J0Tm9kZTogdmlld3BvcnRSZWYuY3VycmVudCxcbiAgICAgIGl0ZW1BbW91bnQsXG4gICAgfSk7XG4gIH0sIFtpdGVtQW1vdW50LCBzY3JvbGxlZEl0ZW1Db3VudF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgb25Db250ZW50SGVpZ2h0Q2hhbmdlPy4oKTtcbiAgfSwgW21heENvbnRlbnRIZWlnaHQsIG9uQ29udGVudEhlaWdodENoYW5nZV0pO1xuXG4gIGNvbnN0IGl0ZW1Db3VudFRvQmVSZW5kZXJlZCA9IE1hdGgubWluKFxuICAgIE1hdGgubWF4KDAsIGl0ZW1BbW91bnQgLSBzY3JvbGxlZEl0ZW1Db3VudCksXG4gICAgYW1vdW50T2ZJdGVtc0luVmlld1xuICApO1xuXG4gIC8qIFNvbWV0aW1lcywgb3B0aW9uIHRleHQgY2FuIHJlZmxvdyBkdWUgdGhlIGFwcGVhcmFuY2Ugb2YgdmVydGljYWwgc2Nyb2xsYmFyIHdoaWNoIGNhbiBjYXVzZSB0ZXh0IHRvIHdyYXAgdG8gbXVsdGlwbGUgbGluZXMgY2F1c2luZyB2aWV3cG9ydFJlZiBoZWlnaHQgdG8gY2hhbmdlLiBJZiBjb250YWluZXIgaGFzIHNjcm9sbCB0aGVuIHdlIGRpc3BhdGNoIGB1cGRhdGVWaWV3cG9ydGAgb25lIG1vcmUgdGltZSB0byBnZXQgdGhlIGNvcnJlY3QgaGVpZ2h0LCB3aXRoIGEgc2V0VGltZW91dCB0byBhbGxvdyB0aW1lIGZvciByZWZsb3cuXG4gICAqL1xuICBpZiAoXG4gICAgbWF4Q29udGVudEhlaWdodCA+IG1heEhlaWdodCAmJlxuICAgICFhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ICYmXG4gICAgdmlld3BvcnRSZWYuY3VycmVudFxuICApIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGRpc3BhdGNoKHtcbiAgICAgICAgdHlwZTogXCJ1cGRhdGVWaWV3cG9ydFwiLFxuICAgICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICAgIGl0ZW1BbW91bnQsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgICBhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ID0gdHJ1ZTtcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZENvbnRhaW5lclxuICAgICAgcmVmPXtzY3JvbGxhYmxlQ29udGFpbmVyUmVmfVxuICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHR9XG4gICAgICBzdHlsZT17e1xuICAgICAgICBtYXhIZWlnaHQsXG4gICAgICAgIC4uLmNvbnRhaW5lclN0eWxlLFxuICAgICAgfX1cbiAgICAgIGRhdGEtZTJlLXRlc3QtaWQ9e2RhdGFFMmVUZXN0SWR9XG4gICAgICBkYXRhLWRzLWlkPVwiVmlydHVhbFNjcm9sbExpc3RcIlxuICAgICAgb25TY3JvbGw9eyhlOiBSZWFjdC5VSUV2ZW50PEhUTUxFbGVtZW50PikgPT4ge1xuICAgICAgICBkaXNwYXRjaCh7XG4gICAgICAgICAgdHlwZTogXCJzY3JvbGxcIixcbiAgICAgICAgICBzY3JvbGxUb3A6IGUuY3VycmVudFRhcmdldC5zY3JvbGxUb3AsXG4gICAgICAgIH0pO1xuICAgICAgfX1cbiAgICA+XG4gICAgICB7aXRlbUFtb3VudCA9PT0gMCA/IChcbiAgICAgICAgZW1wdHlTdGF0ZSgpXG4gICAgICApIDogKFxuICAgICAgICA8U3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgbWF4Q29udGVudEhlaWdodD17bWF4Q29udGVudEhlaWdodH0+XG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgcmVmPXt2aWV3cG9ydFJlZn1cbiAgICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICAgIHRyYW5zZm9ybTogYHRyYW5zbGF0ZVkoJHtzY3JvbGxlZEluUHh9cHhgLFxuICAgICAgICAgICAgfX1cbiAgICAgICAgICAgIHJvbGU9XCJsaXN0Ym94XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7ISFpdGVtQ291bnRUb0JlUmVuZGVyZWQgJiZcbiAgICAgICAgICAgICAgbmV3IEFycmF5KGl0ZW1Db3VudFRvQmVSZW5kZXJlZClcbiAgICAgICAgICAgICAgICAuZmlsbCgwKVxuICAgICAgICAgICAgICAgIC5tYXAoKF8sIGluZGV4KSA9PiBpdGVtVGVtcGxhdGUoc2Nyb2xsZWRJdGVtQ291bnQgKyBpbmRleCkpfVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICA8L1N0eWxlZFNjcm9sbGFibGVDb250ZW50PlxuICAgICAgKX1cbiAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgKTtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLd0IifQ== */"),StyledScrollableContent=styled("div",{target:"e19y1j3j1",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbiAgb25Db250ZW50SGVpZ2h0Q2hhbmdlLFxufTogVmlydHVhbFNjcm9sbExpc3RQcm9wcyk6IFJlYWN0LlJlYWN0RWxlbWVudCB7XG4gIGNvbnN0IHNjcm9sbGFibGVDb250YWluZXJSZWYgPSB1c2VSZWYobnVsbCk7XG4gIGNvbnN0IHZpZXdwb3J0UmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCBhZGp1c3RlZEZvclJlZmxvdyA9IHVzZVJlZihmYWxzZSk7XG5cbiAgY29uc3QgW1xuICAgIHtcbiAgICAgIHNjcm9sbGVkSXRlbUNvdW50LFxuICAgICAgYW1vdW50T2ZJdGVtc0luVmlldyxcbiAgICAgIHNjcm9sbGVkSW5QeCxcbiAgICAgIG1heENvbnRlbnRIZWlnaHQgPSBtYXhIZWlnaHQsXG4gICAgICByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uLFxuICAgIH0sXG4gICAgZGlzcGF0Y2gsXG4gIF0gPSB1c2VSZWR1Y2VyKFZpcnR1YWxTY3JvbGxSZWR1Y2VyLCB7fSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7IHR5cGU6IFwicmVzZXRcIiwgaXRlbUhlaWdodCwgbWF4SGVpZ2h0LCBvdmVyc2NhbiB9KTtcbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gMDtcbiAgfSwgW2l0ZW1IZWlnaHQsIGl0ZW1BbW91bnQsIG1heEhlaWdodF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goe1xuICAgICAgdHlwZTogXCJyZWNvbW1lbmRTY3JvbGxQb3NpdGlvblwiLFxuICAgICAgc2Nyb2xsVG9wOiBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wLFxuICAgICAgaXRlbVRvQmVJblZpZXc6IGl0ZW1JblZpZXcsXG4gICAgfSk7XG4gIH0sIFtpdGVtSW5WaWV3XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbiA9PT0gbnVsbCkgcmV0dXJuO1xuXG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb247XG4gIH0sIFtyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uXSk7XG5cbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcblxuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgIHZpZXdwb3J0Tm9kZTogdmlld3BvcnRSZWYuY3VycmVudCxcbiAgICAgIGl0ZW1BbW91bnQsXG4gICAgfSk7XG4gIH0sIFtpdGVtQW1vdW50LCBzY3JvbGxlZEl0ZW1Db3VudF0pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgb25Db250ZW50SGVpZ2h0Q2hhbmdlPy4oKTtcbiAgfSwgW21heENvbnRlbnRIZWlnaHQsIG9uQ29udGVudEhlaWdodENoYW5nZV0pO1xuXG4gIGNvbnN0IGl0ZW1Db3VudFRvQmVSZW5kZXJlZCA9IE1hdGgubWluKFxuICAgIE1hdGgubWF4KDAsIGl0ZW1BbW91bnQgLSBzY3JvbGxlZEl0ZW1Db3VudCksXG4gICAgYW1vdW50T2ZJdGVtc0luVmlld1xuICApO1xuXG4gIC8qIFNvbWV0aW1lcywgb3B0aW9uIHRleHQgY2FuIHJlZmxvdyBkdWUgdGhlIGFwcGVhcmFuY2Ugb2YgdmVydGljYWwgc2Nyb2xsYmFyIHdoaWNoIGNhbiBjYXVzZSB0ZXh0IHRvIHdyYXAgdG8gbXVsdGlwbGUgbGluZXMgY2F1c2luZyB2aWV3cG9ydFJlZiBoZWlnaHQgdG8gY2hhbmdlLiBJZiBjb250YWluZXIgaGFzIHNjcm9sbCB0aGVuIHdlIGRpc3BhdGNoIGB1cGRhdGVWaWV3cG9ydGAgb25lIG1vcmUgdGltZSB0byBnZXQgdGhlIGNvcnJlY3QgaGVpZ2h0LCB3aXRoIGEgc2V0VGltZW91dCB0byBhbGxvdyB0aW1lIGZvciByZWZsb3cuXG4gICAqL1xuICBpZiAoXG4gICAgbWF4Q29udGVudEhlaWdodCA+IG1heEhlaWdodCAmJlxuICAgICFhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ICYmXG4gICAgdmlld3BvcnRSZWYuY3VycmVudFxuICApIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGRpc3BhdGNoKHtcbiAgICAgICAgdHlwZTogXCJ1cGRhdGVWaWV3cG9ydFwiLFxuICAgICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICAgIGl0ZW1BbW91bnQsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgICBhZGp1c3RlZEZvclJlZmxvdy5jdXJyZW50ID0gdHJ1ZTtcbiAgfVxuXG4gIHJldHVybiAoXG4gICAgPFN0eWxlZENvbnRhaW5lclxuICAgICAgcmVmPXtzY3JvbGxhYmxlQ29udGFpbmVyUmVmfVxuICAgICAgbWF4SGVpZ2h0PXttYXhIZWlnaHR9XG4gICAgICBzdHlsZT17e1xuICAgICAgICBtYXhIZWlnaHQsXG4gICAgICAgIC4uLmNvbnRhaW5lclN0eWxlLFxuICAgICAgfX1cbiAgICAgIGRhdGEtZTJlLXRlc3QtaWQ9e2RhdGFFMmVUZXN0SWR9XG4gICAgICBkYXRhLWRzLWlkPVwiVmlydHVhbFNjcm9sbExpc3RcIlxuICAgICAgb25TY3JvbGw9eyhlOiBSZWFjdC5VSUV2ZW50PEhUTUxFbGVtZW50PikgPT4ge1xuICAgICAgICBkaXNwYXRjaCh7XG4gICAgICAgICAgdHlwZTogXCJzY3JvbGxcIixcbiAgICAgICAgICBzY3JvbGxUb3A6IGUuY3VycmVudFRhcmdldC5zY3JvbGxUb3AsXG4gICAgICAgIH0pO1xuICAgICAgfX1cbiAgICA+XG4gICAgICB7aXRlbUFtb3VudCA9PT0gMCA/IChcbiAgICAgICAgZW1wdHlTdGF0ZSgpXG4gICAgICApIDogKFxuICAgICAgICA8U3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgbWF4Q29udGVudEhlaWdodD17bWF4Q29udGVudEhlaWdodH0+XG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgcmVmPXt2aWV3cG9ydFJlZn1cbiAgICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICAgIHRyYW5zZm9ybTogYHRyYW5zbGF0ZVkoJHtzY3JvbGxlZEluUHh9cHhgLFxuICAgICAgICAgICAgfX1cbiAgICAgICAgICAgIHJvbGU9XCJsaXN0Ym94XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7ISFpdGVtQ291bnRUb0JlUmVuZGVyZWQgJiZcbiAgICAgICAgICAgICAgbmV3IEFycmF5KGl0ZW1Db3VudFRvQmVSZW5kZXJlZClcbiAgICAgICAgICAgICAgICAuZmlsbCgwKVxuICAgICAgICAgICAgICAgIC5tYXAoKF8sIGluZGV4KSA9PiBpdGVtVGVtcGxhdGUoc2Nyb2xsZWRJdGVtQ291bnQgKyBpbmRleCkpfVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICA8L1N0eWxlZFNjcm9sbGFibGVDb250ZW50PlxuICAgICAgKX1cbiAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgKTtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFjZ0MifQ== */");export function VirtualScrollList({maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,containerStyle,"data-e2e-test-id":dataE2eTestId,onContentHeightChange}){let scrollableContainerRef=useRef(null),viewportRef=useRef(null),adjustedForReflow=useRef(!1),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=useReducer(VirtualScrollReducer,{});useLayoutEffect(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),useEffect(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),useEffect(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),useLayoutEffect(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]),useEffect(()=>{onContentHeightChange?.()},[maxContentHeight,onContentHeightChange]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return maxContentHeight>maxHeight&&!adjustedForReflow.current&&viewportRef.current&&(setTimeout(()=>{dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})}),adjustedForReflow.current=!0),React.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight,...containerStyle},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():React.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},React.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`},role:"listbox"},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|
|
1
|
+
import styled from"@emotion/styled";import React,{useRef,useReducer,useLayoutEffect,useEffect}from"react";import{VirtualScrollReducer}from"./VirtualScrollListReducer";let StyledContainer=styled("div",{target:"e1hq8rdx0",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgaWQ6IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBpZCxcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBjb250YWluZXJTdHlsZSxcbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI6IGRhdGFFMmVUZXN0SWQsXG4gIG9uQ29udGVudEhlaWdodENoYW5nZSxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcbiAgY29uc3QgYWRqdXN0ZWRGb3JSZWZsb3cgPSB1c2VSZWYoZmFsc2UpO1xuXG4gIGNvbnN0IFtcbiAgICB7XG4gICAgICBzY3JvbGxlZEl0ZW1Db3VudCxcbiAgICAgIGFtb3VudE9mSXRlbXNJblZpZXcsXG4gICAgICBzY3JvbGxlZEluUHgsXG4gICAgICBtYXhDb250ZW50SGVpZ2h0ID0gbWF4SGVpZ2h0LFxuICAgICAgcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbixcbiAgICB9LFxuICAgIGRpc3BhdGNoLFxuICBdID0gdXNlUmVkdWNlcihWaXJ0dWFsU2Nyb2xsUmVkdWNlciwge30pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goeyB0eXBlOiBcInJlc2V0XCIsIGl0ZW1IZWlnaHQsIG1heEhlaWdodCwgb3ZlcnNjYW4gfSk7XG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IDA7XG4gIH0sIFtpdGVtSGVpZ2h0LCBpdGVtQW1vdW50LCBtYXhIZWlnaHRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwicmVjb21tZW5kU2Nyb2xsUG9zaXRpb25cIixcbiAgICAgIHNjcm9sbFRvcDogc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCxcbiAgICAgIGl0ZW1Ub0JlSW5WaWV3OiBpdGVtSW5WaWV3LFxuICAgIH0pO1xuICB9LCBbaXRlbUluVmlld10pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24gPT09IG51bGwpIHJldHVybjtcblxuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uO1xuICB9LCBbcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbl0pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCF2aWV3cG9ydFJlZi5jdXJyZW50KSByZXR1cm47XG5cbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIG9uQ29udGVudEhlaWdodENoYW5nZT8uKCk7XG4gIH0sIFttYXhDb250ZW50SGVpZ2h0LCBvbkNvbnRlbnRIZWlnaHRDaGFuZ2VdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICAvKiBTb21ldGltZXMsIG9wdGlvbiB0ZXh0IGNhbiByZWZsb3cgZHVlIHRoZSBhcHBlYXJhbmNlIG9mIHZlcnRpY2FsIHNjcm9sbGJhciB3aGljaCBjYW4gY2F1c2UgdGV4dCB0byB3cmFwIHRvIG11bHRpcGxlIGxpbmVzIGNhdXNpbmcgdmlld3BvcnRSZWYgaGVpZ2h0IHRvIGNoYW5nZS4gSWYgY29udGFpbmVyIGhhcyBzY3JvbGwgdGhlbiB3ZSBkaXNwYXRjaCBgdXBkYXRlVmlld3BvcnRgIG9uZSBtb3JlIHRpbWUgdG8gZ2V0IHRoZSBjb3JyZWN0IGhlaWdodCwgd2l0aCBhIHNldFRpbWVvdXQgdG8gYWxsb3cgdGltZSBmb3IgcmVmbG93LlxuICAgKi9cbiAgaWYgKFxuICAgIG1heENvbnRlbnRIZWlnaHQgPiBtYXhIZWlnaHQgJiZcbiAgICAhYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCAmJlxuICAgIHZpZXdwb3J0UmVmLmN1cnJlbnRcbiAgKSB7XG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICBkaXNwYXRjaCh7XG4gICAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgICAgdmlld3BvcnROb2RlOiB2aWV3cG9ydFJlZi5jdXJyZW50LFxuICAgICAgICBpdGVtQW1vdW50LFxuICAgICAgfSk7XG4gICAgfSk7XG4gICAgYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCA9IHRydWU7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgICByb2xlPVwibGlzdGJveFwiXG4gICAgICAgICAgICBpZD17aWR9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS3dCIn0= */"),StyledScrollableContent=styled("div",{target:"e1hq8rdx1",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgaWQ6IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG4gIG9uQ29udGVudEhlaWdodENoYW5nZT86ICgpID0+IHZvaWQ7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBpZCxcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBjb250YWluZXJTdHlsZSxcbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI6IGRhdGFFMmVUZXN0SWQsXG4gIG9uQ29udGVudEhlaWdodENoYW5nZSxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcbiAgY29uc3QgYWRqdXN0ZWRGb3JSZWZsb3cgPSB1c2VSZWYoZmFsc2UpO1xuXG4gIGNvbnN0IFtcbiAgICB7XG4gICAgICBzY3JvbGxlZEl0ZW1Db3VudCxcbiAgICAgIGFtb3VudE9mSXRlbXNJblZpZXcsXG4gICAgICBzY3JvbGxlZEluUHgsXG4gICAgICBtYXhDb250ZW50SGVpZ2h0ID0gbWF4SGVpZ2h0LFxuICAgICAgcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbixcbiAgICB9LFxuICAgIGRpc3BhdGNoLFxuICBdID0gdXNlUmVkdWNlcihWaXJ0dWFsU2Nyb2xsUmVkdWNlciwge30pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgZGlzcGF0Y2goeyB0eXBlOiBcInJlc2V0XCIsIGl0ZW1IZWlnaHQsIG1heEhlaWdodCwgb3ZlcnNjYW4gfSk7XG4gICAgc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCA9IDA7XG4gIH0sIFtpdGVtSGVpZ2h0LCBpdGVtQW1vdW50LCBtYXhIZWlnaHRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHtcbiAgICAgIHR5cGU6IFwicmVjb21tZW5kU2Nyb2xsUG9zaXRpb25cIixcbiAgICAgIHNjcm9sbFRvcDogc2Nyb2xsYWJsZUNvbnRhaW5lclJlZi5jdXJyZW50LnNjcm9sbFRvcCxcbiAgICAgIGl0ZW1Ub0JlSW5WaWV3OiBpdGVtSW5WaWV3LFxuICAgIH0pO1xuICB9LCBbaXRlbUluVmlld10pO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24gPT09IG51bGwpIHJldHVybjtcblxuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSByZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uO1xuICB9LCBbcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbl0pO1xuXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCF2aWV3cG9ydFJlZi5jdXJyZW50KSByZXR1cm47XG5cbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIG9uQ29udGVudEhlaWdodENoYW5nZT8uKCk7XG4gIH0sIFttYXhDb250ZW50SGVpZ2h0LCBvbkNvbnRlbnRIZWlnaHRDaGFuZ2VdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICAvKiBTb21ldGltZXMsIG9wdGlvbiB0ZXh0IGNhbiByZWZsb3cgZHVlIHRoZSBhcHBlYXJhbmNlIG9mIHZlcnRpY2FsIHNjcm9sbGJhciB3aGljaCBjYW4gY2F1c2UgdGV4dCB0byB3cmFwIHRvIG11bHRpcGxlIGxpbmVzIGNhdXNpbmcgdmlld3BvcnRSZWYgaGVpZ2h0IHRvIGNoYW5nZS4gSWYgY29udGFpbmVyIGhhcyBzY3JvbGwgdGhlbiB3ZSBkaXNwYXRjaCBgdXBkYXRlVmlld3BvcnRgIG9uZSBtb3JlIHRpbWUgdG8gZ2V0IHRoZSBjb3JyZWN0IGhlaWdodCwgd2l0aCBhIHNldFRpbWVvdXQgdG8gYWxsb3cgdGltZSBmb3IgcmVmbG93LlxuICAgKi9cbiAgaWYgKFxuICAgIG1heENvbnRlbnRIZWlnaHQgPiBtYXhIZWlnaHQgJiZcbiAgICAhYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCAmJlxuICAgIHZpZXdwb3J0UmVmLmN1cnJlbnRcbiAgKSB7XG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICBkaXNwYXRjaCh7XG4gICAgICAgIHR5cGU6IFwidXBkYXRlVmlld3BvcnRcIixcbiAgICAgICAgdmlld3BvcnROb2RlOiB2aWV3cG9ydFJlZi5jdXJyZW50LFxuICAgICAgICBpdGVtQW1vdW50LFxuICAgICAgfSk7XG4gICAgfSk7XG4gICAgYWRqdXN0ZWRGb3JSZWZsb3cuY3VycmVudCA9IHRydWU7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgICByb2xlPVwibGlzdGJveFwiXG4gICAgICAgICAgICBpZD17aWR9XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY2dDIn0= */");export function VirtualScrollList({id,maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,containerStyle,"data-e2e-test-id":dataE2eTestId,onContentHeightChange}){let scrollableContainerRef=useRef(null),viewportRef=useRef(null),adjustedForReflow=useRef(!1),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=useReducer(VirtualScrollReducer,{});useLayoutEffect(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),useEffect(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),useEffect(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),useLayoutEffect(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]),useEffect(()=>{onContentHeightChange?.()},[maxContentHeight,onContentHeightChange]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return maxContentHeight>maxHeight&&!adjustedForReflow.current&&viewportRef.current&&(setTimeout(()=>{dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})}),adjustedForReflow.current=!0),React.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight,...containerStyle},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():React.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},React.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`},role:"listbox",id:id},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|