@commercetools-uikit/selectable-search-input 19.4.0 → 19.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -92,6 +92,7 @@ export default Example;
92
92
  | `showSubmitButton` | `boolean` | | `true` | Show submit button in the input |
93
93
  | `selectDataProps` | `Record` | | | used to pass data-\* props to the select component.
eg: selectDataProps={\[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]} |
94
94
  | `inputDataProps` | `Record` | | | used to pass data-\* props to the input element.
eg: inputDataProps={\[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]} |
95
+ | `selectCustomComponents` | `ReactSelectProps['components']` | | | Map of components to overwrite the default ones, see what components you can override&#xA;<br/>&#xA;[Props from React select was used](https://react-select.com/props) |
95
96
 
96
97
  ## Signatures
97
98
 
@@ -230,13 +230,13 @@ const SelectableSelect = props => {
230
230
  isSearchable: props.isReadOnly ? false : props.isMenuSearchable,
231
231
  maxMenuHeight: props.maxMenuHeight,
232
232
  closeMenuOnSelect: props.closeMenuOnSelect,
233
- components: {
233
+ components: _objectSpread$1({
234
234
  SingleValue: innerProps => jsxRuntime.jsx(SingleValue, _objectSpread$1({
235
235
  id: props.id,
236
236
  dataProps: props.dataProps
237
237
  }, innerProps)),
238
238
  DropdownIndicator: selectUtils.DropdownIndicator
239
- },
239
+ }, props.selectCustomComponents),
240
240
  options: props.options,
241
241
  menuIsOpen: props.isReadOnly ? false : undefined,
242
242
  placeholder: "",
@@ -280,7 +280,7 @@ const Container = /*#__PURE__*/_styled__default["default"]("div", process.env.NO
280
280
  } : {
281
281
  name: "zjik7",
282
282
  styles: "display:flex",
283
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+B4B","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
283
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+B4B","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
284
284
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
285
285
  });
286
286
  const getTextInputName = name => name ? "".concat(name, ".textInput") : undefined;
@@ -467,10 +467,11 @@ const SelectableSearchInput = props => {
467
467
  handleDropdownChange: handleDropdownChange,
468
468
  textInputRef: textInputRef,
469
469
  selectedOption: selectedOption,
470
- dataProps: transformedSelectDataProps
470
+ dataProps: transformedSelectDataProps,
471
+ selectCustomComponents: props.selectCustomComponents
471
472
  }))
472
473
  }), jsxRuntime.jsxs("div", {
473
- css: [getSelectableSearchInputContainerStyles(props), dropdownHasFocus && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";&:hover{border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAwciB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAocU","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
474
+ css: [getSelectableSearchInputContainerStyles(props), dropdownHasFocus && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";&:hover{border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+ciB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA2cU","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
474
475
  children: [jsxRuntime.jsx("input", _objectSpread(_objectSpread(_objectSpread({
475
476
  ref: textInputRef,
476
477
  id: SelectableSearchInput.getTextInputId(selectablSearchInputId),
@@ -568,7 +569,7 @@ SelectableSearchInput.getDropdownId = getDropdownName;
568
569
  var SelectableSearchInput$1 = SelectableSearchInput;
569
570
 
570
571
  // NOTE: This string will be replaced on build time with the package version.
571
- var version = "19.4.0";
572
+ var version = "19.5.0";
572
573
 
573
574
  exports["default"] = SelectableSearchInput$1;
574
575
  exports.version = version;
@@ -225,13 +225,13 @@ const SelectableSelect = props => {
225
225
  isSearchable: props.isReadOnly ? false : props.isMenuSearchable,
226
226
  maxMenuHeight: props.maxMenuHeight,
227
227
  closeMenuOnSelect: props.closeMenuOnSelect,
228
- components: {
228
+ components: _objectSpread$1({
229
229
  SingleValue: innerProps => jsxRuntime.jsx(SingleValue, _objectSpread$1({
230
230
  id: props.id,
231
231
  dataProps: props.dataProps
232
232
  }, innerProps)),
233
233
  DropdownIndicator: selectUtils.DropdownIndicator
234
- },
234
+ }, props.selectCustomComponents),
235
235
  options: props.options,
236
236
  menuIsOpen: props.isReadOnly ? false : undefined,
237
237
  placeholder: "",
@@ -443,7 +443,8 @@ const SelectableSearchInput = props => {
443
443
  handleDropdownChange: handleDropdownChange,
444
444
  textInputRef: textInputRef,
445
445
  selectedOption: selectedOption,
446
- dataProps: transformedSelectDataProps
446
+ dataProps: transformedSelectDataProps,
447
+ selectCustomComponents: props.selectCustomComponents
447
448
  }))
448
449
  }), jsxRuntime.jsxs("div", {
449
450
  css: [getSelectableSearchInputContainerStyles(props), dropdownHasFocus && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";&:hover{border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";}" + ("" ), "" ), "" , "" ],
@@ -504,7 +505,7 @@ SelectableSearchInput.getDropdownId = getDropdownName;
504
505
  var SelectableSearchInput$1 = SelectableSearchInput;
505
506
 
506
507
  // NOTE: This string will be replaced on build time with the package version.
507
- var version = "19.4.0";
508
+ var version = "19.5.0";
508
509
 
509
510
  exports["default"] = SelectableSearchInput$1;
510
511
  exports.version = version;
@@ -204,13 +204,13 @@ const SelectableSelect = props => {
204
204
  isSearchable: props.isReadOnly ? false : props.isMenuSearchable,
205
205
  maxMenuHeight: props.maxMenuHeight,
206
206
  closeMenuOnSelect: props.closeMenuOnSelect,
207
- components: {
207
+ components: _objectSpread$1({
208
208
  SingleValue: innerProps => jsx(SingleValue, _objectSpread$1({
209
209
  id: props.id,
210
210
  dataProps: props.dataProps
211
211
  }, innerProps)),
212
212
  DropdownIndicator
213
- },
213
+ }, props.selectCustomComponents),
214
214
  options: props.options,
215
215
  menuIsOpen: props.isReadOnly ? false : undefined,
216
216
  placeholder: "",
@@ -254,7 +254,7 @@ const Container = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "producti
254
254
  } : {
255
255
  name: "zjik7",
256
256
  styles: "display:flex",
257
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+B4B","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
257
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+B4B","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
258
258
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
259
259
  });
260
260
  const getTextInputName = name => name ? "".concat(name, ".textInput") : undefined;
@@ -441,10 +441,11 @@ const SelectableSearchInput = props => {
441
441
  handleDropdownChange: handleDropdownChange,
442
442
  textInputRef: textInputRef,
443
443
  selectedOption: selectedOption,
444
- dataProps: transformedSelectDataProps
444
+ dataProps: transformedSelectDataProps,
445
+ selectCustomComponents: props.selectCustomComponents
445
446
  }))
446
447
  }), jsxs("div", {
447
- css: [getSelectableSearchInputContainerStyles(props), dropdownHasFocus && !props.isReadOnly && /*#__PURE__*/css("border-left-color:", designTokens.borderColorForInputWhenFocused, ";&:hover{border-left-color:", designTokens.borderColorForInputWhenFocused, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAwciB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAocU","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
448
+ css: [getSelectableSearchInputContainerStyles(props), dropdownHasFocus && !props.isReadOnly && /*#__PURE__*/css("border-left-color:", designTokens.borderColorForInputWhenFocused, ";&:hover{border-left-color:", designTokens.borderColorForInputWhenFocused, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA+ciB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:SelectableSearchInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AA2cU","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  useState,\n  useCallback,\n  useRef,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { SearchIcon, CloseIcon } from '@commercetools-uikit/icons';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { warnIfMenuPortalPropsAreMissing } from '@commercetools-uikit/select-utils';\nimport {\n  getClearIconButtonStyles,\n  getSearchIconButtonStyles,\n  getSelectableSearchInputContainerStyles,\n  getSelectableSearchInputStyles,\n} from './selectable-search-input.styles';\nimport SelectableSelect from './selectable-select';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport { css } from '@emotion/react';\nimport { type Props as ReactSelectProps } from 'react-select';\n\nconst Container = styled.div`\n  display: flex;\n`;\n\nconst getTextInputName = (name?: string) =>\n  name ? `${name}.textInput` : undefined;\n\nconst getDropdownName = (name?: string) =>\n  name ? `${name}.dropdown` : undefined;\n\nexport type TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\nexport type TValue = {\n  text: string;\n  option: string;\n};\n\nexport type TOption = {\n  value: string;\n  label?: ReactNode;\n};\n\nexport type TOptionObject = {\n  options: TOption[];\n};\n\nexport type TOptions = TOption[] | TOptionObject[];\n\nexport type TSelectableSearchInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML autocomplete property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * Used as HTML name of the input component property.\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of text input and selected option.\n   */\n  value: TValue;\n  _experimentalValue?: TValue;\n  /**\n   * Called with the event of the input or dropdown when either the selectable dropdown or the text input have changed.\n   * The change event from the text input has a suffix of `.textInput` and the change event from the dropdown has a suffix of `.dropdown`.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Handler when the search button is clicked.\n   */\n  onSubmit: (value: TValue) => void;\n  /**\n   * Handler when the clear button is clicked.\n   */\n  onReset?: () => void;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates if the input has invalid values\n   */\n  hasError?: boolean;\n  /**\n   * Indicates if the input has warning values\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Indicates if the input should be cleared when the clear button is clicked.\n   * Defaults to true.\n   *\n   */\n  isClearable?: boolean;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?: 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';\n  /**\n   * Array of options that populate the select menu\n   */\n  options: TOptions;\n  /**\n   * z-index value for the menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * Dom element to portal the select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Handle change events on the menu input\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  onMenuInputChange?: ReactSelectProps['onInputChange'];\n  /**\n   * Can be used to render a custom value when there are no options (either because of no search results, or all options have been used, or there were none in the first place). Gets called with { inputValue: String }.\n   * <br />\n   * `inputValue` will be an empty string when no search text is present.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  noMenuOptionsMessage?: ReactSelectProps['noOptionsMessage'];\n  /**\n   * Whether to enable search functionality.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  isMenuSearchable?: ReactSelectProps['isSearchable'];\n  /**\n   * Maximum height of the menu before scrolling\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  maxMenuHeight?: ReactSelectProps['maxMenuHeight'];\n  /**\n   * Whether the menu should close after a value is selected. Defaults to `true`.\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  closeMenuOnSelect?: ReactSelectProps['closeMenuOnSelect'];\n  /**\n   * Horizontal size limit for the dropdown menu.\n   */\n  menuHorizontalConstraint?: 3 | 4 | 5;\n  /**\n   * Show submit button in the input\n   */\n  showSubmitButton?: boolean;\n  /**\n   *  used to pass data-* props to the select component.\n   * eg: selectDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  selectDataProps?: Record<string, string>;\n  /**\n   *  used to pass data-* props to the input element.\n   * eg: inputDataProps={[{ 'prop-1': 'value-1' }, { 'prop-2': 'value-2' }]}\n   */\n  inputDataProps?: Record<string, string>;\n  /**\n   * Map of components to overwrite the default ones, see what components you can override\n   * <br/>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  selectCustomComponents?: ReactSelectProps['components'];\n};\n\nconst defaultProps: Pick<\n  TSelectableSearchInputProps,\n  | 'horizontalConstraint'\n  | 'isClearable'\n  | 'menuHorizontalConstraint'\n  | 'showSubmitButton'\n  | 'menuPortalZIndex'\n  | 'options'\n> = {\n  horizontalConstraint: 'scale',\n  isClearable: true,\n  menuHorizontalConstraint: 3,\n  showSubmitButton: true,\n  menuPortalZIndex: 1,\n  options: [],\n};\n\nconst selectableSearchInputSequentialId = createSequentialId(\n  'selectable-search-input-'\n);\n\nconst isOptionObject = (\n  option: TOption | TOptionObject\n): option is TOptionObject => (option as TOptionObject).options !== undefined;\n\nconst transformDataProps = (dataProps?: Record<string, string>) =>\n  Object.fromEntries(\n    Object.entries(dataProps || {}).map(([key, value]) => [\n      `data-${key}`,\n      value,\n    ])\n  );\n\nconst SelectableSearchInput = (props: TSelectableSearchInputProps) => {\n  const [dropdownHasFocus, toggleDropdownHasFocus] = useToggleState(false);\n  const [searchValue, setSearchValue] = useState(props.value.text || '');\n  const [searchOption, setSearchOption] = useState(props.value.option || '');\n  const containerRef = useRef<HTMLDivElement>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const legacyDataProps = filterDataAttributes(props);\n  const transformedSelectDataProps = transformDataProps(props.selectDataProps);\n  const transformedInputDataProps = transformDataProps(props.inputDataProps);\n  const searchInputValue = props._experimentalValue?.text ?? searchValue;\n  const searchInputOption = props._experimentalValue?.option ?? searchOption;\n\n  const optionsWithoutGroups = props.options.flatMap((option) => {\n    if (isOptionObject(option)) {\n      return option.options;\n    }\n    return option;\n  });\n\n  const selectedOption = optionsWithoutGroups.find(\n    (option) => option.value === searchInputOption\n  );\n\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: props.menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'SelectableSearchInput',\n  });\n\n  const { onFocus, onBlur, name } = props;\n  const handleTextInputFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onFocus, selectablSearchInputId, name]);\n\n  const handleTextInputBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(name),\n        },\n      });\n    }\n  }, [onBlur, selectablSearchInputId, name]);\n\n  const handleClear = () => {\n    setSearchValue('');\n    if (props.onReset) {\n      props.onReset();\n    }\n  };\n\n  const handleTextInputChange = (event: ChangeEvent<HTMLInputElement>) => {\n    setSearchValue(event.target.value);\n    if (props.onChange) {\n      props.onChange({\n        target: {\n          id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n          name: getTextInputName(props.name),\n          value: event.target.value,\n        },\n      });\n    }\n  };\n\n  const handleSubmit = (\n    event:\n      | KeyboardEvent<HTMLButtonElement>\n      | MouseEvent<HTMLButtonElement>\n      | KeyboardEvent<HTMLInputElement>\n  ) => {\n    event.preventDefault();\n    if (props.onSubmit) {\n      props.onSubmit({\n        text: searchInputValue,\n        option: selectedOption?.value ?? '',\n      });\n    }\n  };\n\n  const dropdownName = getDropdownName(props.name);\n  const dropdownId = SelectableSearchInput.getDropdownId(\n    selectablSearchInputId\n  );\n\n  const handleDropdownFocus = useCallback(() => {\n    if (onFocus) {\n      onFocus({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(true);\n  }, [onFocus, toggleDropdownHasFocus, dropdownName, dropdownId]);\n\n  const handleDropdownBlur = useCallback(() => {\n    if (onBlur) {\n      onBlur({\n        target: {\n          id: dropdownId,\n          name: dropdownName,\n        },\n      });\n    }\n    toggleDropdownHasFocus(false);\n  }, [toggleDropdownHasFocus, onBlur, dropdownName, dropdownId]);\n\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n          },\n        });\n        onBlur({\n          target: {\n            id: SelectableSearchInput.getTextInputId(selectablSearchInputId),\n            name: getTextInputName(name),\n          },\n        });\n      }\n    },\n    [onBlur, selectablSearchInputId, name]\n  );\n\n  const handleDropdownChange = useCallback(\n    (nextSelectedOptions) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (props.onChange) {\n        props.onChange({\n          target: {\n            id: SelectableSearchInput.getDropdownId(selectablSearchInputId),\n            name: getDropdownName(name),\n            value: nextSelectedOptions.value,\n          },\n        });\n      }\n      textInputRef.current?.focus();\n    },\n    [props.onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={props.menuHorizontalConstraint}>\n          <SelectableSelect\n            {...props}\n            id={SelectableSearchInput.getDropdownId(selectablSearchInputId)}\n            name={getDropdownName(props.name)}\n            dropdownHasFocus={dropdownHasFocus}\n            isCondensed={props.isCondensed ?? false}\n            handleDropdownFocus={handleDropdownFocus}\n            handleDropdownBlur={handleDropdownBlur}\n            handleDropdownChange={handleDropdownChange}\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(props),\n            dropdownHasFocus &&\n              !props.isReadOnly &&\n              css`\n                border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                &:hover {\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                }\n              `,\n          ]}\n        >\n          <input\n            ref={textInputRef}\n            id={SelectableSearchInput.getTextInputId(selectablSearchInputId)}\n            name={getTextInputName(props.name)}\n            type=\"text\"\n            value={searchInputValue}\n            onChange={handleTextInputChange}\n            onBlur={handleTextInputBlur}\n            onFocus={handleTextInputFocus}\n            disabled={props.isDisabled}\n            placeholder={props.placeholder}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            autoComplete={props.autoComplete}\n            aria-readonly={props.isReadOnly}\n            contentEditable={!props.isReadOnly}\n            css={getSelectableSearchInputStyles(props)}\n            {...transformedInputDataProps}\n            {...legacyDataProps}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n            data-testid=\"selectable-input\"\n            onKeyDown={(event) => {\n              if (!props.isReadOnly && event.key === 'Enter') {\n                handleSubmit(event);\n              }\n            }}\n          />\n          {props.isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? 'small' : 'medium'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(props)}\n              />\n            )}\n          {props.showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? 'medium' : 'big'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(props)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.defaultProps = defaultProps;\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
448
449
  children: [jsx("input", _objectSpread(_objectSpread(_objectSpread({
449
450
  ref: textInputRef,
450
451
  id: SelectableSearchInput.getTextInputId(selectablSearchInputId),
@@ -542,6 +543,6 @@ SelectableSearchInput.getDropdownId = getDropdownName;
542
543
  var SelectableSearchInput$1 = SelectableSearchInput;
543
544
 
544
545
  // NOTE: This string will be replaced on build time with the package version.
545
- var version = "19.4.0";
546
+ var version = "19.5.0";
546
547
 
547
548
  export { SelectableSearchInput$1 as default, version };
@@ -55,6 +55,7 @@ export type TSelectableSearchInputProps = {
55
55
  showSubmitButton?: boolean;
56
56
  selectDataProps?: Record<string, string>;
57
57
  inputDataProps?: Record<string, string>;
58
+ selectCustomComponents?: ReactSelectProps['components'];
58
59
  };
59
60
  declare const SelectableSearchInput: {
60
61
  (props: TSelectableSearchInputProps): import("@emotion/react/jsx-runtime").JSX.Element;
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@commercetools-uikit/selectable-search-input",
3
3
  "description": "A controlled selectable search input component for single-line strings with validation states.",
4
- "version": "19.4.0",
4
+ "version": "19.5.0",
5
5
  "bugs": "https://github.com/commercetools/ui-kit/issues",
6
6
  "repository": {
7
7
  "type": "git",
@@ -21,14 +21,14 @@
21
21
  "dependencies": {
22
22
  "@babel/runtime": "^7.20.13",
23
23
  "@babel/runtime-corejs3": "^7.20.13",
24
- "@commercetools-uikit/constraints": "19.4.0",
25
- "@commercetools-uikit/design-system": "19.4.0",
26
- "@commercetools-uikit/hooks": "19.4.0",
27
- "@commercetools-uikit/icons": "19.4.0",
28
- "@commercetools-uikit/input-utils": "19.4.0",
29
- "@commercetools-uikit/secondary-icon-button": "19.4.0",
30
- "@commercetools-uikit/select-utils": "19.4.0",
31
- "@commercetools-uikit/utils": "19.4.0",
24
+ "@commercetools-uikit/constraints": "19.5.0",
25
+ "@commercetools-uikit/design-system": "19.5.0",
26
+ "@commercetools-uikit/hooks": "19.5.0",
27
+ "@commercetools-uikit/icons": "19.5.0",
28
+ "@commercetools-uikit/input-utils": "19.5.0",
29
+ "@commercetools-uikit/secondary-icon-button": "19.5.0",
30
+ "@commercetools-uikit/select-utils": "19.5.0",
31
+ "@commercetools-uikit/utils": "19.5.0",
32
32
  "@emotion/react": "^11.10.5",
33
33
  "@emotion/styled": "^11.10.5",
34
34
  "lodash": "4.17.21",