@commercetools-uikit/money-input 19.22.1 → 19.22.2

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.
@@ -771,8 +771,7 @@ const TooltipWrapper = /*#__PURE__*/_styled__default["default"]("div", process.e
771
771
  styles: "display:flex"
772
772
  } : {
773
773
  name: "zjik7",
774
- styles: "display:flex",
775
- map: "/*# 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\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\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            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"]} */",
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\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\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            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"]} */",
776
775
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
777
776
  });
778
777
  const moneyInputSequentialId = utils.createSequentialId('money-input-');
@@ -1068,8 +1067,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
1068
1067
  styles: "position:relative;width:100%"
1069
1068
  } : {
1070
1069
  name: "h5hvsa-MoneyInput",
1071
- styles: "position:relative;width:100%;label:MoneyInput;",
1072
- map: "/*# 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\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\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            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"]} */",
1070
+ 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\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\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            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"]} */",
1073
1071
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1074
1072
  };
1075
1073
  var _ref2 = process.env.NODE_ENV === "production" ? {
@@ -1077,8 +1075,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
1077
1075
  styles: "font-family:inherit;width:100%;position:relative;display:flex"
1078
1076
  } : {
1079
1077
  name: "1w49f4-MoneyInput",
1080
- styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;",
1081
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAgxBgB","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\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\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            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"]} */",
1078
+ 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":"AAgxBgB","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\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\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            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"]} */",
1082
1079
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1083
1080
  };
1084
1081
  const MoneyInput = _ref5 => {
@@ -1424,7 +1421,7 @@ MoneyInput.isTouched = touched => Boolean(touched && touched.currencyCode && tou
1424
1421
  var MoneyInput$1 = MoneyInput;
1425
1422
 
1426
1423
  // NOTE: This string will be replaced on build time with the package version.
1427
- var version = "19.22.1";
1424
+ var version = "19.22.2";
1428
1425
 
1429
1426
  exports["default"] = MoneyInput$1;
1430
1427
  exports.version = version;