@commercetools-uikit/selectable-search-input 19.26.0 → 20.0.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/dist/commercetools-uikit-selectable-search-input.cjs.dev.js +14 -69
- package/dist/commercetools-uikit-selectable-search-input.cjs.prod.js +10 -12
- package/dist/commercetools-uikit-selectable-search-input.esm.js +14 -68
- package/dist/declarations/src/selectable-search-input.d.ts +1 -1
- package/package.json +16 -17
|
@@ -14,7 +14,6 @@ var _defineProperty = require('@babel/runtime-corejs3/helpers/defineProperty');
|
|
|
14
14
|
var _objectWithoutProperties = require('@babel/runtime-corejs3/helpers/objectWithoutProperties');
|
|
15
15
|
var _slicedToArray = require('@babel/runtime-corejs3/helpers/slicedToArray');
|
|
16
16
|
var _styled = require('@emotion/styled/base');
|
|
17
|
-
var _pt = require('prop-types');
|
|
18
17
|
var _Object$fromEntries = require('@babel/runtime-corejs3/core-js-stable/object/from-entries');
|
|
19
18
|
var _mapInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/map');
|
|
20
19
|
var _Object$entries = require('@babel/runtime-corejs3/core-js-stable/object/entries');
|
|
@@ -47,7 +46,6 @@ var _Object$getOwnPropertyDescriptors__default = /*#__PURE__*/_interopDefault(_O
|
|
|
47
46
|
var _Object$defineProperties__default = /*#__PURE__*/_interopDefault(_Object$defineProperties);
|
|
48
47
|
var _Object$defineProperty__default = /*#__PURE__*/_interopDefault(_Object$defineProperty);
|
|
49
48
|
var _styled__default = /*#__PURE__*/_interopDefault(_styled);
|
|
50
|
-
var _pt__default = /*#__PURE__*/_interopDefault(_pt);
|
|
51
49
|
var _Object$fromEntries__default = /*#__PURE__*/_interopDefault(_Object$fromEntries);
|
|
52
50
|
var _mapInstanceProperty__default = /*#__PURE__*/_interopDefault(_mapInstanceProperty);
|
|
53
51
|
var _Object$entries__default = /*#__PURE__*/_interopDefault(_Object$entries);
|
|
@@ -208,11 +206,6 @@ const SingleValue = _ref => {
|
|
|
208
206
|
}))
|
|
209
207
|
}));
|
|
210
208
|
};
|
|
211
|
-
SingleValue.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
212
|
-
children: _pt__default["default"].node,
|
|
213
|
-
dataProps: _pt__default["default"].objectOf(_pt__default["default"].string),
|
|
214
|
-
id: _pt__default["default"].string
|
|
215
|
-
} : {};
|
|
216
209
|
const SelectableSelect = props => {
|
|
217
210
|
const intl = reactIntl.useIntl();
|
|
218
211
|
const dropdownStyles = createSelectableSelectStyles({
|
|
@@ -257,18 +250,9 @@ const SelectableSelect = props => {
|
|
|
257
250
|
})
|
|
258
251
|
});
|
|
259
252
|
};
|
|
260
|
-
SelectableSelect.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
261
|
-
dropdownHasFocus: _pt__default["default"].bool.isRequired,
|
|
262
|
-
isCondensed: _pt__default["default"].bool.isRequired,
|
|
263
|
-
handleDropdownFocus: _pt__default["default"].func.isRequired,
|
|
264
|
-
handleDropdownBlur: _pt__default["default"].func.isRequired,
|
|
265
|
-
textInputRef: _pt__default["default"].any.isRequired,
|
|
266
|
-
selectedOption: _pt__default["default"].any,
|
|
267
|
-
dataProps: _pt__default["default"].objectOf(_pt__default["default"].string)
|
|
268
|
-
} : {};
|
|
269
253
|
var SelectableSelect$1 = SelectableSelect;
|
|
270
254
|
|
|
271
|
-
const _excluded = ["horizontalConstraint", "isClearable", "menuHorizontalConstraint", "showSubmitButton", "menuPortalZIndex"];
|
|
255
|
+
const _excluded = ["horizontalConstraint", "isClearable", "menuHorizontalConstraint", "showSubmitButton", "menuPortalZIndex", "onChange"];
|
|
272
256
|
function ownKeys(e, r) { var t = _Object$keys__default["default"](e); if (_Object$getOwnPropertySymbols__default["default"]) { var o = _Object$getOwnPropertySymbols__default["default"](e); r && (o = _filterInstanceProperty__default["default"](o).call(o, function (r) { return _Object$getOwnPropertyDescriptor__default["default"](e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
273
257
|
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var _context4, _context5; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty__default["default"](_context4 = ownKeys(Object(t), !0)).call(_context4, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](e, _Object$getOwnPropertyDescriptors__default["default"](t)) : _forEachInstanceProperty__default["default"](_context5 = ownKeys(Object(t))).call(_context5, function (r) { _Object$defineProperty__default["default"](e, r, _Object$getOwnPropertyDescriptor__default["default"](t, r)); }); } return e; }
|
|
274
258
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
@@ -282,7 +266,7 @@ const Container = /*#__PURE__*/_styled__default["default"]("div", process.env.NO
|
|
|
282
266
|
styles: "display:flex"
|
|
283
267
|
} : {
|
|
284
268
|
name: "zjik7",
|
|
285
|
-
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAmC4B","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  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  ...props\n}: 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 allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\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: 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={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
|
|
269
|
+
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["selectable-search-input.tsx"],"names":[],"mappings":"AAoC4B","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  type FocusEvent,\n  useState,\n  useCallback,\n  useRef,\n  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  onChange,\n  ...props\n}: 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>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    onChange,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\n    );\n  }\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: 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 (onChange) {\n      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: FocusEvent) => {\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: { value: string } & Record<string, unknown>) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (onChange) {\n        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    [onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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={\n              handleDropdownChange as ReactSelectProps['onChange']\n            }\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */",
|
|
286
270
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
287
271
|
});
|
|
288
272
|
const getTextInputName = name => name ? `${name}.textInput` : undefined;
|
|
@@ -310,6 +294,7 @@ const SelectableSearchInput = _ref3 => {
|
|
|
310
294
|
showSubmitButton = _ref3$showSubmitButto === void 0 ? true : _ref3$showSubmitButto,
|
|
311
295
|
_ref3$menuPortalZInde = _ref3.menuPortalZIndex,
|
|
312
296
|
menuPortalZIndex = _ref3$menuPortalZInde === void 0 ? 1 : _ref3$menuPortalZInde,
|
|
297
|
+
onChange = _ref3.onChange,
|
|
313
298
|
props = _objectWithoutProperties(_ref3, _excluded);
|
|
314
299
|
const _useToggleState = hooks.useToggleState(false),
|
|
315
300
|
_useToggleState2 = _slicedToArray(_useToggleState, 2),
|
|
@@ -330,7 +315,8 @@ const SelectableSearchInput = _ref3 => {
|
|
|
330
315
|
isClearable,
|
|
331
316
|
menuHorizontalConstraint,
|
|
332
317
|
showSubmitButton,
|
|
333
|
-
menuPortalZIndex
|
|
318
|
+
menuPortalZIndex,
|
|
319
|
+
onChange
|
|
334
320
|
}, props);
|
|
335
321
|
const legacyDataProps = utils.filterDataAttributes(props);
|
|
336
322
|
const transformedSelectDataProps = transformDataProps(props.selectDataProps);
|
|
@@ -349,7 +335,7 @@ const SelectableSearchInput = _ref3 => {
|
|
|
349
335
|
}
|
|
350
336
|
const selectablSearchInputId = hooks.useFieldId(props.id, selectableSearchInputSequentialId);
|
|
351
337
|
if (!props.isReadOnly) {
|
|
352
|
-
process.env.NODE_ENV !== "production" ? utils.warning(typeof
|
|
338
|
+
process.env.NODE_ENV !== "production" ? utils.warning(typeof onChange === 'function', 'SelectableSearchInput: `onChange` is required when is not read only.') : void 0;
|
|
353
339
|
}
|
|
354
340
|
selectUtils.warnIfMenuPortalPropsAreMissing({
|
|
355
341
|
menuPortalZIndex: menuPortalZIndex,
|
|
@@ -387,8 +373,8 @@ const SelectableSearchInput = _ref3 => {
|
|
|
387
373
|
};
|
|
388
374
|
const handleTextInputChange = event => {
|
|
389
375
|
setSearchValue(event.target.value);
|
|
390
|
-
if (
|
|
391
|
-
|
|
376
|
+
if (onChange) {
|
|
377
|
+
onChange({
|
|
392
378
|
target: {
|
|
393
379
|
id: SelectableSearchInput.getTextInputId(selectablSearchInputId),
|
|
394
380
|
name: getTextInputName(props.name),
|
|
@@ -450,8 +436,8 @@ const SelectableSearchInput = _ref3 => {
|
|
|
450
436
|
}, [onBlur, selectablSearchInputId, name]);
|
|
451
437
|
const handleDropdownChange = react$1.useCallback(nextSelectedOptions => {
|
|
452
438
|
setSearchOption(nextSelectedOptions.value);
|
|
453
|
-
if (
|
|
454
|
-
|
|
439
|
+
if (onChange) {
|
|
440
|
+
onChange({
|
|
455
441
|
target: {
|
|
456
442
|
id: SelectableSearchInput.getDropdownId(selectablSearchInputId),
|
|
457
443
|
name: getDropdownName(name),
|
|
@@ -460,7 +446,7 @@ const SelectableSearchInput = _ref3 => {
|
|
|
460
446
|
});
|
|
461
447
|
}
|
|
462
448
|
textInputRef.current?.focus();
|
|
463
|
-
}, [
|
|
449
|
+
}, [onChange, selectablSearchInputId, name]);
|
|
464
450
|
return jsxRuntime.jsx(Constraints__default["default"].Horizontal, {
|
|
465
451
|
max: horizontalConstraint,
|
|
466
452
|
children: jsxRuntime.jsxs(Container, {
|
|
@@ -483,7 +469,7 @@ const SelectableSearchInput = _ref3 => {
|
|
|
483
469
|
selectCustomComponents: props.selectCustomComponents
|
|
484
470
|
}))
|
|
485
471
|
}), jsxRuntime.jsxs("div", {
|
|
486
|
-
css: [getSelectableSearchInputContainerStyles(allProps), 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":"AAieiB","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  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  ...props\n}: 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 allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\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: 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={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\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":"AA6dU","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  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  ...props\n}: 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 allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\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: 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={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
|
|
472
|
+
css: [getSelectableSearchInputContainerStyles(allProps), 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":"AAseiB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  type FocusEvent,\n  useState,\n  useCallback,\n  useRef,\n  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  onChange,\n  ...props\n}: 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>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    onChange,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\n    );\n  }\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: 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 (onChange) {\n      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: FocusEvent) => {\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: { value: string } & Record<string, unknown>) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (onChange) {\n        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    [onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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={\n              handleDropdownChange as ReactSelectProps['onChange']\n            }\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\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":"AAkeU","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  type FocusEvent,\n  useState,\n  useCallback,\n  useRef,\n  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  onChange,\n  ...props\n}: 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>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    onChange,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\n    );\n  }\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: 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 (onChange) {\n      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: FocusEvent) => {\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: { value: string } & Record<string, unknown>) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (onChange) {\n        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    [onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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={\n              handleDropdownChange as ReactSelectProps['onChange']\n            }\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"],
|
|
487
473
|
children: [jsxRuntime.jsx("input", _objectSpread(_objectSpread(_objectSpread({
|
|
488
474
|
ref: textInputRef,
|
|
489
475
|
id: SelectableSearchInput.getTextInputId(selectablSearchInputId),
|
|
@@ -525,7 +511,7 @@ const SelectableSearchInput = _ref3 => {
|
|
|
525
511
|
css: getSearchIconButtonStyles(allProps),
|
|
526
512
|
isDisabled: props.isDisabled
|
|
527
513
|
}), props.rightActionIcon && props.rightActionProps && jsxRuntime.jsx("div", {
|
|
528
|
-
css: /*#__PURE__*/react.css("order:4;margin-left:", designSystem.designTokens.spacing30, ";" + (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+hBsB","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  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  ...props\n}: 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 allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\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: 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={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"),
|
|
514
|
+
css: /*#__PURE__*/react.css("order:4;margin-left:", designSystem.designTokens.spacing30, ";" + (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":"AAoiBsB","file":"selectable-search-input.tsx","sourcesContent":["import {\n  type MouseEvent,\n  type KeyboardEvent,\n  type ChangeEvent,\n  type ReactNode,\n  type FocusEvent,\n  useState,\n  useCallback,\n  useRef,\n  ReactElement,\n} from 'react';\nimport SecondaryIconButton from '@commercetools-uikit/secondary-icon-button';\nimport IconButton, {\n  type TIconButtonProps,\n} from '@commercetools-uikit/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  /**\n   * Custom action icon to be displayed on the right side of the input.\n   */\n  rightActionIcon?: ReactElement;\n  /**\n   * Props for the right-action icon-button. Required when rightActionIcon is provided.\n   * At least a `label` and an `onClick` prop/function need to be provided.\n   */\n  rightActionProps?: TIconButtonProps;\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 = ({\n  horizontalConstraint = 'scale',\n  isClearable = true,\n  menuHorizontalConstraint = 3,\n  showSubmitButton = true,\n  menuPortalZIndex = 1,\n  onChange,\n  ...props\n}: 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>(null);\n  const textInputRef = useRef<HTMLInputElement>(null);\n\n  const allProps = {\n    horizontalConstraint,\n    isClearable,\n    menuHorizontalConstraint,\n    showSubmitButton,\n    menuPortalZIndex,\n    onChange,\n    ...props,\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  if (props.rightActionIcon && !props.rightActionProps) {\n    warning(\n      false,\n      'SelectableSearchInput: `rightActionIcon` is provided but `rightActionProps` is missing. Provide an object with a `label` and `onClick` property.'\n    );\n  }\n  const selectablSearchInputId = useFieldId(\n    props.id,\n    selectableSearchInputSequentialId\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof onChange === 'function',\n      'SelectableSearchInput: `onChange` is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: 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 (onChange) {\n      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: FocusEvent) => {\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: { value: string } & Record<string, unknown>) => {\n      setSearchOption(nextSelectedOptions.value);\n      if (onChange) {\n        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    [onChange, selectablSearchInputId, name]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <Container\n        ref={containerRef}\n        onBlur={handleContainerBlur}\n        data-testid=\"selectable-search-input-container\"\n      >\n        <Constraints.Horizontal max={menuHorizontalConstraint}>\n          <SelectableSelect\n            {...allProps}\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={\n              handleDropdownChange as ReactSelectProps['onChange']\n            }\n            textInputRef={textInputRef}\n            selectedOption={selectedOption}\n            dataProps={transformedSelectDataProps}\n            selectCustomComponents={props.selectCustomComponents}\n          />\n        </Constraints.Horizontal>\n        <div\n          css={[\n            getSelectableSearchInputContainerStyles(allProps),\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(allProps)}\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          {isClearable &&\n            searchInputValue &&\n            !props.isDisabled &&\n            !props.isReadOnly && (\n              <SecondaryIconButton\n                icon={<CloseIcon />}\n                size={props.isCondensed ? '10' : '20'}\n                label={'clear-button'}\n                onClick={handleClear}\n                css={getClearIconButtonStyles(allProps)}\n              />\n            )}\n          {showSubmitButton && (\n            <SecondaryIconButton\n              icon={<SearchIcon />}\n              size={props.isCondensed ? '20' : '40'}\n              label={'search-button'}\n              onClick={handleSubmit}\n              css={getSearchIconButtonStyles(allProps)}\n              isDisabled={props.isDisabled}\n            />\n          )}\n\n          {props.rightActionIcon && props.rightActionProps && (\n            <div\n              css={css`\n                order: 4;\n                margin-left: ${designTokens.spacing30};\n              `}\n            >\n              <IconButton\n                theme=\"primary\"\n                isDisabled={props.isDisabled || props.isReadOnly}\n                size={props.isCondensed ? '10' : '20'}\n                icon={props.rightActionIcon}\n                {...props.rightActionProps}\n              />\n            </div>\n          )}\n        </div>\n      </Container>\n    </Constraints.Horizontal>\n  );\n};\n\nSelectableSearchInput.displayName = 'SelectableSearchInput';\nSelectableSearchInput.isEmpty = (\n  formValue: TSelectableSearchInputProps['value']\n) => !formValue || formValue.text.trim() === '';\nSelectableSearchInput.getTextInputId = getTextInputName;\nSelectableSearchInput.getDropdownId = getDropdownName;\n\nexport default SelectableSearchInput;\n"]} */"),
|
|
529
515
|
children: jsxRuntime.jsx(IconButton__default["default"], _objectSpread({
|
|
530
516
|
theme: "primary",
|
|
531
517
|
isDisabled: props.isDisabled || props.isReadOnly,
|
|
@@ -537,47 +523,6 @@ const SelectableSearchInput = _ref3 => {
|
|
|
537
523
|
})
|
|
538
524
|
});
|
|
539
525
|
};
|
|
540
|
-
SelectableSearchInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
541
|
-
id: _pt__default["default"].string,
|
|
542
|
-
autoComplete: _pt__default["default"].string,
|
|
543
|
-
'aria-invalid': _pt__default["default"].bool,
|
|
544
|
-
'aria-errormessage': _pt__default["default"].string,
|
|
545
|
-
name: _pt__default["default"].string,
|
|
546
|
-
value: _pt__default["default"].shape({
|
|
547
|
-
text: _pt__default["default"].string.isRequired,
|
|
548
|
-
option: _pt__default["default"].string.isRequired
|
|
549
|
-
}).isRequired,
|
|
550
|
-
_experimentalValue: _pt__default["default"].shape({
|
|
551
|
-
text: _pt__default["default"].string.isRequired,
|
|
552
|
-
option: _pt__default["default"].string.isRequired
|
|
553
|
-
}),
|
|
554
|
-
onChange: _pt__default["default"].func,
|
|
555
|
-
onBlur: _pt__default["default"].func,
|
|
556
|
-
onFocus: _pt__default["default"].func,
|
|
557
|
-
onSubmit: _pt__default["default"].func.isRequired,
|
|
558
|
-
onReset: _pt__default["default"].func,
|
|
559
|
-
isAutofocussed: _pt__default["default"].bool,
|
|
560
|
-
isDisabled: _pt__default["default"].bool,
|
|
561
|
-
isReadOnly: _pt__default["default"].bool,
|
|
562
|
-
hasError: _pt__default["default"].bool,
|
|
563
|
-
hasWarning: _pt__default["default"].bool,
|
|
564
|
-
placeholder: _pt__default["default"].string,
|
|
565
|
-
isClearable: _pt__default["default"].bool,
|
|
566
|
-
isCondensed: _pt__default["default"].bool,
|
|
567
|
-
horizontalConstraint: _pt__default["default"].oneOf([10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
568
|
-
options: _pt__default["default"].oneOfType([_pt__default["default"].arrayOf(_pt__default["default"].shape({
|
|
569
|
-
value: _pt__default["default"].string.isRequired,
|
|
570
|
-
label: _pt__default["default"].node
|
|
571
|
-
})), _pt__default["default"].arrayOf(_pt__default["default"].shape({
|
|
572
|
-
options: _pt__default["default"].arrayOf(_pt__default["default"].shape({
|
|
573
|
-
value: _pt__default["default"].string.isRequired,
|
|
574
|
-
label: _pt__default["default"].node
|
|
575
|
-
})).isRequired
|
|
576
|
-
}))]).isRequired,
|
|
577
|
-
menuPortalZIndex: _pt__default["default"].number,
|
|
578
|
-
menuHorizontalConstraint: _pt__default["default"].oneOf([3, 4, 5]),
|
|
579
|
-
showSubmitButton: _pt__default["default"].bool
|
|
580
|
-
} : {};
|
|
581
526
|
SelectableSearchInput.displayName = 'SelectableSearchInput';
|
|
582
527
|
SelectableSearchInput.isEmpty = formValue => {
|
|
583
528
|
var _context3;
|
|
@@ -588,7 +533,7 @@ SelectableSearchInput.getDropdownId = getDropdownName;
|
|
|
588
533
|
var SelectableSearchInput$1 = SelectableSearchInput;
|
|
589
534
|
|
|
590
535
|
// NOTE: This string will be replaced on build time with the package version.
|
|
591
|
-
var version = "
|
|
536
|
+
var version = "20.0.0";
|
|
592
537
|
|
|
593
538
|
exports["default"] = SelectableSearchInput$1;
|
|
594
539
|
exports.version = version;
|