@commercetools-uikit/localized-money-input 14.0.1 → 14.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -56,10 +56,10 @@ export default Example;
56
56
  | `id` | `string` | | | Used as HTML id property. |
57
57
  | `name` | `string` | | | The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`). |
58
58
  | `value` | `Record` | ✅ | | value of possible currency&#xA;<br/>&#xA;the input doesn't accept a "currencies" prop, instead all possible&#xA;currencies have to exist (with empty or filled strings) on the value:&#xA;{ EUR: {amount: '12.00', currencyCode: 'EUR'}, USD: {amount: '', currencyCode: 'USD'}} |
59
- | `onChange` | `ChangeEventHandler` | | | Called with the event of the input. |
59
+ | `onChange` | `Function`<br/>[See signature.](#signature-onChange) | | | Called with the event of the input. |
60
60
  | `selectedCurrency` | `string` | ✅ | | the currently selected currency |
61
61
  | `onBlur` | `Function`<br/>[See signature.](#signature-onBlur) | | | Called when input is blurred |
62
- | `onFocus` | `FocusEventHandler` | | | Called when input is focused |
62
+ | `onFocus` | `Function`<br/>[See signature.](#signature-onFocus) | | | Called when input is focused |
63
63
  | `hideCurrencyExpansionControls` | `boolean` | | | Will hide the currency expansion controls when set to `true`. All currencies will be shown when set to `true`. |
64
64
  | `defaultExpandCurrencies` | `boolean` | | | Controls whether one or all currencirs are visible by default |
65
65
  | `isDisabled` | `boolean` | | | Indicates that the input cannot be modified (e.g not authorized, or changes currently saving). |
@@ -73,10 +73,22 @@ export default Example;
73
73
 
74
74
  ## Signatures
75
75
 
76
+ ### Signature `onChange`
77
+
78
+ ```ts
79
+ (event: TCustomEvent) => void
80
+ ```
81
+
76
82
  ### Signature `onBlur`
77
83
 
78
84
  ```ts
79
- (event: TEvent) => void
85
+ (event: TCustomEvent) => void
86
+ ```
87
+
88
+ ### Signature `onFocus`
89
+
90
+ ```ts
91
+ (event: TCustomEvent) => void
80
92
  ```
81
93
 
82
94
  ## Static methods
@@ -95,7 +95,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
95
95
  } : {
96
96
  name: "1ysn02y-LocalizedInput",
97
97
  styles: "width:100%;position:relative;display:flex;label:LocalizedInput;",
98
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-money-input.tsx"],"names":[],"mappings":"AAwOgB","file":"localized-money-input.tsx","sourcesContent":["import {\n  useCallback,\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  type ReactNode,\n} from 'react';\nimport { useIntl } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useToggleState, useFieldId } from '@commercetools-uikit/hooks';\nimport MoneyInput, {\n  type TCurrencyCode,\n  type TMoneyValue,\n  type TValue,\n} from '@commercetools-uikit/money-input';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { CoinsIcon } from '@commercetools-uikit/icons';\nimport {\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport messages from './messages';\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n};\n\ntype TLocalizedMoneyInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   * <br/>\n   * the input doesn't accept a \"currencies\" prop, instead all possible\n   * currencies have to exist (with empty or filled strings) on the value:\n   * { EUR: {amount: '12.00', currencyCode: 'EUR'}, USD: {amount: '', currencyCode: 'USD'}}\n   */\n  value: Record<string, TValue>;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: ChangeEventHandler;\n  /**\n   * the currently selected currency\n   */\n  selectedCurrency: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: FocusEventHandler;\n  /**\n   * Will hide the currency expansion controls when set to `true`. All currencies will be shown when set to `true`.\n   */\n  hideCurrencyExpansionControls?: boolean;\n  /**\n   * Controls whether one or all currencirs are visible by default\n   */\n  defaultExpandCurrencies?: 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   * Placeholder text for the input\n   */\n  placeholder?: Record<string, string>;\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 input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  errors?: Record<string, ReactNode>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\ntype TLocalizedInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   */\n  value: TValue;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: ChangeEventHandler;\n  /**\n   * Called with the event of the input.\n   */\n  currency: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: FocusEventHandler;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  error?: ReactNode;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n};\n\nconst sequentialId = createSequentialId('localized-money-input-');\n\n// sorts the currencies with the following priority:\n// - The selected currency is placed first (e.g EUR)\n// - All other currencies follow, sorted alphabetically as well\nexport const sortCurrencies = (\n  selectedCurrency: string,\n  allCurrencies: string[]\n) => {\n  const remainingCurrencies = allCurrencies.filter(\n    (currency) => currency !== selectedCurrency\n  );\n  return [selectedCurrency, ...remainingCurrencies.sort()];\n};\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the currency to the target.\n      // That way the users  can read\n      // event.target.currency and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the currency from\n      // the input name anymore.\n      //\n      event.target.currency = props.currency;\n      onChange?.(event);\n    },\n    [props.currency, onChange]\n  );\n  return (\n    <Stack scale=\"xs\">\n      <div\n        css={css`\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n      >\n        <MoneyInput\n          name={props.name}\n          value={props.value}\n          onChange={handleChange}\n          onBlur={props.onBlur}\n          isDisabled={props.isDisabled}\n          isReadOnly={props.isReadOnly}\n          placeholder={props.placeholder}\n          hasError={props.hasError}\n          hasWarning={props.hasWarning}\n          {...filterDataAttributes(props)}\n        />\n      </div>\n      {(props.error || props.warning) && (\n        <div>{props.error ? props.error : props.warning}</div>\n      )}\n    </Stack>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst LocalizedMoneyInput = (props: TLocalizedMoneyInputProps) => {\n  const intl = useIntl();\n\n  const defaultExpansionState =\n    props.hideCurrencyExpansionControls ||\n    props.defaultExpandCurrencies ||\n    // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areCurrenciesExpanded, toggleCurrencies] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedMoneyInputToggle = useCallback(\n    () => toggleCurrencies(),\n    [toggleCurrencies]\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingCurrencies =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedCurrency);\n\n  const hasWarningInRemainingCurrencies =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedCurrency);\n\n  if (hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areCurrenciesExpanded) {\n      toggleCurrencies();\n    }\n  }\n\n  const currencies = sortCurrencies(\n    props.selectedCurrency,\n    Object.keys(props.value)\n  );\n\n  const shouldRenderCurrencyControl =\n    currencies.length > 1 && !props.hideCurrencyExpansionControls;\n\n  if (props.hideCurrencyExpansionControls) {\n    warning(\n      typeof props.defaultExpandCurrencies !== 'boolean',\n      'LocalizedMoneyInput: \"defaultExpandCurrencies\" does not have any effect when \"hideCurrencyExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack scale=\"s\">\n          {currencies.map((currency, index) => {\n            const isFirstCurrency = index === 0;\n            if (!isFirstCurrency && !areCurrenciesExpanded) return null;\n\n            return (\n              <LocalizedInput\n                key={currency}\n                id={LocalizedMoneyInput.getId(id, currency)}\n                name={LocalizedMoneyInput.getName(props.name, currency)}\n                value={props.value[currency]}\n                onChange={props.onChange}\n                currency={currency}\n                placeholder={\n                  props.placeholder ? props.placeholder[currency] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[currency])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[currency])\n                )}\n                warning={props.warnings && props.warnings[currency]}\n                error={props.errors && props.errors[currency]}\n                {...createLocalizedDataAttributes(props, currency)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderCurrencyControl && (\n          <LocalizedInputToggle\n            icon={<CoinsIcon />}\n            onClick={onLocalizedMoneyInputToggle}\n            isOpen={areCurrenciesExpanded}\n            isDisabled={\n              areCurrenciesExpanded &&\n              Boolean(\n                hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies\n              )\n            }\n            showMessage={intl.formatMessage(messages.show, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n            hideMessage={intl.formatMessage(messages.hide, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedMoneyInput.displayName = 'LocalizedMoneyInput';\n\nLocalizedMoneyInput.getId = getId;\n\nLocalizedMoneyInput.getName = getName;\n\nLocalizedMoneyInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedMoneyInput.convertToMoneyValues = (\n  values: TValue[],\n  locale: string\n): Array<TMoneyValue | null> =>\n  Object.values(values).map<TMoneyValue | null>((value) => {\n    return MoneyInput.convertToMoneyValue(value, locale);\n  });\n\nLocalizedMoneyInput.parseMoneyValues = (\n  moneyValues: TMoneyValue[] = [],\n  locale: string\n): Record<TCurrencyCode, TValue> =>\n  moneyValues.reduce<Record<TCurrencyCode, TValue>>((allValues, moneyValue) => {\n    const value = MoneyInput.parseMoneyValue(moneyValue, locale);\n    return {\n      ...allValues,\n      [value.currencyCode]: value,\n    };\n  }, {} as Record<TCurrencyCode, TValue>);\n\nLocalizedMoneyInput.getHighPrecisionCurrencies = (\n  values: Record<TCurrencyCode, TValue>,\n  locale: string\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isHighPrecision(values[currencyCode], locale)\n  );\n};\n\nLocalizedMoneyInput.getEmptyCurrencies = (\n  values: Record<TCurrencyCode, TValue>\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isEmpty(values[currencyCode])\n  );\n};\n\nexport default LocalizedMoneyInput;\n"]} */",
98
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-money-input.tsx"],"names":[],"mappings":"AAoOgB","file":"localized-money-input.tsx","sourcesContent":["import { useCallback, type ReactNode } from 'react';\nimport { useIntl } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useToggleState, useFieldId } from '@commercetools-uikit/hooks';\nimport MoneyInput, {\n  type TCurrencyCode,\n  type TMoneyValue,\n  type TValue,\n} from '@commercetools-uikit/money-input';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { CoinsIcon } from '@commercetools-uikit/icons';\nimport {\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport messages from './messages';\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TLocalizedMoneyInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   * <br/>\n   * the input doesn't accept a \"currencies\" prop, instead all possible\n   * currencies have to exist (with empty or filled strings) on the value:\n   * { EUR: {amount: '12.00', currencyCode: 'EUR'}, USD: {amount: '', currencyCode: 'USD'}}\n   */\n  value: Record<string, TValue>;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * the currently selected currency\n   */\n  selectedCurrency: 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   * Will hide the currency expansion controls when set to `true`. All currencies will be shown when set to `true`.\n   */\n  hideCurrencyExpansionControls?: boolean;\n  /**\n   * Controls whether one or all currencirs are visible by default\n   */\n  defaultExpandCurrencies?: 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   * Placeholder text for the input\n   */\n  placeholder?: Record<string, string>;\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 input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  errors?: Record<string, ReactNode>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\ntype TLocalizedInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   */\n  value: TValue;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called with the event of the input.\n   */\n  currency: 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   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  error?: ReactNode;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n};\n\nconst sequentialId = createSequentialId('localized-money-input-');\n\n// sorts the currencies with the following priority:\n// - The selected currency is placed first (e.g EUR)\n// - All other currencies follow, sorted alphabetically as well\nexport const sortCurrencies = (\n  selectedCurrency: string,\n  allCurrencies: string[]\n) => {\n  const remainingCurrencies = allCurrencies.filter(\n    (currency) => currency !== selectedCurrency\n  );\n  return [selectedCurrency, ...remainingCurrencies.sort()];\n};\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the currency to the target.\n      // That way the users  can read\n      // event.target.currency and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the currency from\n      // the input name anymore.\n      //\n      event.target.currency = props.currency;\n      onChange?.(event);\n    },\n    [props.currency, onChange]\n  );\n  return (\n    <Stack scale=\"xs\">\n      <div\n        css={css`\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n      >\n        <MoneyInput\n          name={props.name}\n          value={props.value}\n          onChange={handleChange}\n          onBlur={props.onBlur}\n          isDisabled={props.isDisabled}\n          isReadOnly={props.isReadOnly}\n          placeholder={props.placeholder}\n          hasError={props.hasError}\n          hasWarning={props.hasWarning}\n          {...filterDataAttributes(props)}\n        />\n      </div>\n      {(props.error || props.warning) && (\n        <div>{props.error ? props.error : props.warning}</div>\n      )}\n    </Stack>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst LocalizedMoneyInput = (props: TLocalizedMoneyInputProps) => {\n  const intl = useIntl();\n\n  const defaultExpansionState =\n    props.hideCurrencyExpansionControls ||\n    props.defaultExpandCurrencies ||\n    // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areCurrenciesExpanded, toggleCurrencies] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedMoneyInputToggle = useCallback(\n    () => toggleCurrencies(),\n    [toggleCurrencies]\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingCurrencies =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedCurrency);\n\n  const hasWarningInRemainingCurrencies =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedCurrency);\n\n  if (hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areCurrenciesExpanded) {\n      toggleCurrencies();\n    }\n  }\n\n  const currencies = sortCurrencies(\n    props.selectedCurrency,\n    Object.keys(props.value)\n  );\n\n  const shouldRenderCurrencyControl =\n    currencies.length > 1 && !props.hideCurrencyExpansionControls;\n\n  if (props.hideCurrencyExpansionControls) {\n    warning(\n      typeof props.defaultExpandCurrencies !== 'boolean',\n      'LocalizedMoneyInput: \"defaultExpandCurrencies\" does not have any effect when \"hideCurrencyExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack scale=\"s\">\n          {currencies.map((currency, index) => {\n            const isFirstCurrency = index === 0;\n            if (!isFirstCurrency && !areCurrenciesExpanded) return null;\n\n            return (\n              <LocalizedInput\n                key={currency}\n                id={LocalizedMoneyInput.getId(id, currency)}\n                name={LocalizedMoneyInput.getName(props.name, currency)}\n                value={props.value[currency]}\n                onChange={props.onChange}\n                currency={currency}\n                placeholder={\n                  props.placeholder ? props.placeholder[currency] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[currency])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[currency])\n                )}\n                warning={props.warnings && props.warnings[currency]}\n                error={props.errors && props.errors[currency]}\n                {...createLocalizedDataAttributes(props, currency)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderCurrencyControl && (\n          <LocalizedInputToggle\n            icon={<CoinsIcon />}\n            onClick={onLocalizedMoneyInputToggle}\n            isOpen={areCurrenciesExpanded}\n            isDisabled={\n              areCurrenciesExpanded &&\n              Boolean(\n                hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies\n              )\n            }\n            showMessage={intl.formatMessage(messages.show, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n            hideMessage={intl.formatMessage(messages.hide, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedMoneyInput.displayName = 'LocalizedMoneyInput';\n\nLocalizedMoneyInput.getId = getId;\n\nLocalizedMoneyInput.getName = getName;\n\nLocalizedMoneyInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedMoneyInput.convertToMoneyValues = (\n  values: TValue[],\n  locale: string\n): Array<TMoneyValue | null> =>\n  Object.values(values).map<TMoneyValue | null>((value) => {\n    return MoneyInput.convertToMoneyValue(value, locale);\n  });\n\nLocalizedMoneyInput.parseMoneyValues = (\n  moneyValues: TMoneyValue[] = [],\n  locale: string\n): Record<TCurrencyCode, TValue> =>\n  moneyValues.reduce<Record<TCurrencyCode, TValue>>((allValues, moneyValue) => {\n    const value = MoneyInput.parseMoneyValue(moneyValue, locale);\n    return {\n      ...allValues,\n      [value.currencyCode]: value,\n    };\n  }, {} as Record<TCurrencyCode, TValue>);\n\nLocalizedMoneyInput.getHighPrecisionCurrencies = (\n  values: Record<TCurrencyCode, TValue>,\n  locale: string\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isHighPrecision(values[currencyCode], locale)\n  );\n};\n\nLocalizedMoneyInput.getEmptyCurrencies = (\n  values: Record<TCurrencyCode, TValue>\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isEmpty(values[currencyCode])\n  );\n};\n\nexport default LocalizedMoneyInput;\n"]} */",
99
99
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
100
100
  };
101
101
 
@@ -291,7 +291,7 @@ LocalizedMoneyInput.getEmptyCurrencies = function (values) {
291
291
  var LocalizedMoneyInput$1 = LocalizedMoneyInput;
292
292
 
293
293
  // NOTE: This string will be replaced on build time with the package version.
294
- var version = "14.0.1";
294
+ var version = "14.0.6";
295
295
 
296
296
  exports["default"] = LocalizedMoneyInput$1;
297
297
  exports.version = version;
@@ -248,7 +248,7 @@ LocalizedMoneyInput.getEmptyCurrencies = function (values) {
248
248
  var LocalizedMoneyInput$1 = LocalizedMoneyInput;
249
249
 
250
250
  // NOTE: This string will be replaced on build time with the package version.
251
- var version = "14.0.1";
251
+ var version = "14.0.6";
252
252
 
253
253
  exports["default"] = LocalizedMoneyInput$1;
254
254
  exports.version = version;
@@ -71,7 +71,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
71
71
  } : {
72
72
  name: "1ysn02y-LocalizedInput",
73
73
  styles: "width:100%;position:relative;display:flex;label:LocalizedInput;",
74
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-money-input.tsx"],"names":[],"mappings":"AAwOgB","file":"localized-money-input.tsx","sourcesContent":["import {\n  useCallback,\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  type ReactNode,\n} from 'react';\nimport { useIntl } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useToggleState, useFieldId } from '@commercetools-uikit/hooks';\nimport MoneyInput, {\n  type TCurrencyCode,\n  type TMoneyValue,\n  type TValue,\n} from '@commercetools-uikit/money-input';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { CoinsIcon } from '@commercetools-uikit/icons';\nimport {\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport messages from './messages';\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n};\n\ntype TLocalizedMoneyInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   * <br/>\n   * the input doesn't accept a \"currencies\" prop, instead all possible\n   * currencies have to exist (with empty or filled strings) on the value:\n   * { EUR: {amount: '12.00', currencyCode: 'EUR'}, USD: {amount: '', currencyCode: 'USD'}}\n   */\n  value: Record<string, TValue>;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: ChangeEventHandler;\n  /**\n   * the currently selected currency\n   */\n  selectedCurrency: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: FocusEventHandler;\n  /**\n   * Will hide the currency expansion controls when set to `true`. All currencies will be shown when set to `true`.\n   */\n  hideCurrencyExpansionControls?: boolean;\n  /**\n   * Controls whether one or all currencirs are visible by default\n   */\n  defaultExpandCurrencies?: 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   * Placeholder text for the input\n   */\n  placeholder?: Record<string, string>;\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 input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  errors?: Record<string, ReactNode>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\ntype TLocalizedInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   */\n  value: TValue;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: ChangeEventHandler;\n  /**\n   * Called with the event of the input.\n   */\n  currency: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: FocusEventHandler;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  error?: ReactNode;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n};\n\nconst sequentialId = createSequentialId('localized-money-input-');\n\n// sorts the currencies with the following priority:\n// - The selected currency is placed first (e.g EUR)\n// - All other currencies follow, sorted alphabetically as well\nexport const sortCurrencies = (\n  selectedCurrency: string,\n  allCurrencies: string[]\n) => {\n  const remainingCurrencies = allCurrencies.filter(\n    (currency) => currency !== selectedCurrency\n  );\n  return [selectedCurrency, ...remainingCurrencies.sort()];\n};\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the currency to the target.\n      // That way the users  can read\n      // event.target.currency and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the currency from\n      // the input name anymore.\n      //\n      event.target.currency = props.currency;\n      onChange?.(event);\n    },\n    [props.currency, onChange]\n  );\n  return (\n    <Stack scale=\"xs\">\n      <div\n        css={css`\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n      >\n        <MoneyInput\n          name={props.name}\n          value={props.value}\n          onChange={handleChange}\n          onBlur={props.onBlur}\n          isDisabled={props.isDisabled}\n          isReadOnly={props.isReadOnly}\n          placeholder={props.placeholder}\n          hasError={props.hasError}\n          hasWarning={props.hasWarning}\n          {...filterDataAttributes(props)}\n        />\n      </div>\n      {(props.error || props.warning) && (\n        <div>{props.error ? props.error : props.warning}</div>\n      )}\n    </Stack>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst LocalizedMoneyInput = (props: TLocalizedMoneyInputProps) => {\n  const intl = useIntl();\n\n  const defaultExpansionState =\n    props.hideCurrencyExpansionControls ||\n    props.defaultExpandCurrencies ||\n    // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areCurrenciesExpanded, toggleCurrencies] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedMoneyInputToggle = useCallback(\n    () => toggleCurrencies(),\n    [toggleCurrencies]\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingCurrencies =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedCurrency);\n\n  const hasWarningInRemainingCurrencies =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedCurrency);\n\n  if (hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areCurrenciesExpanded) {\n      toggleCurrencies();\n    }\n  }\n\n  const currencies = sortCurrencies(\n    props.selectedCurrency,\n    Object.keys(props.value)\n  );\n\n  const shouldRenderCurrencyControl =\n    currencies.length > 1 && !props.hideCurrencyExpansionControls;\n\n  if (props.hideCurrencyExpansionControls) {\n    warning(\n      typeof props.defaultExpandCurrencies !== 'boolean',\n      'LocalizedMoneyInput: \"defaultExpandCurrencies\" does not have any effect when \"hideCurrencyExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack scale=\"s\">\n          {currencies.map((currency, index) => {\n            const isFirstCurrency = index === 0;\n            if (!isFirstCurrency && !areCurrenciesExpanded) return null;\n\n            return (\n              <LocalizedInput\n                key={currency}\n                id={LocalizedMoneyInput.getId(id, currency)}\n                name={LocalizedMoneyInput.getName(props.name, currency)}\n                value={props.value[currency]}\n                onChange={props.onChange}\n                currency={currency}\n                placeholder={\n                  props.placeholder ? props.placeholder[currency] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[currency])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[currency])\n                )}\n                warning={props.warnings && props.warnings[currency]}\n                error={props.errors && props.errors[currency]}\n                {...createLocalizedDataAttributes(props, currency)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderCurrencyControl && (\n          <LocalizedInputToggle\n            icon={<CoinsIcon />}\n            onClick={onLocalizedMoneyInputToggle}\n            isOpen={areCurrenciesExpanded}\n            isDisabled={\n              areCurrenciesExpanded &&\n              Boolean(\n                hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies\n              )\n            }\n            showMessage={intl.formatMessage(messages.show, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n            hideMessage={intl.formatMessage(messages.hide, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedMoneyInput.displayName = 'LocalizedMoneyInput';\n\nLocalizedMoneyInput.getId = getId;\n\nLocalizedMoneyInput.getName = getName;\n\nLocalizedMoneyInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedMoneyInput.convertToMoneyValues = (\n  values: TValue[],\n  locale: string\n): Array<TMoneyValue | null> =>\n  Object.values(values).map<TMoneyValue | null>((value) => {\n    return MoneyInput.convertToMoneyValue(value, locale);\n  });\n\nLocalizedMoneyInput.parseMoneyValues = (\n  moneyValues: TMoneyValue[] = [],\n  locale: string\n): Record<TCurrencyCode, TValue> =>\n  moneyValues.reduce<Record<TCurrencyCode, TValue>>((allValues, moneyValue) => {\n    const value = MoneyInput.parseMoneyValue(moneyValue, locale);\n    return {\n      ...allValues,\n      [value.currencyCode]: value,\n    };\n  }, {} as Record<TCurrencyCode, TValue>);\n\nLocalizedMoneyInput.getHighPrecisionCurrencies = (\n  values: Record<TCurrencyCode, TValue>,\n  locale: string\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isHighPrecision(values[currencyCode], locale)\n  );\n};\n\nLocalizedMoneyInput.getEmptyCurrencies = (\n  values: Record<TCurrencyCode, TValue>\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isEmpty(values[currencyCode])\n  );\n};\n\nexport default LocalizedMoneyInput;\n"]} */",
74
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-money-input.tsx"],"names":[],"mappings":"AAoOgB","file":"localized-money-input.tsx","sourcesContent":["import { useCallback, type ReactNode } from 'react';\nimport { useIntl } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useToggleState, useFieldId } from '@commercetools-uikit/hooks';\nimport MoneyInput, {\n  type TCurrencyCode,\n  type TMoneyValue,\n  type TValue,\n} from '@commercetools-uikit/money-input';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { CoinsIcon } from '@commercetools-uikit/icons';\nimport {\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport {\n  createSequentialId,\n  filterDataAttributes,\n  warning,\n} from '@commercetools-uikit/utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport messages from './messages';\n\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TLocalizedMoneyInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   * <br/>\n   * the input doesn't accept a \"currencies\" prop, instead all possible\n   * currencies have to exist (with empty or filled strings) on the value:\n   * { EUR: {amount: '12.00', currencyCode: 'EUR'}, USD: {amount: '', currencyCode: 'USD'}}\n   */\n  value: Record<string, TValue>;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * the currently selected currency\n   */\n  selectedCurrency: 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   * Will hide the currency expansion controls when set to `true`. All currencies will be shown when set to `true`.\n   */\n  hideCurrencyExpansionControls?: boolean;\n  /**\n   * Controls whether one or all currencirs are visible by default\n   */\n  defaultExpandCurrencies?: 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   * Placeholder text for the input\n   */\n  placeholder?: Record<string, string>;\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 input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  errors?: Record<string, ReactNode>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\ntype TLocalizedInputProps = {\n  /**\n   * Used as HTML id property.\n   */\n  id?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`).\n   */\n  name?: string;\n  /**\n   * value of possible currency\n   */\n  value: TValue;\n  /**\n   * Called with the event of the input.\n   */\n  onChange?: (event: TCustomEvent) => void;\n  /**\n   * Called with the event of the input.\n   */\n  currency: 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   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * A map of errors. Error messages for known errors are rendered automatically.\n   * <br>\n   * Unknown errors will be forwarded to `renderError`\n   */\n  error?: ReactNode;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n};\n\nconst sequentialId = createSequentialId('localized-money-input-');\n\n// sorts the currencies with the following priority:\n// - The selected currency is placed first (e.g EUR)\n// - All other currencies follow, sorted alphabetically as well\nexport const sortCurrencies = (\n  selectedCurrency: string,\n  allCurrencies: string[]\n) => {\n  const remainingCurrencies = allCurrencies.filter(\n    (currency) => currency !== selectedCurrency\n  );\n  return [selectedCurrency, ...remainingCurrencies.sort()];\n};\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the currency to the target.\n      // That way the users  can read\n      // event.target.currency and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the currency from\n      // the input name anymore.\n      //\n      event.target.currency = props.currency;\n      onChange?.(event);\n    },\n    [props.currency, onChange]\n  );\n  return (\n    <Stack scale=\"xs\">\n      <div\n        css={css`\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n      >\n        <MoneyInput\n          name={props.name}\n          value={props.value}\n          onChange={handleChange}\n          onBlur={props.onBlur}\n          isDisabled={props.isDisabled}\n          isReadOnly={props.isReadOnly}\n          placeholder={props.placeholder}\n          hasError={props.hasError}\n          hasWarning={props.hasWarning}\n          {...filterDataAttributes(props)}\n        />\n      </div>\n      {(props.error || props.warning) && (\n        <div>{props.error ? props.error : props.warning}</div>\n      )}\n    </Stack>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst LocalizedMoneyInput = (props: TLocalizedMoneyInputProps) => {\n  const intl = useIntl();\n\n  const defaultExpansionState =\n    props.hideCurrencyExpansionControls ||\n    props.defaultExpandCurrencies ||\n    // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areCurrenciesExpanded, toggleCurrencies] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedMoneyInputToggle = useCallback(\n    () => toggleCurrencies(),\n    [toggleCurrencies]\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingCurrencies =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedCurrency);\n\n  const hasWarningInRemainingCurrencies =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedCurrency);\n\n  if (hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areCurrenciesExpanded) {\n      toggleCurrencies();\n    }\n  }\n\n  const currencies = sortCurrencies(\n    props.selectedCurrency,\n    Object.keys(props.value)\n  );\n\n  const shouldRenderCurrencyControl =\n    currencies.length > 1 && !props.hideCurrencyExpansionControls;\n\n  if (props.hideCurrencyExpansionControls) {\n    warning(\n      typeof props.defaultExpandCurrencies !== 'boolean',\n      'LocalizedMoneyInput: \"defaultExpandCurrencies\" does not have any effect when \"hideCurrencyExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack scale=\"s\">\n          {currencies.map((currency, index) => {\n            const isFirstCurrency = index === 0;\n            if (!isFirstCurrency && !areCurrenciesExpanded) return null;\n\n            return (\n              <LocalizedInput\n                key={currency}\n                id={LocalizedMoneyInput.getId(id, currency)}\n                name={LocalizedMoneyInput.getName(props.name, currency)}\n                value={props.value[currency]}\n                onChange={props.onChange}\n                currency={currency}\n                placeholder={\n                  props.placeholder ? props.placeholder[currency] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[currency])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[currency])\n                )}\n                warning={props.warnings && props.warnings[currency]}\n                error={props.errors && props.errors[currency]}\n                {...createLocalizedDataAttributes(props, currency)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderCurrencyControl && (\n          <LocalizedInputToggle\n            icon={<CoinsIcon />}\n            onClick={onLocalizedMoneyInputToggle}\n            isOpen={areCurrenciesExpanded}\n            isDisabled={\n              areCurrenciesExpanded &&\n              Boolean(\n                hasErrorInRemainingCurrencies || hasWarningInRemainingCurrencies\n              )\n            }\n            showMessage={intl.formatMessage(messages.show, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n            hideMessage={intl.formatMessage(messages.hide, {\n              remainingCurrencies: currencies.length - 1,\n            })}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedMoneyInput.displayName = 'LocalizedMoneyInput';\n\nLocalizedMoneyInput.getId = getId;\n\nLocalizedMoneyInput.getName = getName;\n\nLocalizedMoneyInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedMoneyInput.convertToMoneyValues = (\n  values: TValue[],\n  locale: string\n): Array<TMoneyValue | null> =>\n  Object.values(values).map<TMoneyValue | null>((value) => {\n    return MoneyInput.convertToMoneyValue(value, locale);\n  });\n\nLocalizedMoneyInput.parseMoneyValues = (\n  moneyValues: TMoneyValue[] = [],\n  locale: string\n): Record<TCurrencyCode, TValue> =>\n  moneyValues.reduce<Record<TCurrencyCode, TValue>>((allValues, moneyValue) => {\n    const value = MoneyInput.parseMoneyValue(moneyValue, locale);\n    return {\n      ...allValues,\n      [value.currencyCode]: value,\n    };\n  }, {} as Record<TCurrencyCode, TValue>);\n\nLocalizedMoneyInput.getHighPrecisionCurrencies = (\n  values: Record<TCurrencyCode, TValue>,\n  locale: string\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isHighPrecision(values[currencyCode], locale)\n  );\n};\n\nLocalizedMoneyInput.getEmptyCurrencies = (\n  values: Record<TCurrencyCode, TValue>\n): TCurrencyCode[] => {\n  const typedCurrencyCodes = Object.keys(values) as TCurrencyCode[];\n  return typedCurrencyCodes.filter((currencyCode) =>\n    MoneyInput.isEmpty(values[currencyCode])\n  );\n};\n\nexport default LocalizedMoneyInput;\n"]} */",
75
75
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
76
76
  };
77
77
 
@@ -267,6 +267,6 @@ LocalizedMoneyInput.getEmptyCurrencies = function (values) {
267
267
  var LocalizedMoneyInput$1 = LocalizedMoneyInput;
268
268
 
269
269
  // NOTE: This string will be replaced on build time with the package version.
270
- var version = "14.0.1";
270
+ var version = "14.0.6";
271
271
 
272
272
  export { LocalizedMoneyInput$1 as default, version };
@@ -1,20 +1,21 @@
1
- import { type FocusEventHandler, type ChangeEventHandler, type ReactNode } from 'react';
1
+ import { type ReactNode } from 'react';
2
2
  import { type TCurrencyCode, type TMoneyValue, type TValue } from '@commercetools-uikit/money-input';
3
- declare type TEvent = {
3
+ declare type TCustomEvent = {
4
4
  target: {
5
5
  id?: string;
6
6
  name?: string;
7
7
  value?: string | string[] | null;
8
8
  };
9
+ persist?: () => void;
9
10
  };
10
11
  declare type TLocalizedMoneyInputProps = {
11
12
  id?: string;
12
13
  name?: string;
13
14
  value: Record<string, TValue>;
14
- onChange?: ChangeEventHandler;
15
+ onChange?: (event: TCustomEvent) => void;
15
16
  selectedCurrency: string;
16
- onBlur?: (event: TEvent) => void;
17
- onFocus?: FocusEventHandler;
17
+ onBlur?: (event: TCustomEvent) => void;
18
+ onFocus?: (event: TCustomEvent) => void;
18
19
  hideCurrencyExpansionControls?: boolean;
19
20
  defaultExpandCurrencies?: boolean;
20
21
  isDisabled?: boolean;
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@commercetools-uikit/localized-money-input",
3
3
  "description": "A controlled input component for localized money values with validation states.",
4
- "version": "14.0.1",
4
+ "version": "14.0.6",
5
5
  "bugs": "https://github.com/commercetools/ui-kit/issues",
6
6
  "repository": {
7
7
  "type": "git",
@@ -23,15 +23,15 @@
23
23
  "@babel/runtime-corejs3": "^7.17.2",
24
24
  "@commercetools-uikit/constraints": "14.0.1",
25
25
  "@commercetools-uikit/design-system": "14.0.0",
26
- "@commercetools-uikit/flat-button": "14.0.1",
27
- "@commercetools-uikit/hooks": "14.0.1",
26
+ "@commercetools-uikit/flat-button": "14.0.6",
27
+ "@commercetools-uikit/hooks": "14.0.3",
28
28
  "@commercetools-uikit/icons": "14.0.1",
29
- "@commercetools-uikit/input-utils": "14.0.1",
29
+ "@commercetools-uikit/input-utils": "14.0.6",
30
30
  "@commercetools-uikit/localized-utils": "14.0.1",
31
- "@commercetools-uikit/money-input": "14.0.1",
32
- "@commercetools-uikit/select-utils": "14.0.1",
33
- "@commercetools-uikit/spacings-stack": "14.0.1",
34
- "@commercetools-uikit/tooltip": "14.0.1",
31
+ "@commercetools-uikit/money-input": "14.0.6",
32
+ "@commercetools-uikit/select-utils": "14.0.6",
33
+ "@commercetools-uikit/spacings-stack": "14.0.6",
34
+ "@commercetools-uikit/tooltip": "14.0.3",
35
35
  "@commercetools-uikit/utils": "14.0.1",
36
36
  "@emotion/react": "^11.4.0",
37
37
  "@emotion/styled": "^11.3.0",