@commercetools-uikit/money-input 20.4.0 → 20.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -728,6 +728,25 @@ const currencies = {
|
|
|
728
728
|
},
|
|
729
729
|
ZWR: {
|
|
730
730
|
fractionDigits: 2
|
|
731
|
+
},
|
|
732
|
+
// Zero-fraction variants (non-ISO): base currency with 0 decimal places.
|
|
733
|
+
CZK0: {
|
|
734
|
+
fractionDigits: 0
|
|
735
|
+
},
|
|
736
|
+
HUF0: {
|
|
737
|
+
fractionDigits: 0
|
|
738
|
+
},
|
|
739
|
+
ILS0: {
|
|
740
|
+
fractionDigits: 0
|
|
741
|
+
},
|
|
742
|
+
KZT0: {
|
|
743
|
+
fractionDigits: 0
|
|
744
|
+
},
|
|
745
|
+
TRY0: {
|
|
746
|
+
fractionDigits: 0
|
|
747
|
+
},
|
|
748
|
+
TWD0: {
|
|
749
|
+
fractionDigits: 0
|
|
731
750
|
}
|
|
732
751
|
};
|
|
733
752
|
var allCurrencies = currencies;
|
|
@@ -773,7 +792,7 @@ const TooltipWrapper = /*#__PURE__*/_styled__default["default"]("div", process.e
|
|
|
773
792
|
styles: "display:flex"
|
|
774
793
|
} : {
|
|
775
794
|
name: "zjik7",
|
|
776
|
-
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA0CiC","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
795
|
+
styles: "display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA0CiC","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
777
796
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
778
797
|
});
|
|
779
798
|
const moneyInputSequentialId = utils.createSequentialId('money-input-');
|
|
@@ -1027,12 +1046,15 @@ const createMoneyValue = (rawAmount, locale, currencyCode) => {
|
|
|
1027
1046
|
fractionDigits: currency.fractionDigits
|
|
1028
1047
|
};
|
|
1029
1048
|
};
|
|
1030
|
-
const createEmptyMoneyValue = currencyCode =>
|
|
1031
|
-
|
|
1032
|
-
|
|
1033
|
-
|
|
1034
|
-
|
|
1035
|
-
|
|
1049
|
+
const createEmptyMoneyValue = currencyCode => {
|
|
1050
|
+
var _allCurrencies$curren;
|
|
1051
|
+
return {
|
|
1052
|
+
type: 'centPrecision',
|
|
1053
|
+
currencyCode,
|
|
1054
|
+
centAmount: NaN,
|
|
1055
|
+
fractionDigits: (_allCurrencies$curren = allCurrencies[currencyCode].fractionDigits) !== null && _allCurrencies$curren !== void 0 ? _allCurrencies$curren : 2
|
|
1056
|
+
};
|
|
1057
|
+
};
|
|
1036
1058
|
const getAmountAsNumberFromMoneyValue = moneyValue => moneyValue.type === 'highPrecision' ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits : moneyValue.centAmount / 10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;
|
|
1037
1059
|
|
|
1038
1060
|
// gets called with a string and should return a formatted string
|
|
@@ -1052,7 +1074,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
1052
1074
|
styles: "position:relative;width:100%"
|
|
1053
1075
|
} : {
|
|
1054
1076
|
name: "h5hvsa-MoneyInput",
|
|
1055
|
-
styles: "position:relative;width:100%;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAi1BkB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
1077
|
+
styles: "position:relative;width:100%;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAi1BkB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
1056
1078
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1057
1079
|
};
|
|
1058
1080
|
var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
@@ -1060,7 +1082,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
|
1060
1082
|
styles: "font-family:inherit;width:100%;position:relative;display:flex"
|
|
1061
1083
|
} : {
|
|
1062
1084
|
name: "1w49f4-MoneyInput",
|
|
1063
|
-
styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAwxBgB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
1085
|
+
styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAwxBgB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */",
|
|
1064
1086
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1065
1087
|
};
|
|
1066
1088
|
const MoneyInput = _ref5 => {
|
|
@@ -1298,7 +1320,7 @@ const MoneyInput = _ref5 => {
|
|
|
1298
1320
|
hasFocus
|
|
1299
1321
|
})),
|
|
1300
1322
|
// accounts for size of icon
|
|
1301
|
-
props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.designTokens.spacing40, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAm2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), currencyHasFocus && !props.isDisabled && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAy2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA81BY","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"],
|
|
1323
|
+
props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.designTokens.spacing40, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAm2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), currencyHasFocus && !props.isDisabled && !props.isReadOnly && /*#__PURE__*/react.css("border-left-color:", designSystem.designTokens.borderColorForInputWhenFocused, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAy2BmB","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA81BY","file":"money-input.tsx","sourcesContent":["import {\n  useRef,\n  useCallback,\n  type ReactNode,\n  type ComponentType,\n  type ChangeEvent,\n  type FocusEvent,\n} from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { designTokens } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n  createSequentialId,\n} from '@commercetools-uikit/utils';\nimport Tooltip, { type TTooltipProps } from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n  warnIfMenuPortalPropsAreMissing,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport allCurrencies from './currencies';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst moneyInputSequentialId = createSequentialId('money-input-');\n\nconst getPortalId = (id?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles(props)}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (\n  input: TInputProps & {\n    currencyHasFocus?: boolean;\n  }\n) => void;\n\nexport type TInputProps = {\n  isCondensed?: boolean;\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  menuPortalZIndex?: number;\n  /** @deprecated */\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = ({\n  hasWarning,\n  hasError,\n  isCondensed,\n  isDisabled,\n  isReadOnly,\n  menuPortalZIndex,\n  currencyHasFocus,\n}) => {\n  const selectStyles = createSelectStyles({\n    hasWarning,\n    hasError,\n    menuPortalZIndex,\n    isCondensed,\n    isReadOnly,\n    isDisabled,\n  });\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      padding: `0 ${designTokens.spacing25}`,\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '80px',\n      height: '100%',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${designTokens.borderColorForInputWhenDisabled} !important`;\n        if (isReadOnly)\n          return `${designTokens.borderColorForInputWhenReadonly} !important`;\n        if (hasError) return designTokens.borderColorForInputWhenError;\n        if (hasWarning) return designTokens.borderColorForInputWhenWarning;\n        if (currencyHasFocus) {\n          return designTokens.borderColorForInputWhenFocused;\n        }\n        return designTokens.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return designTokens.backgroundColorForInputWhenReadonly;\n        return base.backgroundColor;\n      })(),\n      '&:hover': {\n        borderColor: designTokens.borderColorForInput,\n      },\n      '&:hover:not(:read-only):not(:disabled)': {\n        backgroundColor: designTokens.backgroundColorForInputWhenHovered,\n      },\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return designTokens.fontColorForInputWhenDisabled;\n        if (hasError) return designTokens.fontColorForInputWhenError;\n        if (hasWarning) return designTokens.fontColorForInputWhenWarning;\n        if (isReadOnly) return designTokens.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly\n        ? designTokens.fontColorForInputWhenReadonly\n        : designTokens.colorNeutral40,\n    }),\n  };\n};\n\nexport type TCurrencyCode = keyof typeof allCurrencies;\n\ntype TMoneyConditionalProps =\n  | { type: 'highPrecision'; preciseAmount: number }\n  | {\n      /**\n       * Usually either a `centPrecision` or a `highPrecision`.\n       */\n      type: 'centPrecision';\n      preciseAmount?: never;\n    };\nexport type TMoneyValue = {\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  fractionDigits: number;\n} & TMoneyConditionalProps;\n// The MoneyInput component always operates on a value consisting of:\n// ```\n// { amount: String, currencyCode: String }\n// ```\n//\n// The amount may only use a dot as the decimal separator.\n// The `currencyCode` must be supported by the API.\n//\n// The `MoneyInput` does not do any validation on its own. It only serves as a way\n// to get the amount and `currencyCode` input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports two types of prices: `centPrecision` and `highPrecision`.\n// The `MoneyInput` itself does not know about these. However,\n// it has two static methods defined (`convertToMoneyValue` and `parseMoneyValue`),\n// which can be used to convert between `MoneyInput` value and the `MoneyValue`\n// supported by the API.\n// Some places in the API do not support `highPrecision` prices, but the\n// `convertToMoneyValue `will always return either a `centPrecision `or a\n// `highPrecision` price. It's up the `MoneyInput`'s parent to show a validation\n// error in case a `highPrecision` price is used.\n//\n// A value is considered as to have `highPrecision` when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, `EUR 42.00` is always a `centPrecision` price, while `EUR 42.001` is always a\n// `highPrecision` price. It is not possible to have `EUR 42.00` as a `highPrecision`\n// price.\n//\n// The first time the component renders, we want to try to show the `centAmount`\n// as a formatted number. To achieve this, the `parseMoneyValue` function can\n// be used to turn the API value into a value the `MoneyInput` understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an `MoneyValue` with `centPrecision` would be\n// ```\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// ```\n// which equals `EUR 42.00`.\n//\n// A full example of an `MoneyValue` with `highPrecision` would be\n// ```\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// ```\n// which equals `EUR 0.0123456`\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^-0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\nexport const createMoneyValue = (\n  rawAmount: string,\n  locale: string,\n  currencyCode?: TCurrencyCode | ''\n): TMoneyValue | null => {\n  if (!currencyCode) return null;\n\n  const currency = allCurrencies[currencyCode];\n  if (!currency) return null;\n  // The user may enter a value with a comma, dot, or apostrophe as the decimal separator.\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `MoneyInput: A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\nconst createEmptyMoneyValue = (currencyCode: TCurrencyCode): TMoneyValue => ({\n  type: 'centPrecision',\n  currencyCode,\n  centAmount: NaN,\n  fractionDigits: allCurrencies[currencyCode].fractionDigits ?? 2,\n});\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  locale: string,\n  currencyCode: TCurrencyCode\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue =\n    createMoneyValue(rawAmount, locale, currencyCode) ||\n    createEmptyMoneyValue(currencyCode);\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : allCurrencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\nexport type TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode | '';\n};\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies?: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TCustomEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TCustomEvent) => void;\n  /**\n   * Use this property to reduce the paddings of the component for a ui compact variant\n   */\n  isCondensed?: boolean;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   * <br>\n   * Use in conjunction with `menuPortalTarget`\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll?: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Indicates that the currency input cannot be modified.\n   */\n  isCurrencyInputDisabled?: boolean;\n};\n\nconst MoneyInput = ({\n  currencies = [],\n  horizontalConstraint = 'scale',\n  menuPortalZIndex = 1,\n  ...props\n}: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  const moneyInputId = useFieldId(props.id, moneyInputSequentialId);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  warnIfMenuPortalPropsAreMissing({\n    menuPortalZIndex: menuPortalZIndex,\n    menuPortalTarget: props.menuPortalTarget,\n    componentName: 'MoneyInput',\n  });\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(moneyInputId),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, moneyInputId, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (\n      amount.length > 0 &&\n      props.value.currencyCode &&\n      allCurrencies[props.value.currencyCode]\n    ) {\n      const formattedAmount = formatAmount(\n        amount,\n        intl.locale,\n        props.value.currencyCode\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange?.(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    moneyInputId,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event: ChangeEvent | FocusEvent) => {\n      if (isNumberish((event.target as HTMLInputElement)?.value)) {\n        onChange?.({\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n            value: (event.target as HTMLInputElement)?.value,\n          },\n        });\n      }\n    },\n    [onChange, moneyInputId, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option: { value: TCurrencyCode }) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          intl.locale,\n          currencyCode\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n            value: currencyCode || '',\n          },\n        };\n        onChange?.(fakeCurrencyEvent);\n\n        // change amount if necessary\n        if (props.value.amount !== nextAmount) {\n          onChange?.({\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(moneyInputId),\n              name: getAmountInputName(props.name),\n              value: nextAmount,\n            },\n          });\n        }\n\n        amountInputRef.current?.focus();\n      }\n    },\n    [\n      intl.locale,\n      onChange,\n      moneyInputId,\n      props.name,\n      props.value.amount,\n      props.value.currencyCode,\n    ]\n  );\n\n  const handleCurrencyFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, moneyInputId]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const currencySelectStyles = createCurrencySelectStyles({\n    hasWarning: props.hasWarning,\n    hasError: props.hasError,\n    isCondensed: props.isCondensed,\n    isDisabled: props.isDisabled,\n    isReadOnly: props.isReadOnly,\n    menuPortalZIndex: menuPortalZIndex,\n    currencyHasFocus,\n  });\n  const options = currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event: FocusEvent) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current?.contains(event.relatedTarget as Node)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(moneyInputId),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(moneyInputId),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, moneyInputId, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps: TTooltipProps & { id: string }) => (\n      <Portal {...remainingProps} id={props.id!} />\n    ),\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={(event) =>\n          handleContainerBlur(event as FocusEvent<HTMLDivElement>)\n        }\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(moneyInputId) as string}\n            isCondensed={props.isCondensed}\n            isDisabled={props.isDisabled}\n            isReadOnly={props.isReadOnly}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isCurrencyInputDisabled || props.isDisabled}\n            isSearchable={false}\n            components={\n              {\n                SingleValue: (innerProps) => (\n                  <SingleValue\n                    {...innerProps}\n                    id={MoneyInput.getCurrencyDropdownId(moneyInputId)}\n                  />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator: props.isCurrencyInputDisabled\n                  ? null\n                  : DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            menuPlacement=\"auto\"\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange as ReactSelectProps['onChange']}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(moneyInputId)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${designTokens.spacing40};\n                `,\n              currencyHasFocus &&\n                !props.isDisabled &&\n                !props.isReadOnly &&\n                css`\n                  border-left-color: ${designTokens.borderColorForInputWhenFocused};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes({\n              horizontalConstraint,\n              menuPortalZIndex,\n              ...props,\n            })}\n            /* ARIA */\n            aria-invalid={props['aria-invalid']}\n            aria-errormessage={props['aria-errormessage']}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${designTokens.spacing20} -${designTokens.spacing10} ${designTokens.spacing20} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal as ComponentType,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value: TValue, locale: string) =>\n  createMoneyValue(\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale,\n    value.currencyCode\n  );\n\nMoneyInput.parseMoneyValue = (\n  moneyValue: TMoneyValue,\n  locale: string\n): TValue => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(allCurrencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    locale,\n    moneyValue.currencyCode\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string): boolean => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue?.type === 'highPrecision';\n};\n\ntype TTouched = {\n  amount?: boolean;\n  currencyCode?: boolean;\n};\n\nMoneyInput.isTouched = (touched?: TTouched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nexport default MoneyInput;\n"]} */"],
|
|
1302
1324
|
placeholder: props.placeholder,
|
|
1303
1325
|
onChange: handleAmountChange,
|
|
1304
1326
|
onBlur: handleAmountBlur,
|
|
@@ -1385,7 +1407,7 @@ MoneyInput.isTouched = touched => Boolean(touched && touched.currencyCode && tou
|
|
|
1385
1407
|
var MoneyInput$1 = MoneyInput;
|
|
1386
1408
|
|
|
1387
1409
|
// NOTE: This string will be replaced on build time with the package version.
|
|
1388
|
-
var version = "20.
|
|
1410
|
+
var version = "20.5.0";
|
|
1389
1411
|
|
|
1390
1412
|
exports["default"] = MoneyInput$1;
|
|
1391
1413
|
exports.version = version;
|