@commercetools-uikit/money-input 13.0.2 → 14.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -770,7 +770,7 @@ var TooltipWrapper = _styled__default["default"]("div", process.env.NODE_ENV ===
770
770
  } : {
771
771
  name: "zjik7",
772
772
  styles: "display:flex",
773
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAkCiC","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\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 prices two types of prices: centPrecision and\n// highPrecision. 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, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 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//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 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\n// 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\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[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    `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};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[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\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\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   * 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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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 (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\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          currencyCode,\n          intl.locale\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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\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(currencies, 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    moneyValue.currencyCode,\n    locale\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) => {\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 && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
773
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAkCiC","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof currencies;\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 = currencies[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 ** currencies[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    : currencies[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 TEvent = {\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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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      currencies[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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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(currencies, 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\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
774
774
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
775
775
  });
776
776
 
@@ -891,35 +891,39 @@ var createCurrencySelectStyles = function createCurrencySelectStyles(_ref4, them
891
891
  };
892
892
  }
893
893
  });
894
- }; // The MoneyInput component always operates on a value consisting of:
895
- // { amount: String, currencyCode: String }
894
+ };
895
+
896
+ // The MoneyInput component always operates on a value consisting of:
897
+ // ```
898
+ // { amount: String, currencyCode: String }
899
+ // ```
896
900
  //
897
901
  // The amount may only use a dot as the decimal separator.
898
- // The currencyCode must be supported by the API.
902
+ // The `currencyCode` must be supported by the API.
899
903
  //
900
- // The MoneyInput does not do any validation on its own. It only serves as a way
901
- // to get the amount and currencyCode input from the user. Validation is always
904
+ // The `MoneyInput` does not do any validation on its own. It only serves as a way
905
+ // to get the amount and `currencyCode` input from the user. Validation is always
902
906
  // up to the parent.
903
907
  //
904
- // The CTP API supports prices two types of prices: centPrecision and
905
- // highPrecision. The MoneyInput itself does not know about these. However,
906
- // it has two static methods defined (convertToMoneyValue and parseMoneyValue),
907
- // which can be used to convert between MoneyInput value and the MoneyValue
908
+ // The CTP API supports two types of prices: `centPrecision` and `highPrecision`.
909
+ // The `MoneyInput` itself does not know about these. However,
910
+ // it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),
911
+ // which can be used to convert between `MoneyInput` value and the `MoneyValue`
908
912
  // supported by the API.
909
- // Some places in the API do not support highPrecision prices, but the
910
- // convertToMoneyValue will always return either a centPrecision or a
911
- // highPrecision price. It's up the MoneyInput's parent to show a validation
912
- // error in case a highPrecision price is used.
913
+ // Some places in the API do not support `highPrecision` prices, but the
914
+ // `convertToMoneyValue `will always return either a `centPrecision `or a
915
+ // `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation
916
+ // error in case a `highPrecision` price is used.
913
917
  //
914
- // A value is considered as to have highPrecision when the number of supplied
918
+ // A value is considered as to have `highPrecision` when the number of supplied
915
919
  // fraction digits exceed the number of fraction digits the currency uses. For
916
- // example, 42.00 is always a centPrecision price, while 42.001 is always a
917
- // highPrecision price. It is not possible to hae 42.00 as a highPrecision
920
+ // example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a
921
+ // `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`
918
922
  // price.
919
923
  //
920
- // The first time the component renders, we want to try to show the centAmount
921
- // as a formatted number. To achieve this, the parseMoneyValue function can
922
- // be used to turn the API value into a value the MoneyInput understands.
924
+ // The first time the component renders, we want to try to show the `centAmount`
925
+ // as a formatted number. To achieve this, the `parseMoneyValue` function can
926
+ // be used to turn the API value into a value the `MoneyInput` understands.
923
927
  // During this transformation, the money value will get formatted into "amount".
924
928
  //
925
929
  // When the user changes the value, we don't want to format again. We only format
@@ -927,16 +931,19 @@ var createCurrencySelectStyles = function createCurrencySelectStyles(_ref4, them
927
931
  // formatting would mess with the user's input.
928
932
  //
929
933
  //
930
- // A full example of an MoneyValue with centPrecision would be
934
+ // A full example of an `MoneyValue` with `centPrecision` would be
935
+ // ```
931
936
  // {
932
937
  // "type": "centPrecision",
933
938
  // "currencyCode": "EUR",
934
939
  // "centAmount": 4200,
935
940
  // "fractionDigits": 2
936
941
  // }
937
- // which equals 42.00 €
942
+ // ```
943
+ // which equals `EUR 42.00`.
938
944
  //
939
- // A full example of an MoneyValue with highPrecision would be
945
+ // A full example of an `MoneyValue` with `highPrecision` would be
946
+ // ```
940
947
  // {
941
948
  // "type": "highPrecision",
942
949
  // "currencyCode": "EUR",
@@ -944,7 +951,8 @@ var createCurrencySelectStyles = function createCurrencySelectStyles(_ref4, them
944
951
  // "preciseAmount": 123456,
945
952
  // "fractionDigits": 7
946
953
  // }
947
- // which equals 0.0123456 €
954
+ // ```
955
+ // which equals `EUR 0.0123456`
948
956
  // Parsing
949
957
  // Since most users are not careful about how they enter values, we will parse
950
958
  // both `.` and `,` as decimal separators.
@@ -952,10 +960,7 @@ var createCurrencySelectStyles = function createCurrencySelectStyles(_ref4, them
952
960
  // When a value is `1,000.00` we also parse it as `1000`.
953
961
  //
954
962
  // This means the highest amount always wins. We do this by comparing the last
955
- // position of `.` and `,`. Whatever occurs later is used as the decimal
956
- // separator.
957
-
958
-
963
+ // position of `.` and `,`. Whatever occurs later is used as the decimal separator.
959
964
  var parseRawAmountToNumber = function parseRawAmountToNumber(rawAmount, locale) {
960
965
  var fractionsSeparator;
961
966
 
@@ -992,14 +997,14 @@ var parseRawAmountToNumber = function parseRawAmountToNumber(rawAmount, locale)
992
997
  //
993
998
  // This function expects the "amount" to be a trimmed value.
994
999
 
995
- var createMoneyValue = function createMoneyValue(currencyCode, rawAmount, locale) {
1000
+ var createMoneyValue = function createMoneyValue(rawAmount, locale, currencyCode) {
996
1001
  var _context3, _context4;
997
1002
 
998
1003
  if (!currencyCode) return null;
999
1004
  var currency = currencies[currencyCode];
1000
1005
  if (!currency) return null;
1001
1006
  if (rawAmount.length === 0 || !utils.isNumberish(rawAmount)) return null;
1002
- process.env.NODE_ENV !== "production" ? utils.warning(locale || currency.fractionDigits !== 0, "A locale must be provided when currency has no fraction digits (".concat(currencyCode, ")")) : void 0;
1007
+ process.env.NODE_ENV !== "production" ? utils.warning(locale || currency.fractionDigits !== 0, "MoneyInput: A locale must be provided when currency has no fraction digits (".concat(currencyCode, ")")) : void 0;
1003
1008
  var amountAsNumber = parseRawAmountToNumber(rawAmount, locale);
1004
1009
  if (isNaN(amountAsNumber)) return null; // The cent amount is rounded to the currencie's default number
1005
1010
  // of fraction digits for prices with high precision.
@@ -1049,17 +1054,23 @@ var createMoneyValue = function createMoneyValue(currencyCode, rawAmount, locale
1049
1054
  };
1050
1055
  };
1051
1056
 
1057
+ var createEmptyMoneyValue = function createEmptyMoneyValue(currencyCode) {
1058
+ return {
1059
+ type: 'centPrecision',
1060
+ currencyCode: currencyCode,
1061
+ centAmount: NaN,
1062
+ fractionDigits: 2
1063
+ };
1064
+ };
1065
+
1052
1066
  var getAmountAsNumberFromMoneyValue = function getAmountAsNumberFromMoneyValue(moneyValue) {
1053
1067
  return moneyValue.type === 'highPrecision' ? moneyValue.preciseAmount / Math.pow(10, moneyValue.fractionDigits) : moneyValue.centAmount / Math.pow(10, currencies[moneyValue.currencyCode].fractionDigits);
1054
1068
  }; // gets called with a string and should return a formatted string
1055
1069
 
1056
1070
 
1057
- var formatAmount = function formatAmount(rawAmount, currencyCode, locale) {
1071
+ var formatAmount = function formatAmount(rawAmount, locale, currencyCode) {
1058
1072
  // fallback in case the user didn't enter an amount yet (or it's invalid)
1059
- var moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {
1060
- currencyCode: currencyCode,
1061
- centAmount: NaN
1062
- };
1073
+ var moneyValue = createMoneyValue(rawAmount, locale, currencyCode) || createEmptyMoneyValue(currencyCode);
1063
1074
  var amount = getAmountAsNumberFromMoneyValue(moneyValue);
1064
1075
  var fractionDigits = moneyValue.preciseAmount ? moneyValue.fractionDigits : currencies[moneyValue.currencyCode].fractionDigits;
1065
1076
  return isNaN(amount) ? '' : amount.toLocaleString(locale, {
@@ -1087,7 +1098,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
1087
1098
  } : {
1088
1099
  name: "h5hvsa-MoneyInput",
1089
1100
  styles: "position:relative;width:100%;label:MoneyInput;",
1090
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAswBkB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\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 prices two types of prices: centPrecision and\n// highPrecision. 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, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 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//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 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\n// 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\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[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    `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};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[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\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\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   * 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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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 (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\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          currencyCode,\n          intl.locale\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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\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(currencies, 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    moneyValue.currencyCode,\n    locale\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) => {\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 && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
1101
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAgyBkB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof currencies;\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 = currencies[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 ** currencies[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    : currencies[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 TEvent = {\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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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      currencies[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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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(currencies, 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\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
1091
1102
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1092
1103
  };
1093
1104
 
@@ -1097,7 +1108,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
1097
1108
  } : {
1098
1109
  name: "1w49f4-MoneyInput",
1099
1110
  styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;",
1100
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAotBgB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\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 prices two types of prices: centPrecision and\n// highPrecision. 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, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 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//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 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\n// 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\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[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    `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};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[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\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\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   * 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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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 (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\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          currencyCode,\n          intl.locale\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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\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(currencies, 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    moneyValue.currencyCode,\n    locale\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) => {\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 && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
1111
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA8uBgB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof currencies;\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 = currencies[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 ** currencies[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    : currencies[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 TEvent = {\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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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      currencies[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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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(currencies, 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\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
1101
1112
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
1102
1113
  };
1103
1114
 
@@ -1143,8 +1154,8 @@ var MoneyInput = function MoneyInput(props) {
1143
1154
  toggleAmountHasFocus(false); // Skip formatting for empty value or when the input is used with an
1144
1155
  // unknown currency.
1145
1156
 
1146
- if (amount.length > 0 && currencies[props.value.currencyCode]) {
1147
- var formattedAmount = formatAmount(amount, props.value.currencyCode, intl.locale); // When the user entered a value with centPrecision, we can format
1157
+ if (amount.length > 0 && props.value.currencyCode && currencies[props.value.currencyCode]) {
1158
+ var formattedAmount = formatAmount(amount, intl.locale, props.value.currencyCode); // When the user entered a value with centPrecision, we can format
1148
1159
  // the resulting value to that currency, e.g. 20.1 to 20.10
1149
1160
 
1150
1161
  if (String(formattedAmount) !== amount) {
@@ -1185,7 +1196,7 @@ var MoneyInput = function MoneyInput(props) {
1185
1196
  // currency's number of fraction digits.
1186
1197
  // When the currency was a high-precision price, then no digits should
1187
1198
  // be lost
1188
- var formattedAmount = formatAmount(_trimInstanceProperty__default["default"](_context6 = props.value.amount).call(_context6), currencyCode, intl.locale); // The user could be changing the currency before entering any amount,
1199
+ var formattedAmount = formatAmount(_trimInstanceProperty__default["default"](_context6 = props.value.amount).call(_context6), intl.locale, currencyCode); // The user could be changing the currency before entering any amount,
1189
1200
  // or while the amount is invalid. In these cases, we don't attempt to
1190
1201
  // format the amount.
1191
1202
 
@@ -1332,7 +1343,7 @@ var MoneyInput = function MoneyInput(props) {
1332
1343
  "data-testid": "currency-dropdown"
1333
1344
  }), jsxRuntime.jsxs("div", {
1334
1345
  css: _ref,
1335
- children: [jsxRuntime.jsx("input", _objectSpread({
1346
+ children: [jsxRuntime.jsx("input", _objectSpread(_objectSpread({
1336
1347
  ref: amountInputRef,
1337
1348
  id: MoneyInput.getAmountInputId(moneyInputId),
1338
1349
  autoComplete: props.autoComplete,
@@ -1343,14 +1354,18 @@ var MoneyInput = function MoneyInput(props) {
1343
1354
  css: [getAmountInputStyles(_objectSpread(_objectSpread({}, props), {}, {
1344
1355
  hasFocus: hasFocus
1345
1356
  })), // accounts for size of icon
1346
- props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.customProperties.spacingL, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAwxBmB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\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 prices two types of prices: centPrecision and\n// highPrecision. 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, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 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//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 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\n// 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\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[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    `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};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[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\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\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   * 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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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 (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\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          currencyCode,\n          intl.locale\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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\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(currencies, 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    moneyValue.currencyCode,\n    locale\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) => {\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 && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAmxBY","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\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 prices two types of prices: centPrecision and\n// highPrecision. 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, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 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//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 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\n// 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\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[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    `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};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[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\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\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   * 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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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 (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\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          currencyCode,\n          intl.locale\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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\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(currencies, 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    moneyValue.currencyCode,\n    locale\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) => {\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 && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"],
1357
+ props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.customProperties.spacingL, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAkzBmB","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof currencies;\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 = currencies[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 ** currencies[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    : currencies[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 TEvent = {\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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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      currencies[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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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(currencies, 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\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA6yBY","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, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } 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} 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 currencies from './currencies.json';\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  isDisabled?: 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()}>\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 = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.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 vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof currencies;\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 = currencies[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 ** currencies[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    : currencies[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 TEvent = {\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: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\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: TEvent) => 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   */\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\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: 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  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      currencies[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 = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.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={props.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            isDisabled={props.isDisabled}\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.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,\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: ${vars.spacingL};\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(props)}\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: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 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(currencies, 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\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"],
1347
1358
  placeholder: props.placeholder,
1348
1359
  onChange: handleAmountChange,
1349
1360
  onBlur: handleAmountBlur,
1350
1361
  disabled: props.isDisabled,
1351
1362
  readOnly: props.isReadOnly,
1352
1363
  autoFocus: props.isAutofocussed
1353
- }, utils.filterDataAttributes(props))), props.hasHighPrecisionBadge && isHighPrecision && jsxRuntime.jsxs(jsxRuntime.Fragment, {
1364
+ }, utils.filterDataAttributes(props)), {}, {
1365
+ /* ARIA */
1366
+ "aria-invalid": props['aria-invalid'],
1367
+ "aria-errormessage": props['aria-errormessage']
1368
+ })), props.hasHighPrecisionBadge && isHighPrecision && jsxRuntime.jsxs(jsxRuntime.Fragment, {
1354
1369
  children: [!props.isDisabled && jsxRuntime.jsx("div", {
1355
1370
  id: getPortalId(props.id)
1356
1371
  }), jsxRuntime.jsx("div", {
@@ -1388,10 +1403,12 @@ var MoneyInput = function MoneyInput(props) {
1388
1403
  MoneyInput.propTypes = process.env.NODE_ENV !== "production" ? {
1389
1404
  id: _pt__default["default"].string,
1390
1405
  autoComplete: _pt__default["default"].string,
1406
+ 'aria-invalid': _pt__default["default"].bool,
1407
+ 'aria-errormessage': _pt__default["default"].string,
1391
1408
  name: _pt__default["default"].string,
1392
1409
  value: _pt__default["default"].shape({
1393
1410
  amount: _pt__default["default"].string.isRequired,
1394
- currencyCode: _pt__default["default"].any.isRequired
1411
+ currencyCode: _pt__default["default"].oneOfType([_pt__default["default"].any, _pt__default["default"].oneOf([''])]).isRequired
1395
1412
  }).isRequired,
1396
1413
  currencies: _pt__default["default"].arrayOf(_pt__default["default"].string).isRequired,
1397
1414
  placeholder: _pt__default["default"].string,
@@ -1414,7 +1431,7 @@ MoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;
1414
1431
  MoneyInput.convertToMoneyValue = function (value, locale) {
1415
1432
  var _context11;
1416
1433
 
1417
- return createMoneyValue(value.currencyCode, typeof value.amount === 'string' ? _trimInstanceProperty__default["default"](_context11 = value.amount).call(_context11) : '', locale);
1434
+ return createMoneyValue(typeof value.amount === 'string' ? _trimInstanceProperty__default["default"](_context11 = value.amount).call(_context11) : '', locale, value.currencyCode);
1418
1435
  };
1419
1436
 
1420
1437
  MoneyInput.parseMoneyValue = function (moneyValue, locale) {
@@ -1430,7 +1447,7 @@ MoneyInput.parseMoneyValue = function (moneyValue, locale) {
1430
1447
  typeof moneyValue.centAmount === 'number' || typeof moneyValue.preciseAmount === 'number' && typeof moneyValue.fractionDigits === 'number', 'MoneyInput.parseMoneyValue: Value must contain "amount"') : void 0;
1431
1448
  var amount = formatAmount(getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {
1432
1449
  minimumFractionDigits: moneyValue.fractionDigits
1433
- }), moneyValue.currencyCode, locale);
1450
+ }), locale, moneyValue.currencyCode);
1434
1451
  return {
1435
1452
  amount: amount,
1436
1453
  currencyCode: moneyValue.currencyCode
@@ -1446,7 +1463,7 @@ MoneyInput.isEmpty = function (formValue) {
1446
1463
  MoneyInput.isHighPrecision = function (formValue, locale) {
1447
1464
  process.env.NODE_ENV !== "production" ? utils.warning(!MoneyInput.isEmpty(formValue), 'MoneyValue.isHighPrecision may not be called with an empty money value.') : void 0;
1448
1465
  var moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);
1449
- return moneyValue && moneyValue.type === 'highPrecision';
1466
+ return (moneyValue === null || moneyValue === void 0 ? void 0 : moneyValue.type) === 'highPrecision';
1450
1467
  };
1451
1468
 
1452
1469
  MoneyInput.isTouched = function (touched) {
@@ -1457,7 +1474,7 @@ MoneyInput.defaultProps = defaultProps;
1457
1474
  var MoneyInput$1 = MoneyInput;
1458
1475
 
1459
1476
  // NOTE: This string will be replaced on build time with the package version.
1460
- var version = "13.0.2";
1477
+ var version = "14.0.0";
1461
1478
 
1462
1479
  exports["default"] = MoneyInput$1;
1463
1480
  exports.version = version;