@commercetools-uikit/money-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.
@@ -14,7 +14,6 @@ var _slicedToArray = require('@babel/runtime-corejs3/helpers/slicedToArray');
14
14
  var _defineProperty = require('@babel/runtime-corejs3/helpers/defineProperty');
15
15
  var _objectWithoutProperties = require('@babel/runtime-corejs3/helpers/objectWithoutProperties');
16
16
  var _styled = require('@emotion/styled/base');
17
- var _pt = require('prop-types');
18
17
  var _lastIndexOfInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/last-index-of');
19
18
  var _parseFloat = require('@babel/runtime-corejs3/core-js-stable/parse-float');
20
19
  var _Math$trunc = require('@babel/runtime-corejs3/core-js-stable/math/trunc');
@@ -50,7 +49,6 @@ var _Object$getOwnPropertyDescriptors__default = /*#__PURE__*/_interopDefault(_O
50
49
  var _Object$defineProperties__default = /*#__PURE__*/_interopDefault(_Object$defineProperties);
51
50
  var _Object$defineProperty__default = /*#__PURE__*/_interopDefault(_Object$defineProperty);
52
51
  var _styled__default = /*#__PURE__*/_interopDefault(_styled);
53
- var _pt__default = /*#__PURE__*/_interopDefault(_pt);
54
52
  var _lastIndexOfInstanceProperty__default = /*#__PURE__*/_interopDefault(_lastIndexOfInstanceProperty);
55
53
  var _parseFloat__default = /*#__PURE__*/_interopDefault(_parseFloat);
56
54
  var _Math$trunc__default = /*#__PURE__*/_interopDefault(_Math$trunc);
@@ -759,7 +757,7 @@ var messages = reactIntl.defineMessages({
759
757
  const _excluded = ["id"],
760
758
  _excluded2 = ["currencies", "horizontalConstraint", "menuPortalZIndex"];
761
759
  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; }
762
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var _context11, _context12; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty__default["default"](_context11 = ownKeys(Object(t), !0)).call(_context11, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](e, _Object$getOwnPropertyDescriptors__default["default"](t)) : _forEachInstanceProperty__default["default"](_context12 = ownKeys(Object(t))).call(_context12, function (r) { _Object$defineProperty__default["default"](e, r, _Object$getOwnPropertyDescriptor__default["default"](t, r)); }); } return e; }
760
+ function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var _context1, _context10; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty__default["default"](_context1 = ownKeys(Object(t), !0)).call(_context1, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](e, _Object$getOwnPropertyDescriptors__default["default"](t)) : _forEachInstanceProperty__default["default"](_context10 = ownKeys(Object(t))).call(_context10, function (r) { _Object$defineProperty__default["default"](e, r, _Object$getOwnPropertyDescriptor__default["default"](t, r)); }); } return e; }
763
761
  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)."; }
764
762
  const TooltipWrapper = /*#__PURE__*/_styled__default["default"]("div", process.env.NODE_ENV === "production" ? {
765
763
  target: "e1u90zjc0"
@@ -771,7 +769,7 @@ const TooltipWrapper = /*#__PURE__*/_styled__default["default"]("div", process.e
771
769
  styles: "display:flex"
772
770
  } : {
773
771
  name: "zjik7",
774
- styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAmCiC","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
772
+ styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA0CiC","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
775
773
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
776
774
  });
777
775
  const moneyInputSequentialId = utils.createSequentialId('money-input-');
@@ -784,25 +782,11 @@ const Portal = props => {
784
782
  }
785
783
  return null;
786
784
  };
787
- Portal.propTypes = {
788
- id: _pt__default["default"].string.isRequired,
789
- children: _pt__default["default"].node,
790
- isCondensed: _pt__default["default"].bool,
791
- isDisabled: _pt__default["default"].bool,
792
- isReadOnly: _pt__default["default"].bool
793
- };
794
785
  const CurrencyLabel = props => jsxRuntime.jsx("label", {
795
786
  htmlFor: props.id,
796
787
  css: getCurrencyLabelStyles(props),
797
788
  children: props.children
798
789
  });
799
- CurrencyLabel.propTypes = process.env.NODE_ENV !== "production" ? {
800
- id: _pt__default["default"].string.isRequired,
801
- children: _pt__default["default"].node,
802
- isCondensed: _pt__default["default"].bool,
803
- isDisabled: _pt__default["default"].bool,
804
- isReadOnly: _pt__default["default"].bool
805
- } : {};
806
790
  CurrencyLabel.displayName = 'CurrencyLabel';
807
791
  const SingleValue = _ref3 => {
808
792
  let id = _ref3.id,
@@ -814,10 +798,6 @@ const SingleValue = _ref3 => {
814
798
  })
815
799
  }));
816
800
  };
817
- SingleValue.propTypes = process.env.NODE_ENV !== "production" ? {
818
- id: _pt__default["default"].string,
819
- children: _pt__default["default"].node
820
- } : {};
821
801
  SingleValue.displayName = 'SingleValue';
822
802
  // overwrite styles of createSelectStyles
823
803
  const createCurrencySelectStyles = _ref4 => {
@@ -1068,7 +1048,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
1068
1048
  styles: "position:relative;width:100%"
1069
1049
  } : {
1070
1050
  name: "h5hvsa-MoneyInput",
1071
- styles: "position:relative;width:100%;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAs0BkB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
1051
+ styles: "position:relative;width:100%;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAi1BkB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
1072
1052
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1073
1053
  };
1074
1054
  var _ref2 = process.env.NODE_ENV === "production" ? {
@@ -1076,7 +1056,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
1076
1056
  styles: "font-family:inherit;width:100%;position:relative;display:flex"
1077
1057
  } : {
1078
1058
  name: "1w49f4-MoneyInput",
1079
- styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA+wBgB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
1059
+ styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAwxBgB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
1080
1060
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1081
1061
  };
1082
1062
  const MoneyInput = _ref5 => {
@@ -1144,13 +1124,13 @@ const MoneyInput = _ref5 => {
1144
1124
  }
1145
1125
  }, [intl.locale, onChange, moneyInputId, props.name, props.value.amount, props.value.currencyCode, toggleAmountHasFocus]);
1146
1126
  const handleAmountChange = react$1.useCallback(event => {
1147
- if (utils.isNumberish(event.target.value)) {
1127
+ if (utils.isNumberish(event.target?.value)) {
1148
1128
  onChange?.({
1149
1129
  persist: () => {},
1150
1130
  target: {
1151
1131
  id: MoneyInput.getAmountInputId(moneyInputId),
1152
1132
  name: getAmountInputName(props.name),
1153
- value: event.target.value
1133
+ value: event.target?.value
1154
1134
  }
1155
1135
  });
1156
1136
  }
@@ -1263,7 +1243,7 @@ const MoneyInput = _ref5 => {
1263
1243
  ref: containerRef,
1264
1244
  css: _ref2,
1265
1245
  "data-testid": "money-input-container",
1266
- onBlur: handleContainerBlur,
1246
+ onBlur: event => handleContainerBlur(event),
1267
1247
  children: [hasNoCurrencies ? jsxRuntime.jsx(CurrencyLabel, {
1268
1248
  id: MoneyInput.getAmountInputId(moneyInputId),
1269
1249
  isCondensed: props.isCondensed,
@@ -1310,7 +1290,7 @@ const MoneyInput = _ref5 => {
1310
1290
  hasFocus
1311
1291
  })),
1312
1292
  // accounts for size of icon
1313
- props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.designTokens.spacing40, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAw1BmB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), currencyHasFocus && !props.isDisabled && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA81BmB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAm1BY","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"],
1293
+ props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.designTokens.spacing40, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAm2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), currencyHasFocus && !props.isDisabled && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAy2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA81BY","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\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   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\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   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: 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   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\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   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\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 as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"],
1314
1294
  placeholder: props.placeholder,
1315
1295
  onChange: handleAmountChange,
1316
1296
  onBlur: handleAmountBlur,
@@ -1357,32 +1337,6 @@ const MoneyInput = _ref5 => {
1357
1337
  })
1358
1338
  });
1359
1339
  };
1360
- MoneyInput.propTypes = process.env.NODE_ENV !== "production" ? {
1361
- id: _pt__default["default"].string,
1362
- autoComplete: _pt__default["default"].string,
1363
- 'aria-invalid': _pt__default["default"].bool,
1364
- 'aria-errormessage': _pt__default["default"].string,
1365
- name: _pt__default["default"].string,
1366
- value: _pt__default["default"].shape({
1367
- amount: _pt__default["default"].string.isRequired,
1368
- currencyCode: _pt__default["default"].oneOfType([_pt__default["default"].any, _pt__default["default"].oneOf([''])]).isRequired
1369
- }).isRequired,
1370
- currencies: _pt__default["default"].arrayOf(_pt__default["default"].string),
1371
- placeholder: _pt__default["default"].string,
1372
- onBlur: _pt__default["default"].func,
1373
- onFocus: _pt__default["default"].func,
1374
- isCondensed: _pt__default["default"].bool,
1375
- isDisabled: _pt__default["default"].bool,
1376
- isReadOnly: _pt__default["default"].bool,
1377
- isAutofocussed: _pt__default["default"].bool,
1378
- onChange: _pt__default["default"].func,
1379
- menuPortalZIndex: _pt__default["default"].number,
1380
- hasError: _pt__default["default"].bool,
1381
- hasWarning: _pt__default["default"].bool,
1382
- hasHighPrecisionBadge: _pt__default["default"].bool,
1383
- horizontalConstraint: _pt__default["default"].oneOf([3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
1384
- isCurrencyInputDisabled: _pt__default["default"].bool
1385
- } : {};
1386
1340
  MoneyInput.displayName = 'MoneyInput';
1387
1341
  MoneyInput.getAmountInputId = getAmountInputName;
1388
1342
  MoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;
@@ -1411,8 +1365,8 @@ MoneyInput.parseMoneyValue = (moneyValue, locale) => {
1411
1365
  };
1412
1366
  };
1413
1367
  MoneyInput.isEmpty = formValue => {
1414
- var _context9, _context10;
1415
- return !formValue || _trimInstanceProperty__default["default"](_context9 = formValue.amount).call(_context9) === '' || _trimInstanceProperty__default["default"](_context10 = formValue.currencyCode).call(_context10) === '';
1368
+ var _context9, _context0;
1369
+ return !formValue || _trimInstanceProperty__default["default"](_context9 = formValue.amount).call(_context9) === '' || _trimInstanceProperty__default["default"](_context0 = formValue.currencyCode).call(_context0) === '';
1416
1370
  };
1417
1371
  MoneyInput.isHighPrecision = (formValue, locale) => {
1418
1372
  process.env.NODE_ENV !== "production" ? utils.warning(!MoneyInput.isEmpty(formValue), 'MoneyValue.isHighPrecision may not be called with an empty money value.') : void 0;
@@ -1423,7 +1377,7 @@ MoneyInput.isTouched = touched => Boolean(touched && touched.currencyCode && tou
1423
1377
  var MoneyInput$1 = MoneyInput;
1424
1378
 
1425
1379
  // NOTE: This string will be replaced on build time with the package version.
1426
- var version = "19.26.0";
1380
+ var version = "20.0.0";
1427
1381
 
1428
1382
  exports["default"] = MoneyInput$1;
1429
1383
  exports.version = version;