@amboss/design-system 1.24.2 → 1.24.4

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.
@@ -1 +1 @@
1
- import React,{useState,useRef,useMemo,useEffect,useCallback}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{useAutoPosition}from"../../../shared/useAutoPosition";import{FormField}from"../FormField/FormField";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{Input,InputRaw}from"../Input/Input";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";let StyledContainer=styled("div",{target:"edecq2i0",label:"StyledContainer"})(()=>({position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation.b,\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAoBwB"} */"),StyledInputWrap=styled("div",{target:"edecq2i1",label:"StyledInputWrap"})(()=>({zIndex:0,position:"absolute",pointerEvents:"none",width:"100%",bottom:0,left:0}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation.b,\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAwBwB"} */"),StyledDropdown=styled("div",{target:"edecq2i2",label:"StyledDropdown"})(({theme,dropdownPosition})=>({backgroundColor:theme.values.color.background.primary.default,position:"absolute",fontSize:theme.variables.size.font.s,zIndex:2,width:"100%",margin:`${theme.variables.size.spacing.xs} 0`,cursor:"pointer",borderRadius:theme.variables.size.borderRadius.s,overflow:"hidden",boxShadow:theme.values.elevation.b,boxSizing:"border-box",..."down"===dropdownPosition&&{top:"100%"},..."up"===dropdownPosition&&{bottom:"100%"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation.b,\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAiCuB"} */"),StyledDropdownItem=styled("div",{target:"edecq2i3",label:"StyledDropdownItem"})(({theme,active})=>({padding:theme.variables.size.spacing.xs,borderRadius:theme.variables.size.borderRadius.xs,...active&&{backgroundColor:theme.values.color.background.secondary.hover,borderRadius:0},"&:hover":{backgroundColor:theme.values.color.background.secondary.hover,borderRadius:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation.b,\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAuD2B"} */"),HiddenSelectInput=styled("select",{target:"edecq2i4",label:"HiddenSelectInput"})(()=>({display:"none"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation.b,\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAsE0B"} */"),defaultFilterMethod=(option,value)=>option.label.toLowerCase().indexOf(value.toLowerCase())>-1;export function Select({options=[],name,value,placeholder,emptyStateMessage,hasError,filterMethod=defaultFilterMethod,onChange=()=>null,onBlur=()=>null,onFocus=()=>null,maxHeight=230,autoComplete="on",...rest}){let{disabled}=rest,[isOpen,setIsOpen]=useState(!1),[innerValue,setInnerValue]=useState(""),[preselectedIndex,setPreselectedIndex]=useState(-1),initialValueOption=useMemo(()=>options.find(option=>option.value===value)||{value:"",label:""},[options,value]),[fakeSelectedOption,setFakeSelectedOption]=useState(initialValueOption),fakeSelectRef=useRef(null),dropDownRef=useRef(null),innerInputRef=useRef(null),currentValueRef=useRef(value);currentValueRef.current=value;let forceChangeFakeSelect=useCallback(selectedOption=>{fakeSelectedOption.value!==selectedOption.value&&setFakeSelectedOption(selectedOption)},[fakeSelectedOption]),[verticalPosition]=useAutoPosition(fakeSelectRef,dropDownRef,isOpen),closeDropdown=useCallback((noSelect=!1)=>{if(setIsOpen(!1),!noSelect){let selectedOption=options.find(option=>option.label.toLowerCase().trim()===innerValue.toLowerCase().trim());selectedOption&&forceChangeFakeSelect(selectedOption)}setInnerValue(""),setPreselectedIndex(-1),onBlur()},[onBlur,forceChangeFakeSelect,innerValue,options]);useEffect(()=>{fakeSelectRef.current&&currentValueRef.current!==fakeSelectedOption.value&&fakeSelectRef.current.dispatchEvent(new Event("change",{bubbles:!0}))},[fakeSelectedOption,fakeSelectRef,currentValueRef]),useEffect(()=>{forceChangeFakeSelect(initialValueOption)},[initialValueOption]);let filteredOptions=useMemo(()=>innerValue?options.filter(option=>filterMethod(option,innerValue)):options,[options,filterMethod,innerValue]),displayValue=useMemo(()=>options.some(option=>option.value===value)?options.find(option=>option.value===value).label:"",[value,options]);return useEffect(()=>{setPreselectedIndex(-1)},[filteredOptions]),useKeyboard({Escape:()=>closeDropdown(!0),Enter:()=>{if(filteredOptions[preselectedIndex]){forceChangeFakeSelect(filteredOptions[preselectedIndex]),closeDropdown(!0);return}closeDropdown()},ArrowUp:()=>{setPreselectedIndex(Math.max(preselectedIndex-1,0))},ArrowDown:()=>{setPreselectedIndex(Math.min(preselectedIndex+1,filteredOptions.length-1))}},innerInputRef,isOpen&&!disabled),useKeyboard({"ArrowUp ArrowDown":()=>{setIsOpen(!0)}},innerInputRef,!isOpen&&!disabled),React.createElement(FormField,{"data-ds-id":"Select",...rest},React.createElement(StyledContainer,{onBlur:()=>{closeDropdown(!0)}},React.createElement("div",{style:{zIndex:1}},React.createElement(InputRaw,{areaLabel:rest.label,name:`${name}-innerInput`,value:innerValue,privateProps:{isTransparent:!(isOpen&&innerValue)},icon:isOpen?"chevron-up":"chevron-down",hasError:hasError,disabled:disabled,onFocus:()=>{setIsOpen(!0),onFocus()},onClick:()=>{setIsOpen(!0)},onChange:e=>{e.currentTarget.value&&setIsOpen(!0),setInnerValue(e.currentTarget.value)},ref:innerInputRef,autoComplete:autoComplete})),React.createElement(StyledInputWrap,null,React.createElement(Input,{name:name,value:displayValue,onChange:()=>null,icon:isOpen?"chevron-up":"chevron-down",placeholder:placeholder,tabIndex:-1,autoComplete:"off"})),isOpen&&React.createElement(StyledDropdown,{dropdownPosition:verticalPosition,ref:dropDownRef,onMouseDown:e=>e.preventDefault()},React.createElement(VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:filteredOptions.length,emptyState:()=>React.createElement(Box,{space:"xs"},React.createElement(Text,null,emptyStateMessage||"\uD83E\uDD37\uD83C\uDFFB‍♀️")),itemInView:preselectedIndex,itemTemplate:index=>{let option=filteredOptions[index];return React.createElement(StyledDropdownItem,{key:option.value,active:preselectedIndex===index||value===option.value,onMouseDown:()=>{forceChangeFakeSelect(option),closeDropdown(!0)}},React.createElement(Text,{weight:"bold",color:"primary",size:"s"},option.label))}})),React.createElement(HiddenSelectInput,{onChange:onChange,value:fakeSelectedOption.value,ref:fakeSelectRef,autoComplete:"off"},React.createElement("option",{value:fakeSelectedOption.value},fakeSelectedOption.label))))}
1
+ import React,{useState,useRef,useMemo,useEffect,useCallback}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{useAutoPosition}from"../../../shared/useAutoPosition";import{FormField}from"../FormField/FormField";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{Input,InputRaw}from"../Input/Input";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";let StyledContainer=styled("div",{target:"ekok6ar0",label:"StyledContainer"})(()=>({position:"relative"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation[2],\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAoBwB"} */"),StyledInputWrap=styled("div",{target:"ekok6ar1",label:"StyledInputWrap"})(()=>({zIndex:0,position:"absolute",pointerEvents:"none",width:"100%",bottom:0,left:0}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation[2],\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAwBwB"} */"),StyledDropdown=styled("div",{target:"ekok6ar2",label:"StyledDropdown"})(({theme,dropdownPosition})=>({backgroundColor:theme.values.color.background.primary.default,position:"absolute",fontSize:theme.variables.size.font.s,zIndex:2,width:"100%",margin:`${theme.variables.size.spacing.xs} 0`,cursor:"pointer",borderRadius:theme.variables.size.borderRadius.s,overflow:"hidden",boxShadow:theme.values.elevation[2],boxSizing:"border-box",..."down"===dropdownPosition&&{top:"100%"},..."up"===dropdownPosition&&{bottom:"100%"}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation[2],\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAiCuB"} */"),StyledDropdownItem=styled("div",{target:"ekok6ar3",label:"StyledDropdownItem"})(({theme,active})=>({padding:theme.variables.size.spacing.xs,borderRadius:theme.variables.size.borderRadius.xs,...active&&{backgroundColor:theme.values.color.background.secondary.hover,borderRadius:0},"&:hover":{backgroundColor:theme.values.color.background.secondary.hover,borderRadius:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation[2],\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAuD2B"} */"),HiddenSelectInput=styled("select",{target:"ekok6ar4",label:"HiddenSelectInput"})(()=>({display:"none"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/Select.tsx","sources":["src/components/Form/Select/Select.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useState,\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport type { VerticalPosition } from \"../../../shared/useAutoPosition\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport { FormField } from \"../FormField/FormField\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { Input, InputRaw } from \"../Input/Input\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\n\nconst StyledContainer = styled.div(() => ({\n  position: \"relative\",\n}));\n\nconst StyledInputWrap = styled.div(() => ({\n  zIndex: 0,\n  position: \"absolute\",\n  pointerEvents: \"none\",\n  width: \"100%\",\n  bottom: 0,\n  left: 0,\n}));\n\nconst StyledDropdown = styled.div<{ dropdownPosition: VerticalPosition }>(\n  ({ theme, dropdownPosition }) => ({\n    backgroundColor: theme.values.color.background.primary.default,\n    position: \"absolute\",\n    fontSize: theme.variables.size.font.s,\n    zIndex: 2,\n    width: \"100%\",\n    margin: `${theme.variables.size.spacing.xs} 0`,\n    cursor: \"pointer\",\n    borderRadius: theme.variables.size.borderRadius.s,\n    overflow: \"hidden\",\n    boxShadow: theme.values.elevation[2],\n    boxSizing: \"border-box\",\n    ...(dropdownPosition === \"down\" && {\n      top: \"100%\",\n    }),\n    ...(dropdownPosition === \"up\" && {\n      bottom: \"100%\",\n    }),\n  })\n);\n\nconst StyledDropdownItem = styled.div<{ active: boolean }>(\n  ({ theme, active }) => ({\n    padding: theme.variables.size.spacing.xs,\n    borderRadius: theme.variables.size.borderRadius.xs,\n    ...(active && {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    }),\n    \"&:hover\": {\n      backgroundColor: theme.values.color.background.secondary.hover,\n      borderRadius: 0,\n    },\n  })\n);\n\nconst HiddenSelectInput = styled.select(() => ({\n  display: \"none\",\n}));\n\ntype SelectOption = {\n  value: string;\n  label: string;\n} & Record<string, unknown>;\n\nexport type SelectProps = {\n  name: string;\n  value: string;\n  placeholder?: string;\n  emptyStateMessage?: string;\n  hasError?: boolean;\n  disabled?: boolean;\n  options?: SelectOption[];\n  filterMethod?: (options: SelectOption, value: string) => boolean;\n  onChange?: (e: React.FormEvent<HTMLSelectElement>) => void;\n  onBlur?: () => void;\n  onFocus?: () => void;\n  maxHeight?: number;\n  autoComplete?: string;\n} & FormFieldProps;\n\nconst defaultFilterMethod = (option: SelectOption, value: string): boolean =>\n  option.label.toLowerCase().indexOf(value.toLowerCase()) > -1;\n\nexport function Select({\n  options = [],\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod = defaultFilterMethod,\n  onChange = () => null,\n  onBlur = () => null,\n  onFocus = () => null,\n  maxHeight = 230,\n  autoComplete = \"on\",\n  ...rest\n}: SelectProps): React.ReactElement {\n  const { disabled } = rest;\n  const [isOpen, setIsOpen] = useState(false);\n  const [innerValue, setInnerValue] = useState(\"\");\n  const [preselectedIndex, setPreselectedIndex] = useState(-1);\n  const initialValueOption = useMemo(\n    () =>\n      options.find((option) => option.value === value) || {\n        value: \"\",\n        label: \"\",\n      },\n    [options, value]\n  );\n  const [fakeSelectedOption, setFakeSelectedOption] =\n    useState(initialValueOption);\n  const fakeSelectRef = useRef(null);\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const currentValueRef = useRef(value);\n  currentValueRef.current = value;\n\n  const forceChangeFakeSelect = useCallback(\n    (selectedOption: SelectOption) => {\n      if (fakeSelectedOption.value !== selectedOption.value) {\n        setFakeSelectedOption(selectedOption);\n      }\n    },\n    [fakeSelectedOption]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    fakeSelectRef,\n    dropDownRef,\n    isOpen\n  );\n\n  const closeDropdown = useCallback(\n    (noSelect = false) => {\n      setIsOpen(false);\n      if (!noSelect) {\n        const selectedOption = options.find(\n          (option) =>\n            option.label.toLowerCase().trim() ===\n            innerValue.toLowerCase().trim()\n        );\n        if (selectedOption) {\n          forceChangeFakeSelect(selectedOption);\n        }\n      }\n      setInnerValue(\"\");\n      setPreselectedIndex(-1);\n      onBlur();\n    },\n    [onBlur, forceChangeFakeSelect, innerValue, options]\n  );\n\n  useEffect(() => {\n    if (\n      fakeSelectRef.current &&\n      currentValueRef.current !== fakeSelectedOption.value\n    ) {\n      fakeSelectRef.current.dispatchEvent(\n        new Event(\"change\", { bubbles: true })\n      );\n    }\n  }, [fakeSelectedOption, fakeSelectRef, currentValueRef]);\n\n  // force updating the hidden select if a new initial value gets passed down\n  // without it, there would be a mismatch of selected value and the hidden select,\n  // which would cause the first value to not be selectable\n  useEffect(() => {\n    forceChangeFakeSelect(initialValueOption);\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [initialValueOption]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  const displayValue = useMemo(\n    () =>\n      options.some((option) => option.value === value)\n        ? options.find((option) => option.value === value).label\n        : \"\",\n    [value, options]\n  );\n\n  useEffect(() => {\n    setPreselectedIndex(-1);\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(true),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n          closeDropdown(true);\n          return;\n        }\n        closeDropdown();\n      },\n      ArrowUp: () => {\n        setPreselectedIndex(Math.max(preselectedIndex - 1, 0));\n      },\n      ArrowDown: () => {\n        setPreselectedIndex(\n          Math.min(preselectedIndex + 1, filteredOptions.length - 1)\n        );\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        setIsOpen(true);\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <FormField data-ds-id=\"Select\" {...(rest as FormFieldProps)}>\n      <StyledContainer\n        onBlur={() => {\n          closeDropdown(true);\n        }}\n      >\n        <div style={{ zIndex: 1 }}>\n          <InputRaw\n            areaLabel={rest.label}\n            name={`${name}-innerInput`}\n            value={innerValue}\n            privateProps={{ isTransparent: !(isOpen && innerValue) }}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            hasError={hasError}\n            disabled={disabled}\n            onFocus={() => {\n              setIsOpen(true);\n              onFocus();\n            }}\n            onClick={() => {\n              setIsOpen(true);\n            }}\n            onChange={(e) => {\n              if (e.currentTarget.value) {\n                setIsOpen(true);\n              }\n              setInnerValue(e.currentTarget.value);\n            }}\n            ref={innerInputRef}\n            autoComplete={autoComplete}\n          />\n        </div>\n\n        <StyledInputWrap>\n          <Input\n            name={name}\n            value={displayValue}\n            onChange={() => null}\n            icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n            placeholder={placeholder}\n            tabIndex={-1}\n            autoComplete=\"off\"\n          />\n        </StyledInputWrap>\n\n        {isOpen && (\n          // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n          <StyledDropdown\n            dropdownPosition={verticalPosition}\n            ref={dropDownRef}\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          >\n            <VirtualScrollList\n              maxHeight={maxHeight}\n              itemHeight={36}\n              itemAmount={filteredOptions.length}\n              emptyState={() => (\n                <Box space=\"xs\">\n                  <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n                </Box>\n              )}\n              itemInView={preselectedIndex}\n              itemTemplate={(index: number) => {\n                const option: SelectOption = filteredOptions[index];\n                return (\n                  // eslint-disable-next-line jsx-a11y/no-static-element-interactions\n                  <StyledDropdownItem\n                    key={option.value}\n                    active={\n                      preselectedIndex === index || value === option.value\n                    }\n                    onMouseDown={() => {\n                      forceChangeFakeSelect(option);\n                      closeDropdown(true);\n                    }}\n                  >\n                    <Text weight=\"bold\" color=\"primary\" size=\"s\">\n                      {option.label}\n                    </Text>\n                  </StyledDropdownItem>\n                );\n              }}\n            />\n          </StyledDropdown>\n        )}\n\n        <HiddenSelectInput\n          onChange={onChange}\n          value={fakeSelectedOption.value}\n          ref={fakeSelectRef}\n          autoComplete=\"off\"\n        >\n          <option value={fakeSelectedOption.value}>\n            {fakeSelectedOption.label}\n          </option>\n        </HiddenSelectInput>\n      </StyledContainer>\n    </FormField>\n  );\n}\n"],"names":[],"mappings":"AAsE0B"} */"),defaultFilterMethod=(option,value)=>option.label.toLowerCase().indexOf(value.toLowerCase())>-1;export function Select({options=[],name,value,placeholder,emptyStateMessage,hasError,filterMethod=defaultFilterMethod,onChange=()=>null,onBlur=()=>null,onFocus=()=>null,maxHeight=230,autoComplete="on",...rest}){let{disabled}=rest,[isOpen,setIsOpen]=useState(!1),[innerValue,setInnerValue]=useState(""),[preselectedIndex,setPreselectedIndex]=useState(-1),initialValueOption=useMemo(()=>options.find(option=>option.value===value)||{value:"",label:""},[options,value]),[fakeSelectedOption,setFakeSelectedOption]=useState(initialValueOption),fakeSelectRef=useRef(null),dropDownRef=useRef(null),innerInputRef=useRef(null),currentValueRef=useRef(value);currentValueRef.current=value;let forceChangeFakeSelect=useCallback(selectedOption=>{fakeSelectedOption.value!==selectedOption.value&&setFakeSelectedOption(selectedOption)},[fakeSelectedOption]),[verticalPosition]=useAutoPosition(fakeSelectRef,dropDownRef,isOpen),closeDropdown=useCallback((noSelect=!1)=>{if(setIsOpen(!1),!noSelect){let selectedOption=options.find(option=>option.label.toLowerCase().trim()===innerValue.toLowerCase().trim());selectedOption&&forceChangeFakeSelect(selectedOption)}setInnerValue(""),setPreselectedIndex(-1),onBlur()},[onBlur,forceChangeFakeSelect,innerValue,options]);useEffect(()=>{fakeSelectRef.current&&currentValueRef.current!==fakeSelectedOption.value&&fakeSelectRef.current.dispatchEvent(new Event("change",{bubbles:!0}))},[fakeSelectedOption,fakeSelectRef,currentValueRef]),useEffect(()=>{forceChangeFakeSelect(initialValueOption)},[initialValueOption]);let filteredOptions=useMemo(()=>innerValue?options.filter(option=>filterMethod(option,innerValue)):options,[options,filterMethod,innerValue]),displayValue=useMemo(()=>options.some(option=>option.value===value)?options.find(option=>option.value===value).label:"",[value,options]);return useEffect(()=>{setPreselectedIndex(-1)},[filteredOptions]),useKeyboard({Escape:()=>closeDropdown(!0),Enter:()=>{if(filteredOptions[preselectedIndex]){forceChangeFakeSelect(filteredOptions[preselectedIndex]),closeDropdown(!0);return}closeDropdown()},ArrowUp:()=>{setPreselectedIndex(Math.max(preselectedIndex-1,0))},ArrowDown:()=>{setPreselectedIndex(Math.min(preselectedIndex+1,filteredOptions.length-1))}},innerInputRef,isOpen&&!disabled),useKeyboard({"ArrowUp ArrowDown":()=>{setIsOpen(!0)}},innerInputRef,!isOpen&&!disabled),React.createElement(FormField,{"data-ds-id":"Select",...rest},React.createElement(StyledContainer,{onBlur:()=>{closeDropdown(!0)}},React.createElement("div",{style:{zIndex:1}},React.createElement(InputRaw,{areaLabel:rest.label,name:`${name}-innerInput`,value:innerValue,privateProps:{isTransparent:!(isOpen&&innerValue)},icon:isOpen?"chevron-up":"chevron-down",hasError:hasError,disabled:disabled,onFocus:()=>{setIsOpen(!0),onFocus()},onClick:()=>{setIsOpen(!0)},onChange:e=>{e.currentTarget.value&&setIsOpen(!0),setInnerValue(e.currentTarget.value)},ref:innerInputRef,autoComplete:autoComplete})),React.createElement(StyledInputWrap,null,React.createElement(Input,{name:name,value:displayValue,onChange:()=>null,icon:isOpen?"chevron-up":"chevron-down",placeholder:placeholder,tabIndex:-1,autoComplete:"off"})),isOpen&&React.createElement(StyledDropdown,{dropdownPosition:verticalPosition,ref:dropDownRef,onMouseDown:e=>e.preventDefault()},React.createElement(VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:filteredOptions.length,emptyState:()=>React.createElement(Box,{space:"xs"},React.createElement(Text,null,emptyStateMessage||"\uD83E\uDD37\uD83C\uDFFB‍♀️")),itemInView:preselectedIndex,itemTemplate:index=>{let option=filteredOptions[index];return React.createElement(StyledDropdownItem,{key:option.value,active:preselectedIndex===index||value===option.value,onMouseDown:()=>{forceChangeFakeSelect(option),closeDropdown(!0)}},React.createElement(Text,{weight:"bold",color:"primary",size:"s"},option.label))}})),React.createElement(HiddenSelectInput,{onChange:onChange,value:fakeSelectedOption.value,ref:fakeSelectRef,autoComplete:"off"},React.createElement("option",{value:fakeSelectedOption.value},fakeSelectedOption.label))))}