@commercetools-uikit/money-input 12.2.2 → 12.2.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.
|
@@ -7,7 +7,6 @@ import _Object$getOwnPropertyDescriptors from '@babel/runtime-corejs3/core-js-st
|
|
|
7
7
|
import _Object$defineProperties from '@babel/runtime-corejs3/core-js-stable/object/define-properties';
|
|
8
8
|
import _Object$defineProperty from '@babel/runtime-corejs3/core-js-stable/object/define-property';
|
|
9
9
|
import _typeof from '@babel/runtime-corejs3/helpers/esm/typeof';
|
|
10
|
-
import _extends from '@babel/runtime-corejs3/helpers/esm/extends';
|
|
11
10
|
import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
|
|
12
11
|
import _defineProperty from '@babel/runtime-corejs3/helpers/esm/defineProperty';
|
|
13
12
|
import _objectWithoutProperties from '@babel/runtime-corejs3/helpers/esm/objectWithoutProperties';
|
|
@@ -21,14 +20,14 @@ import _trimInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instanc
|
|
|
21
20
|
import _mapInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/map';
|
|
22
21
|
import _findInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/find';
|
|
23
22
|
import _concatInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/concat';
|
|
24
|
-
import
|
|
23
|
+
import { useRef, useCallback } from 'react';
|
|
25
24
|
import ReactDOM from 'react-dom';
|
|
26
25
|
import PropTypes from 'prop-types';
|
|
27
26
|
import has from 'lodash/has';
|
|
28
27
|
import requiredIf from 'react-required-if';
|
|
29
28
|
import Select, { components } from 'react-select';
|
|
30
29
|
import { defineMessages, useIntl } from 'react-intl';
|
|
31
|
-
import { css, useTheme
|
|
30
|
+
import { css, useTheme } from '@emotion/react';
|
|
32
31
|
import { customProperties } from '@commercetools-uikit/design-system';
|
|
33
32
|
import { isNumberish, filterDataAttributes, warning, SafeHTMLElement } from '@commercetools-uikit/utils';
|
|
34
33
|
import Tooltip from '@commercetools-uikit/tooltip';
|
|
@@ -37,6 +36,7 @@ import { FractionDigitsIcon } from '@commercetools-uikit/icons';
|
|
|
37
36
|
import Constraints from '@commercetools-uikit/constraints';
|
|
38
37
|
import { useToggleState } from '@commercetools-uikit/hooks';
|
|
39
38
|
import { getInputStyles } from '@commercetools-uikit/input-utils';
|
|
39
|
+
import { jsx, jsxs, Fragment } from '@emotion/react/jsx-runtime';
|
|
40
40
|
|
|
41
41
|
var currencies = {
|
|
42
42
|
ADP: {
|
|
@@ -702,16 +702,16 @@ var currencies = {
|
|
|
702
702
|
};
|
|
703
703
|
|
|
704
704
|
var getCurrencyLabelStyles = function getCurrencyLabelStyles() {
|
|
705
|
-
return /*#__PURE__*/css("display:flex;color:", customProperties.fontColorForInputWhenDisabled, ";background-color:", customProperties.backgroundColorForInputWhenDisabled, ";border-top-left-radius:", customProperties.borderRadiusForInput, ";border-bottom-left-radius:", customProperties.borderRadiusForInput, ";border:1px ", customProperties.borderColorForInputWhenDisabled, " solid;border-right:0;padding:0 ", customProperties.spacingS, ";align-items:center;font-size:", customProperties.fontSizeForInput, ";box-sizing:border-box;" + (process.env.NODE_ENV === "production" ? "" : ";label:getCurrencyLabelStyles;"));
|
|
705
|
+
return /*#__PURE__*/css("display:flex;color:", customProperties.fontColorForInputWhenDisabled, ";background-color:", customProperties.backgroundColorForInputWhenDisabled, ";border-top-left-radius:", customProperties.borderRadiusForInput, ";border-bottom-left-radius:", customProperties.borderRadiusForInput, ";border:1px ", customProperties.borderColorForInputWhenDisabled, " solid;border-right:0;padding:0 ", customProperties.spacingS, ";align-items:center;font-size:", customProperties.fontSizeForInput, ";box-sizing:border-box;" + (process.env.NODE_ENV === "production" ? "" : ";label:getCurrencyLabelStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFJd0MiLCJmaWxlIjoibW9uZXktaW5wdXQuc3R5bGVzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgY3VzdG9tUHJvcGVydGllcyBhcyB2YXJzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBnZXRJbnB1dFN0eWxlcyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2lucHV0LXV0aWxzJztcblxuY29uc3QgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyA9ICgpID0+IGNzc2BcbiAgZGlzcGxheTogZmxleDtcbiAgY29sb3I6ICR7dmFycy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJhY2tncm91bmQtY29sb3I6ICR7dmFycy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlcjogMXB4ICR7dmFycy5ib3JkZXJDb2xvckZvcklucHV0V2hlbkRpc2FibGVkfSBzb2xpZDtcbiAgYm9yZGVyLXJpZ2h0OiAwO1xuICBwYWRkaW5nOiAwICR7dmFycy5zcGFjaW5nU307XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGZvbnQtc2l6ZTogJHt2YXJzLmZvbnRTaXplRm9ySW5wdXR9O1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuYDtcblxuY29uc3QgZ2V0QW1vdW50SW5wdXRTdHlsZXMgPSAocHJvcHMpID0+IFtcbiAgZ2V0SW5wdXRTdHlsZXMocHJvcHMpLFxuICBjc3NgXG4gICAgYm9yZGVyLXRvcC1sZWZ0LXJhZGl1czogMDtcbiAgICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAwO1xuICAgIG1hcmdpbi1sZWZ0OiAwO1xuXG4gICAgJjo6cGxhY2Vob2xkZXIge1xuICAgICAgY29sb3I6ICR7dmFycy5wbGFjZWhvbGRlckZvbnRDb2xvckZvcklucHV0fTtcbiAgICB9XG4gIGAsXG5dO1xuXG5jb25zdCBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyA9ICh7IGlzRGlzYWJsZWQgfSkgPT4gY3NzYFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogMDtcbiAgcmlnaHQ6IDA7XG4gIG1hcmdpbi1yaWdodDogJHt2YXJzLnNwYWNpbmdYc307XG4gIGhlaWdodDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgY3Vyc29yOiAke2lzRGlzYWJsZWQgPyAnbm90LWFsbG93ZWQnIDogJ2RlZmF1bHQnfTtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG5gO1xuXG5leHBvcnQge1xuICBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyxcbiAgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyxcbiAgZ2V0QW1vdW50SW5wdXRTdHlsZXMsXG59O1xuIl19 */");
|
|
706
706
|
};
|
|
707
707
|
|
|
708
708
|
var getAmountInputStyles = function getAmountInputStyles(props) {
|
|
709
|
-
return [getInputStyles(props), /*#__PURE__*/css("border-top-left-radius:0;border-bottom-left-radius:0;margin-left:0;&::placeholder{color:", customProperties.placeholderFontColorForInput, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:getAmountInputStyles;"))];
|
|
709
|
+
return [getInputStyles(props), /*#__PURE__*/css("border-top-left-radius:0;border-bottom-left-radius:0;margin-left:0;&::placeholder{color:", customProperties.placeholderFontColorForInput, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:getAmountInputStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvQksiLCJmaWxlIjoibW9uZXktaW5wdXQuc3R5bGVzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgY3VzdG9tUHJvcGVydGllcyBhcyB2YXJzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBnZXRJbnB1dFN0eWxlcyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2lucHV0LXV0aWxzJztcblxuY29uc3QgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyA9ICgpID0+IGNzc2BcbiAgZGlzcGxheTogZmxleDtcbiAgY29sb3I6ICR7dmFycy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJhY2tncm91bmQtY29sb3I6ICR7dmFycy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlcjogMXB4ICR7dmFycy5ib3JkZXJDb2xvckZvcklucHV0V2hlbkRpc2FibGVkfSBzb2xpZDtcbiAgYm9yZGVyLXJpZ2h0OiAwO1xuICBwYWRkaW5nOiAwICR7dmFycy5zcGFjaW5nU307XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGZvbnQtc2l6ZTogJHt2YXJzLmZvbnRTaXplRm9ySW5wdXR9O1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuYDtcblxuY29uc3QgZ2V0QW1vdW50SW5wdXRTdHlsZXMgPSAocHJvcHMpID0+IFtcbiAgZ2V0SW5wdXRTdHlsZXMocHJvcHMpLFxuICBjc3NgXG4gICAgYm9yZGVyLXRvcC1sZWZ0LXJhZGl1czogMDtcbiAgICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAwO1xuICAgIG1hcmdpbi1sZWZ0OiAwO1xuXG4gICAgJjo6cGxhY2Vob2xkZXIge1xuICAgICAgY29sb3I6ICR7dmFycy5wbGFjZWhvbGRlckZvbnRDb2xvckZvcklucHV0fTtcbiAgICB9XG4gIGAsXG5dO1xuXG5jb25zdCBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyA9ICh7IGlzRGlzYWJsZWQgfSkgPT4gY3NzYFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogMDtcbiAgcmlnaHQ6IDA7XG4gIG1hcmdpbi1yaWdodDogJHt2YXJzLnNwYWNpbmdYc307XG4gIGhlaWdodDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgY3Vyc29yOiAke2lzRGlzYWJsZWQgPyAnbm90LWFsbG93ZWQnIDogJ2RlZmF1bHQnfTtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG5gO1xuXG5leHBvcnQge1xuICBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyxcbiAgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyxcbiAgZ2V0QW1vdW50SW5wdXRTdHlsZXMsXG59O1xuIl19 */")];
|
|
710
710
|
};
|
|
711
711
|
|
|
712
712
|
var getHighPrecisionWrapperStyles = function getHighPrecisionWrapperStyles(_ref) {
|
|
713
713
|
var isDisabled = _ref.isDisabled;
|
|
714
|
-
return /*#__PURE__*/css("position:absolute;top:0;right:0;margin-right:", customProperties.spacingXs, ";height:100%;display:flex;align-items:center;cursor:", isDisabled ? 'not-allowed' : 'default', ";justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : ";label:getHighPrecisionWrapperStyles;"));
|
|
714
|
+
return /*#__PURE__*/css("position:absolute;top:0;right:0;margin-right:", customProperties.spacingXs, ";height:100%;display:flex;align-items:center;cursor:", isDisabled ? 'not-allowed' : 'default', ";justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : ";label:getHighPrecisionWrapperStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErQjZEIiwiZmlsZSI6Im1vbmV5LWlucHV0LnN0eWxlcy5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNzcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcbmltcG9ydCB7IGN1c3RvbVByb3BlcnRpZXMgYXMgdmFycyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2Rlc2lnbi1zeXN0ZW0nO1xuaW1wb3J0IHsgZ2V0SW5wdXRTdHlsZXMgfSBmcm9tICdAY29tbWVyY2V0b29scy11aWtpdC9pbnB1dC11dGlscyc7XG5cbmNvbnN0IGdldEN1cnJlbmN5TGFiZWxTdHlsZXMgPSAoKSA9PiBjc3NgXG4gIGRpc3BsYXk6IGZsZXg7XG4gIGNvbG9yOiAke3ZhcnMuZm9udENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWR9O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAke3ZhcnMuYmFja2dyb3VuZENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWR9O1xuICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAke3ZhcnMuYm9yZGVyUmFkaXVzRm9ySW5wdXR9O1xuICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAke3ZhcnMuYm9yZGVyUmFkaXVzRm9ySW5wdXR9O1xuICBib3JkZXI6IDFweCAke3ZhcnMuYm9yZGVyQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH0gc29saWQ7XG4gIGJvcmRlci1yaWdodDogMDtcbiAgcGFkZGluZzogMCAke3ZhcnMuc3BhY2luZ1N9O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBmb250LXNpemU6ICR7dmFycy5mb250U2l6ZUZvcklucHV0fTtcbiAgYm94LXNpemluZzogYm9yZGVyLWJveDtcbmA7XG5cbmNvbnN0IGdldEFtb3VudElucHV0U3R5bGVzID0gKHByb3BzKSA9PiBbXG4gIGdldElucHV0U3R5bGVzKHByb3BzKSxcbiAgY3NzYFxuICAgIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6IDA7XG4gICAgYm9yZGVyLWJvdHRvbS1sZWZ0LXJhZGl1czogMDtcbiAgICBtYXJnaW4tbGVmdDogMDtcblxuICAgICY6OnBsYWNlaG9sZGVyIHtcbiAgICAgIGNvbG9yOiAke3ZhcnMucGxhY2Vob2xkZXJGb250Q29sb3JGb3JJbnB1dH07XG4gICAgfVxuICBgLFxuXTtcblxuY29uc3QgZ2V0SGlnaFByZWNpc2lvbldyYXBwZXJTdHlsZXMgPSAoeyBpc0Rpc2FibGVkIH0pID0+IGNzc2BcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICB0b3A6IDA7XG4gIHJpZ2h0OiAwO1xuICBtYXJnaW4tcmlnaHQ6ICR7dmFycy5zcGFjaW5nWHN9O1xuICBoZWlnaHQ6IDEwMCU7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGN1cnNvcjogJHtpc0Rpc2FibGVkID8gJ25vdC1hbGxvd2VkJyA6ICdkZWZhdWx0J307XG4gIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuYDtcblxuZXhwb3J0IHtcbiAgZ2V0SGlnaFByZWNpc2lvbldyYXBwZXJTdHlsZXMsXG4gIGdldEN1cnJlbmN5TGFiZWxTdHlsZXMsXG4gIGdldEFtb3VudElucHV0U3R5bGVzLFxufTtcbiJdfQ== */");
|
|
715
715
|
};
|
|
716
716
|
|
|
717
717
|
var messages = defineMessages({
|
|
@@ -741,6 +741,7 @@ var TooltipWrapper = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
741
741
|
} : {
|
|
742
742
|
name: "zjik7",
|
|
743
743
|
styles: "display:flex",
|
|
744
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.js"],"names":[],"mappings":"AAgCiC","file":"money-input.js","sourcesContent":["import { useRef, useCallback } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport has from 'lodash/has';\nimport requiredIf from 'react-required-if';\nimport Select, { components } from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  SafeHTMLElement,\n  filterDataAttributes,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport currencies from './currencies.json';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst getPortalId = (id) => `portal-${id}`;\nconst getPortalNode = (id) => document.querySelector(`#${getPortalId(id)}`);\n\nconst Portal = (props) => {\n  const domNode = getPortalNode(props.id);\n  return ReactDOM.createPortal(props.children, domNode);\n};\n\nconst CurrencyLabel = (props) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\nCurrencyLabel.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\nconst SingleValue = ({ id, ...props }) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\nSingleValue.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base, state) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount, locale) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount, 10);\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.\nexport const createMoneyValue = (currencyCode, rawAmount, locale) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (rawAmount, currencyCode, locale) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  };\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name) => (name ? `${name}.amount` : undefined);\nconst getCurrencyDropdownName = (name) =>\n  name ? `${name}.currencyCode` : undefined;\n\nconst MoneyInput = (props) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef();\n  const amountInputRef = useRef();\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(props.id),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, props.id, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    props.id,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange({\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, props.id, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          currencyCode,\n          intl.locale\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(formattedAmount)\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\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            // eslint-disable-next-line @typescript-eslint/no-empty-function\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(props.id),\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      props.id,\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(props.id),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const id = MoneyInput.getCurrencyDropdownId(props.id);\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, props.id, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(props.id)}\n            isDisabled={props.isDisabled}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={id}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isDisabled}\n            isSearchable={false}\n            components={{\n              // eslint-disable-next-line react/display-name\n              SingleValue: (innerProps) => (\n                <SingleValue {...innerProps} id={id} />\n              ),\n              // eslint-disable-next-line react/display-name\n              Input: (ownProps) => (\n                // eslint-disable-next-line react/prop-types\n                <components.Input {...ownProps} readOnly={props.isReadOnly} />\n              ),\n              DropdownIndicator,\n            }}\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(props.id)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${vars.spacingL};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes(props)}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value, locale) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue, locale) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue, locale) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.propTypes = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id: PropTypes.string,\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete: PropTypes.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: PropTypes.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: PropTypes.shape({\n    amount: PropTypes.string.isRequired,\n    currencyCode: PropTypes.string.isRequired,\n  }).isRequired,\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: PropTypes.arrayOf(PropTypes.string).isRequired,\n  /**\n   * Placeholder text for the input\n   */\n  placeholder: PropTypes.string,\n  /**\n   * Called when input is blurred\n   */\n  onBlur: PropTypes.func,\n  /**\n   * Called when input is focused\n   */\n  onFocus: PropTypes.func,\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled: PropTypes.bool,\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly: PropTypes.bool,\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed: PropTypes.bool,\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  /**\n   * Dom element to portal the currency select menu to\n   */\n  menuPortalTarget: PropTypes.instanceOf(SafeHTMLElement),\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex: PropTypes.number,\n  /**\n   * whether the menu should block scroll while open\n   */\n  menuShouldBlockScroll: PropTypes.bool,\n  /**\n   * Indicates that input has errors\n   */\n  hasError: PropTypes.bool,\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning: PropTypes.bool,\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge: PropTypes.bool,\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint: PropTypes.oneOf([\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};\n\nMoneyInput.defaultProps = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nexport default MoneyInput;\n"]} */",
|
|
744
745
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
745
746
|
});
|
|
746
747
|
|
|
@@ -760,8 +761,9 @@ var Portal = function Portal(props) {
|
|
|
760
761
|
var CurrencyLabel = function CurrencyLabel(props) {
|
|
761
762
|
return jsx("label", {
|
|
762
763
|
htmlFor: props.id,
|
|
763
|
-
css: getCurrencyLabelStyles()
|
|
764
|
-
|
|
764
|
+
css: getCurrencyLabelStyles(),
|
|
765
|
+
children: props.children
|
|
766
|
+
});
|
|
765
767
|
};
|
|
766
768
|
|
|
767
769
|
CurrencyLabel.displayName = 'CurrencyLabel';
|
|
@@ -774,9 +776,12 @@ var _SingleValue = function SingleValue(_ref3) {
|
|
|
774
776
|
var id = _ref3.id,
|
|
775
777
|
props = _objectWithoutProperties(_ref3, _excluded);
|
|
776
778
|
|
|
777
|
-
return jsx(components.SingleValue, props,
|
|
778
|
-
|
|
779
|
-
|
|
779
|
+
return jsx(components.SingleValue, _objectSpread(_objectSpread({}, props), {}, {
|
|
780
|
+
children: jsx("label", {
|
|
781
|
+
htmlFor: id,
|
|
782
|
+
children: props.children
|
|
783
|
+
})
|
|
784
|
+
}));
|
|
780
785
|
};
|
|
781
786
|
|
|
782
787
|
_SingleValue.displayName = 'SingleValue';
|
|
@@ -911,7 +916,8 @@ var parseRawAmountToNumber = function parseRawAmountToNumber(rawAmount, locale)
|
|
|
911
916
|
var fractionsSeparator;
|
|
912
917
|
|
|
913
918
|
if (locale) {
|
|
914
|
-
fractionsSeparator = 2.5
|
|
919
|
+
fractionsSeparator = 2.5 // we need any number with fractions, so that we know what is the fraction
|
|
920
|
+
.toLocaleString(locale) // "symbol" for the provided locale
|
|
915
921
|
.replace(/\d/g, ''); // then we remove the numbers and keep the "symbol"
|
|
916
922
|
} else {
|
|
917
923
|
var _context, _context2;
|
|
@@ -1031,6 +1037,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
1031
1037
|
} : {
|
|
1032
1038
|
name: "h5hvsa-MoneyInput",
|
|
1033
1039
|
styles: "position:relative;width:100%;label:MoneyInput;",
|
|
1040
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.js"],"names":[],"mappings":"AAwlBkB","file":"money-input.js","sourcesContent":["import { useRef, useCallback } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport has from 'lodash/has';\nimport requiredIf from 'react-required-if';\nimport Select, { components } from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  SafeHTMLElement,\n  filterDataAttributes,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport currencies from './currencies.json';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst getPortalId = (id) => `portal-${id}`;\nconst getPortalNode = (id) => document.querySelector(`#${getPortalId(id)}`);\n\nconst Portal = (props) => {\n  const domNode = getPortalNode(props.id);\n  return ReactDOM.createPortal(props.children, domNode);\n};\n\nconst CurrencyLabel = (props) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\nCurrencyLabel.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\nconst SingleValue = ({ id, ...props }) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\nSingleValue.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base, state) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount, locale) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount, 10);\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.\nexport const createMoneyValue = (currencyCode, rawAmount, locale) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (rawAmount, currencyCode, locale) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  };\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name) => (name ? `${name}.amount` : undefined);\nconst getCurrencyDropdownName = (name) =>\n  name ? `${name}.currencyCode` : undefined;\n\nconst MoneyInput = (props) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef();\n  const amountInputRef = useRef();\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(props.id),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, props.id, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    props.id,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange({\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, props.id, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          currencyCode,\n          intl.locale\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(formattedAmount)\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\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            // eslint-disable-next-line @typescript-eslint/no-empty-function\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(props.id),\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      props.id,\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(props.id),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const id = MoneyInput.getCurrencyDropdownId(props.id);\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, props.id, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(props.id)}\n            isDisabled={props.isDisabled}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={id}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isDisabled}\n            isSearchable={false}\n            components={{\n              // eslint-disable-next-line react/display-name\n              SingleValue: (innerProps) => (\n                <SingleValue {...innerProps} id={id} />\n              ),\n              // eslint-disable-next-line react/display-name\n              Input: (ownProps) => (\n                // eslint-disable-next-line react/prop-types\n                <components.Input {...ownProps} readOnly={props.isReadOnly} />\n              ),\n              DropdownIndicator,\n            }}\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(props.id)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${vars.spacingL};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes(props)}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value, locale) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue, locale) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue, locale) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.propTypes = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id: PropTypes.string,\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete: PropTypes.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: PropTypes.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: PropTypes.shape({\n    amount: PropTypes.string.isRequired,\n    currencyCode: PropTypes.string.isRequired,\n  }).isRequired,\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: PropTypes.arrayOf(PropTypes.string).isRequired,\n  /**\n   * Placeholder text for the input\n   */\n  placeholder: PropTypes.string,\n  /**\n   * Called when input is blurred\n   */\n  onBlur: PropTypes.func,\n  /**\n   * Called when input is focused\n   */\n  onFocus: PropTypes.func,\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled: PropTypes.bool,\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly: PropTypes.bool,\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed: PropTypes.bool,\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  /**\n   * Dom element to portal the currency select menu to\n   */\n  menuPortalTarget: PropTypes.instanceOf(SafeHTMLElement),\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex: PropTypes.number,\n  /**\n   * whether the menu should block scroll while open\n   */\n  menuShouldBlockScroll: PropTypes.bool,\n  /**\n   * Indicates that input has errors\n   */\n  hasError: PropTypes.bool,\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning: PropTypes.bool,\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge: PropTypes.bool,\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint: PropTypes.oneOf([\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};\n\nMoneyInput.defaultProps = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nexport default MoneyInput;\n"]} */",
|
|
1034
1041
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1035
1042
|
};
|
|
1036
1043
|
|
|
@@ -1040,6 +1047,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
|
1040
1047
|
} : {
|
|
1041
1048
|
name: "1w49f4-MoneyInput",
|
|
1042
1049
|
styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;",
|
|
1050
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.js"],"names":[],"mappings":"AAwiBgB","file":"money-input.js","sourcesContent":["import { useRef, useCallback } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport has from 'lodash/has';\nimport requiredIf from 'react-required-if';\nimport Select, { components } from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  SafeHTMLElement,\n  filterDataAttributes,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport currencies from './currencies.json';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst getPortalId = (id) => `portal-${id}`;\nconst getPortalNode = (id) => document.querySelector(`#${getPortalId(id)}`);\n\nconst Portal = (props) => {\n  const domNode = getPortalNode(props.id);\n  return ReactDOM.createPortal(props.children, domNode);\n};\n\nconst CurrencyLabel = (props) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\nCurrencyLabel.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\nconst SingleValue = ({ id, ...props }) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\nSingleValue.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base, state) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount, locale) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount, 10);\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.\nexport const createMoneyValue = (currencyCode, rawAmount, locale) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (rawAmount, currencyCode, locale) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  };\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name) => (name ? `${name}.amount` : undefined);\nconst getCurrencyDropdownName = (name) =>\n  name ? `${name}.currencyCode` : undefined;\n\nconst MoneyInput = (props) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef();\n  const amountInputRef = useRef();\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(props.id),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, props.id, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    props.id,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange({\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, props.id, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          currencyCode,\n          intl.locale\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(formattedAmount)\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\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            // eslint-disable-next-line @typescript-eslint/no-empty-function\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(props.id),\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      props.id,\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(props.id),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const id = MoneyInput.getCurrencyDropdownId(props.id);\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, props.id, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(props.id)}\n            isDisabled={props.isDisabled}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={id}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isDisabled}\n            isSearchable={false}\n            components={{\n              // eslint-disable-next-line react/display-name\n              SingleValue: (innerProps) => (\n                <SingleValue {...innerProps} id={id} />\n              ),\n              // eslint-disable-next-line react/display-name\n              Input: (ownProps) => (\n                // eslint-disable-next-line react/prop-types\n                <components.Input {...ownProps} readOnly={props.isReadOnly} />\n              ),\n              DropdownIndicator,\n            }}\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(props.id)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${vars.spacingL};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes(props)}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value, locale) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue, locale) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue, locale) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.propTypes = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id: PropTypes.string,\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete: PropTypes.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: PropTypes.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: PropTypes.shape({\n    amount: PropTypes.string.isRequired,\n    currencyCode: PropTypes.string.isRequired,\n  }).isRequired,\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: PropTypes.arrayOf(PropTypes.string).isRequired,\n  /**\n   * Placeholder text for the input\n   */\n  placeholder: PropTypes.string,\n  /**\n   * Called when input is blurred\n   */\n  onBlur: PropTypes.func,\n  /**\n   * Called when input is focused\n   */\n  onFocus: PropTypes.func,\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled: PropTypes.bool,\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly: PropTypes.bool,\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed: PropTypes.bool,\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  /**\n   * Dom element to portal the currency select menu to\n   */\n  menuPortalTarget: PropTypes.instanceOf(SafeHTMLElement),\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex: PropTypes.number,\n  /**\n   * whether the menu should block scroll while open\n   */\n  menuShouldBlockScroll: PropTypes.bool,\n  /**\n   * Indicates that input has errors\n   */\n  hasError: PropTypes.bool,\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning: PropTypes.bool,\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge: PropTypes.bool,\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint: PropTypes.oneOf([\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};\n\nMoneyInput.defaultProps = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nexport default MoneyInput;\n"]} */",
|
|
1043
1051
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1044
1052
|
};
|
|
1045
1053
|
|
|
@@ -1058,10 +1066,10 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1058
1066
|
amountHasFocus = _useToggleState4[0],
|
|
1059
1067
|
toggleAmountHasFocus = _useToggleState4[1];
|
|
1060
1068
|
|
|
1061
|
-
var containerRef =
|
|
1062
|
-
var amountInputRef =
|
|
1069
|
+
var containerRef = useRef();
|
|
1070
|
+
var amountInputRef = useRef();
|
|
1063
1071
|
var onFocus = props.onFocus;
|
|
1064
|
-
var handleAmountFocus =
|
|
1072
|
+
var handleAmountFocus = useCallback(function () {
|
|
1065
1073
|
if (onFocus) onFocus({
|
|
1066
1074
|
target: {
|
|
1067
1075
|
id: MoneyInput.getAmountInputId(props.id),
|
|
@@ -1071,7 +1079,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1071
1079
|
toggleAmountHasFocus(true);
|
|
1072
1080
|
}, [toggleAmountHasFocus, onFocus, props.id, props.name]);
|
|
1073
1081
|
var onChange = props.onChange;
|
|
1074
|
-
var handleAmountBlur =
|
|
1082
|
+
var handleAmountBlur = useCallback(function () {
|
|
1075
1083
|
var _context5;
|
|
1076
1084
|
|
|
1077
1085
|
var amount = _trimInstanceProperty(_context5 = props.value.amount).call(_context5);
|
|
@@ -1098,7 +1106,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1098
1106
|
}
|
|
1099
1107
|
}
|
|
1100
1108
|
}, [intl.locale, onChange, props.id, props.name, props.value.amount, props.value.currencyCode, toggleAmountHasFocus]);
|
|
1101
|
-
var handleAmountChange =
|
|
1109
|
+
var handleAmountChange = useCallback(function (event) {
|
|
1102
1110
|
if (isNumberish(event.target.value)) {
|
|
1103
1111
|
onChange({
|
|
1104
1112
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
@@ -1111,7 +1119,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1111
1119
|
});
|
|
1112
1120
|
}
|
|
1113
1121
|
}, [onChange, props.id, props.name]);
|
|
1114
|
-
var handleCurrencyChange =
|
|
1122
|
+
var handleCurrencyChange = useCallback(function (option) {
|
|
1115
1123
|
var currencyCode = option.value;
|
|
1116
1124
|
|
|
1117
1125
|
if (props.value.currencyCode !== currencyCode) {
|
|
@@ -1155,7 +1163,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1155
1163
|
amountInputRef.current.focus();
|
|
1156
1164
|
}
|
|
1157
1165
|
}, [intl.locale, onChange, props.id, props.name, props.value.amount, props.value.currencyCode]);
|
|
1158
|
-
var handleCurrencyFocus =
|
|
1166
|
+
var handleCurrencyFocus = useCallback(function () {
|
|
1159
1167
|
if (onFocus) onFocus({
|
|
1160
1168
|
target: {
|
|
1161
1169
|
id: MoneyInput.getCurrencyDropdownId(props.id),
|
|
@@ -1164,7 +1172,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1164
1172
|
});
|
|
1165
1173
|
toggleCurrencyHasFocus(true);
|
|
1166
1174
|
}, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);
|
|
1167
|
-
var handleCurrencyBlur =
|
|
1175
|
+
var handleCurrencyBlur = useCallback(function () {
|
|
1168
1176
|
toggleCurrencyHasFocus(false);
|
|
1169
1177
|
}, [toggleCurrencyHasFocus]);
|
|
1170
1178
|
var hasNoCurrencies = props.currencies.length === 0;
|
|
@@ -1206,7 +1214,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1206
1214
|
var id = MoneyInput.getCurrencyDropdownId(props.id);
|
|
1207
1215
|
var isHighPrecision = !MoneyInput.isEmpty(props.value) && MoneyInput.isHighPrecision(props.value, intl.locale);
|
|
1208
1216
|
var onBlur = props.onBlur;
|
|
1209
|
-
var handleContainerBlur =
|
|
1217
|
+
var handleContainerBlur = useCallback(function (event) {
|
|
1210
1218
|
// ensures that both fields are marked as touched when one of them
|
|
1211
1219
|
// is blurred
|
|
1212
1220
|
if (typeof onBlur === 'function' && !containerRef.current.contains(event.relatedTarget)) {
|
|
@@ -1224,100 +1232,108 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1224
1232
|
});
|
|
1225
1233
|
}
|
|
1226
1234
|
}, [onBlur, props.id, props.name]);
|
|
1227
|
-
var TooltipPortal =
|
|
1228
|
-
return jsx(Portal,
|
|
1235
|
+
var TooltipPortal = useCallback(function (remainingProps) {
|
|
1236
|
+
return jsx(Portal, _objectSpread(_objectSpread({}, remainingProps), {}, {
|
|
1229
1237
|
id: props.id
|
|
1230
1238
|
}));
|
|
1231
1239
|
}, [props.id]);
|
|
1232
1240
|
return jsx(Constraints.Horizontal, {
|
|
1233
|
-
max: props.horizontalConstraint
|
|
1234
|
-
|
|
1235
|
-
|
|
1236
|
-
|
|
1237
|
-
|
|
1238
|
-
|
|
1239
|
-
|
|
1240
|
-
|
|
1241
|
-
|
|
1242
|
-
|
|
1243
|
-
|
|
1244
|
-
|
|
1245
|
-
|
|
1246
|
-
|
|
1247
|
-
|
|
1248
|
-
|
|
1249
|
-
|
|
1250
|
-
|
|
1251
|
-
|
|
1252
|
-
|
|
1253
|
-
|
|
1254
|
-
|
|
1255
|
-
|
|
1256
|
-
|
|
1257
|
-
|
|
1258
|
-
|
|
1259
|
-
|
|
1260
|
-
|
|
1261
|
-
|
|
1262
|
-
|
|
1263
|
-
|
|
1264
|
-
|
|
1265
|
-
|
|
1266
|
-
|
|
1267
|
-
|
|
1268
|
-
|
|
1269
|
-
|
|
1270
|
-
|
|
1271
|
-
|
|
1272
|
-
|
|
1273
|
-
|
|
1274
|
-
|
|
1275
|
-
|
|
1276
|
-
|
|
1277
|
-
|
|
1278
|
-
|
|
1279
|
-
|
|
1280
|
-
|
|
1281
|
-
|
|
1282
|
-
|
|
1283
|
-
|
|
1284
|
-
|
|
1285
|
-
|
|
1286
|
-
|
|
1287
|
-
|
|
1288
|
-
|
|
1289
|
-
placeholder: props.placeholder,
|
|
1290
|
-
|
|
1291
|
-
|
|
1292
|
-
|
|
1293
|
-
|
|
1294
|
-
|
|
1295
|
-
|
|
1296
|
-
|
|
1297
|
-
|
|
1298
|
-
|
|
1299
|
-
|
|
1300
|
-
|
|
1301
|
-
|
|
1302
|
-
|
|
1303
|
-
|
|
1304
|
-
|
|
1305
|
-
|
|
1306
|
-
|
|
1307
|
-
|
|
1308
|
-
|
|
1309
|
-
|
|
1310
|
-
|
|
1311
|
-
|
|
1312
|
-
|
|
1313
|
-
|
|
1314
|
-
|
|
1315
|
-
|
|
1316
|
-
|
|
1317
|
-
|
|
1318
|
-
|
|
1319
|
-
|
|
1320
|
-
|
|
1241
|
+
max: props.horizontalConstraint,
|
|
1242
|
+
children: jsxs("div", {
|
|
1243
|
+
ref: containerRef,
|
|
1244
|
+
css: _ref2,
|
|
1245
|
+
"data-testid": "money-input-container",
|
|
1246
|
+
onBlur: handleContainerBlur,
|
|
1247
|
+
children: [hasNoCurrencies ? jsx(CurrencyLabel, {
|
|
1248
|
+
id: MoneyInput.getAmountInputId(props.id),
|
|
1249
|
+
isDisabled: props.isDisabled,
|
|
1250
|
+
children: option && option.label
|
|
1251
|
+
}) : jsx(Select, {
|
|
1252
|
+
inputId: id,
|
|
1253
|
+
name: getCurrencyDropdownName(props.name),
|
|
1254
|
+
value: option,
|
|
1255
|
+
isDisabled: props.isDisabled,
|
|
1256
|
+
isSearchable: false,
|
|
1257
|
+
components: {
|
|
1258
|
+
// eslint-disable-next-line react/display-name
|
|
1259
|
+
SingleValue: function SingleValue(innerProps) {
|
|
1260
|
+
return jsx(_SingleValue, _objectSpread(_objectSpread({}, innerProps), {}, {
|
|
1261
|
+
id: id
|
|
1262
|
+
}));
|
|
1263
|
+
},
|
|
1264
|
+
// eslint-disable-next-line react/display-name
|
|
1265
|
+
Input: function Input(ownProps) {
|
|
1266
|
+
return (// eslint-disable-next-line react/prop-types
|
|
1267
|
+
jsx(components.Input, _objectSpread(_objectSpread({}, ownProps), {}, {
|
|
1268
|
+
readOnly: props.isReadOnly
|
|
1269
|
+
}))
|
|
1270
|
+
);
|
|
1271
|
+
},
|
|
1272
|
+
DropdownIndicator: DropdownIndicator
|
|
1273
|
+
},
|
|
1274
|
+
options: options,
|
|
1275
|
+
menuIsOpen: props.isReadOnly ? false : undefined,
|
|
1276
|
+
placeholder: "",
|
|
1277
|
+
styles: currencySelectStyles,
|
|
1278
|
+
onFocus: handleCurrencyFocus,
|
|
1279
|
+
menuPortalTarget: props.menuPortalTarget,
|
|
1280
|
+
menuShouldBlockScroll: props.menuShouldBlockScroll,
|
|
1281
|
+
onBlur: handleCurrencyBlur,
|
|
1282
|
+
onChange: handleCurrencyChange,
|
|
1283
|
+
"data-testid": "currency-dropdown"
|
|
1284
|
+
}), jsxs("div", {
|
|
1285
|
+
css: _ref,
|
|
1286
|
+
children: [jsx("input", _objectSpread({
|
|
1287
|
+
ref: amountInputRef,
|
|
1288
|
+
id: MoneyInput.getAmountInputId(props.id),
|
|
1289
|
+
autoComplete: props.autoComplete,
|
|
1290
|
+
name: getAmountInputName(props.name),
|
|
1291
|
+
type: "text",
|
|
1292
|
+
onFocus: handleAmountFocus,
|
|
1293
|
+
value: props.value.amount,
|
|
1294
|
+
css: [getAmountInputStyles(_objectSpread(_objectSpread({}, props), {}, {
|
|
1295
|
+
hasFocus: hasFocus
|
|
1296
|
+
})), // accounts for size of icon
|
|
1297
|
+
props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/css("padding-right:", customProperties.spacingL, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.js"],"names":[],"mappings":"AA0mBmB","file":"money-input.js","sourcesContent":["import { useRef, useCallback } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport has from 'lodash/has';\nimport requiredIf from 'react-required-if';\nimport Select, { components } from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  SafeHTMLElement,\n  filterDataAttributes,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport currencies from './currencies.json';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst getPortalId = (id) => `portal-${id}`;\nconst getPortalNode = (id) => document.querySelector(`#${getPortalId(id)}`);\n\nconst Portal = (props) => {\n  const domNode = getPortalNode(props.id);\n  return ReactDOM.createPortal(props.children, domNode);\n};\n\nconst CurrencyLabel = (props) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\nCurrencyLabel.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\nconst SingleValue = ({ id, ...props }) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\nSingleValue.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base, state) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount, locale) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount, 10);\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.\nexport const createMoneyValue = (currencyCode, rawAmount, locale) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (rawAmount, currencyCode, locale) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  };\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name) => (name ? `${name}.amount` : undefined);\nconst getCurrencyDropdownName = (name) =>\n  name ? `${name}.currencyCode` : undefined;\n\nconst MoneyInput = (props) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef();\n  const amountInputRef = useRef();\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(props.id),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, props.id, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    props.id,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange({\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, props.id, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          currencyCode,\n          intl.locale\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(formattedAmount)\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\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            // eslint-disable-next-line @typescript-eslint/no-empty-function\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(props.id),\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      props.id,\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(props.id),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const id = MoneyInput.getCurrencyDropdownId(props.id);\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, props.id, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(props.id)}\n            isDisabled={props.isDisabled}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={id}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isDisabled}\n            isSearchable={false}\n            components={{\n              // eslint-disable-next-line react/display-name\n              SingleValue: (innerProps) => (\n                <SingleValue {...innerProps} id={id} />\n              ),\n              // eslint-disable-next-line react/display-name\n              Input: (ownProps) => (\n                // eslint-disable-next-line react/prop-types\n                <components.Input {...ownProps} readOnly={props.isReadOnly} />\n              ),\n              DropdownIndicator,\n            }}\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(props.id)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${vars.spacingL};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes(props)}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value, locale) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue, locale) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue, locale) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.propTypes = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id: PropTypes.string,\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete: PropTypes.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: PropTypes.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: PropTypes.shape({\n    amount: PropTypes.string.isRequired,\n    currencyCode: PropTypes.string.isRequired,\n  }).isRequired,\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: PropTypes.arrayOf(PropTypes.string).isRequired,\n  /**\n   * Placeholder text for the input\n   */\n  placeholder: PropTypes.string,\n  /**\n   * Called when input is blurred\n   */\n  onBlur: PropTypes.func,\n  /**\n   * Called when input is focused\n   */\n  onFocus: PropTypes.func,\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled: PropTypes.bool,\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly: PropTypes.bool,\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed: PropTypes.bool,\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  /**\n   * Dom element to portal the currency select menu to\n   */\n  menuPortalTarget: PropTypes.instanceOf(SafeHTMLElement),\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex: PropTypes.number,\n  /**\n   * whether the menu should block scroll while open\n   */\n  menuShouldBlockScroll: PropTypes.bool,\n  /**\n   * Indicates that input has errors\n   */\n  hasError: PropTypes.bool,\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning: PropTypes.bool,\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge: PropTypes.bool,\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint: PropTypes.oneOf([\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};\n\nMoneyInput.defaultProps = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\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.js"],"names":[],"mappings":"AAqmBY","file":"money-input.js","sourcesContent":["import { useRef, useCallback } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport has from 'lodash/has';\nimport requiredIf from 'react-required-if';\nimport Select, { components } from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  SafeHTMLElement,\n  filterDataAttributes,\n} from '@commercetools-uikit/utils';\nimport Tooltip from '@commercetools-uikit/tooltip';\nimport {\n  DropdownIndicator,\n  createSelectStyles,\n} from '@commercetools-uikit/select-utils';\nimport { FractionDigitsIcon } from '@commercetools-uikit/icons';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport currencies from './currencies.json';\nimport {\n  getHighPrecisionWrapperStyles,\n  getCurrencyLabelStyles,\n  getAmountInputStyles,\n} from './money-input.styles';\nimport messages from './messages';\n\nconst TooltipWrapper = styled.div`\n  display: flex;\n`;\n\nconst getPortalId = (id) => `portal-${id}`;\nconst getPortalNode = (id) => document.querySelector(`#${getPortalId(id)}`);\n\nconst Portal = (props) => {\n  const domNode = getPortalNode(props.id);\n  return ReactDOM.createPortal(props.children, domNode);\n};\n\nconst CurrencyLabel = (props) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\nCurrencyLabel.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\nconst SingleValue = ({ id, ...props }) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\nSingleValue.propTypes = {\n  id: PropTypes.string,\n  children: PropTypes.node,\n};\n\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base, state) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount, locale) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount, 10);\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.\nexport const createMoneyValue = (currencyCode, rawAmount, locale) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (rawAmount, currencyCode, locale) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  };\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name) => (name ? `${name}.amount` : undefined);\nconst getCurrencyDropdownName = (name) =>\n  name ? `${name}.currencyCode` : undefined;\n\nconst MoneyInput = (props) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef();\n  const amountInputRef = useRef();\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(props.id),\n          name: getAmountInputName(props.name),\n        },\n      });\n    toggleAmountHasFocus(true);\n  }, [toggleAmountHasFocus, onFocus, props.id, props.name]);\n\n  const { onChange } = props;\n  const handleAmountBlur = useCallback(() => {\n    const amount = props.value.amount.trim();\n    toggleAmountHasFocus(false);\n    // Skip formatting for empty value or when the input is used with an\n    // unknown currency.\n    if (amount.length > 0 && currencies[props.value.currencyCode]) {\n      const formattedAmount = formatAmount(\n        amount,\n        props.value.currencyCode,\n        intl.locale\n      );\n\n      // When the user entered a value with centPrecision, we can format\n      // the resulting value to that currency, e.g. 20.1 to 20.10\n      if (String(formattedAmount) !== amount) {\n        // We need to emit an event with the now formatted value\n        const fakeEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: formattedAmount,\n          },\n        };\n        onChange(fakeEvent);\n      }\n    }\n  }, [\n    intl.locale,\n    onChange,\n    props.id,\n    props.name,\n    props.value.amount,\n    props.value.currencyCode,\n    toggleAmountHasFocus,\n  ]);\n\n  const handleAmountChange = useCallback(\n    (event) => {\n      if (isNumberish(event.target.value)) {\n        onChange({\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n            value: event.target.value,\n          },\n        });\n      }\n    },\n    [onChange, props.id, props.name]\n  );\n\n  const handleCurrencyChange = useCallback(\n    (option) => {\n      const currencyCode = option.value;\n      if (props.value.currencyCode !== currencyCode) {\n        // When the user changes from a currency with 3 fraction digits to\n        // a currency with 2 fraction digits, and when the input value was\n        // \"9.000\" (9), then it should change to \"9.00\" to reflect the new\n        // currency's number of fraction digits.\n        // When the currency was a high-precision price, then no digits should\n        // be lost\n        const formattedAmount = formatAmount(\n          props.value.amount.trim(),\n          currencyCode,\n          intl.locale\n        );\n        // The user could be changing the currency before entering any amount,\n        // or while the amount is invalid. In these cases, we don't attempt to\n        // format the amount.\n        const nextAmount = isNaN(formattedAmount)\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\n          // eslint-disable-next-line @typescript-eslint/no-empty-function\n          persist: () => {},\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\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            // eslint-disable-next-line @typescript-eslint/no-empty-function\n            persist: () => {},\n            target: {\n              id: MoneyInput.getAmountInputId(props.id),\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      props.id,\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(props.id),\n          name: getCurrencyDropdownName(props.name),\n        },\n      });\n\n    toggleCurrencyHasFocus(true);\n  }, [onFocus, toggleCurrencyHasFocus, props.name, props.id]);\n\n  const handleCurrencyBlur = useCallback(() => {\n    toggleCurrencyHasFocus(false);\n  }, [toggleCurrencyHasFocus]);\n\n  const hasNoCurrencies = props.currencies.length === 0;\n  const hasFocus = currencyHasFocus || amountHasFocus;\n  const theme = useTheme();\n  const currencySelectStyles = createCurrencySelectStyles(\n    {\n      hasWarning: props.hasWarning,\n      hasError: props.hasError,\n      isDisabled: props.isDisabled,\n      isReadOnly: props.isReadOnly,\n      hasFocus,\n      menuPortalZIndex: props.menuPortalZIndex,\n    },\n    theme\n  );\n  const options = props.currencies.map((currencyCode) => ({\n    label: currencyCode,\n    value: currencyCode,\n  }));\n\n  const option = (() => {\n    const matchedOption = options.find(\n      (optionCandidate) => optionCandidate.value === props.value.currencyCode\n    );\n    if (matchedOption) return matchedOption;\n    // ensure an option is found, even when the currencies don't include\n    // the money value's currencyCode\n    if (props.value.currencyCode.trim() !== '')\n      return {\n        label: props.value.currencyCode,\n        value: props.value.currencyCode,\n      };\n    return null;\n  })();\n\n  const id = MoneyInput.getCurrencyDropdownId(props.id);\n\n  const isHighPrecision =\n    !MoneyInput.isEmpty(props.value) &&\n    MoneyInput.isHighPrecision(props.value, intl.locale);\n\n  const { onBlur } = props;\n  const handleContainerBlur = useCallback(\n    (event) => {\n      // ensures that both fields are marked as touched when one of them\n      // is blurred\n      if (\n        typeof onBlur === 'function' &&\n        !containerRef.current.contains(event.relatedTarget)\n      ) {\n        onBlur({\n          target: {\n            id: MoneyInput.getCurrencyDropdownId(props.id),\n            name: getCurrencyDropdownName(props.name),\n          },\n        });\n        onBlur({\n          target: {\n            id: MoneyInput.getAmountInputId(props.id),\n            name: getAmountInputName(props.name),\n          },\n        });\n      }\n    },\n    [onBlur, props.id, props.name]\n  );\n\n  const TooltipPortal = useCallback(\n    (remainingProps) => <Portal {...remainingProps} id={props.id} />,\n    [props.id]\n  );\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <div\n        ref={containerRef}\n        css={css`\n          font-family: inherit;\n          width: 100%;\n          position: relative;\n          display: flex;\n        `}\n        data-testid=\"money-input-container\"\n        onBlur={handleContainerBlur}\n      >\n        {hasNoCurrencies ? (\n          <CurrencyLabel\n            id={MoneyInput.getAmountInputId(props.id)}\n            isDisabled={props.isDisabled}\n          >\n            {option && option.label}\n          </CurrencyLabel>\n        ) : (\n          <Select\n            inputId={id}\n            name={getCurrencyDropdownName(props.name)}\n            value={option}\n            isDisabled={props.isDisabled}\n            isSearchable={false}\n            components={{\n              // eslint-disable-next-line react/display-name\n              SingleValue: (innerProps) => (\n                <SingleValue {...innerProps} id={id} />\n              ),\n              // eslint-disable-next-line react/display-name\n              Input: (ownProps) => (\n                // eslint-disable-next-line react/prop-types\n                <components.Input {...ownProps} readOnly={props.isReadOnly} />\n              ),\n              DropdownIndicator,\n            }}\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(props.id)}\n            autoComplete={props.autoComplete}\n            name={getAmountInputName(props.name)}\n            type=\"text\"\n            onFocus={handleAmountFocus}\n            value={props.value.amount}\n            css={[\n              getAmountInputStyles({ ...props, hasFocus }),\n              // accounts for size of icon\n              props.hasHighPrecisionBadge &&\n                isHighPrecision &&\n                css`\n                  padding-right: ${vars.spacingL};\n                `,\n            ]}\n            placeholder={props.placeholder}\n            onChange={handleAmountChange}\n            onBlur={handleAmountBlur}\n            disabled={props.isDisabled}\n            readOnly={props.isReadOnly}\n            autoFocus={props.isAutofocussed}\n            {...filterDataAttributes(props)}\n          />\n          {props.hasHighPrecisionBadge && isHighPrecision && (\n            <>\n              {!props.isDisabled && <div id={getPortalId(props.id)} />}\n              <div\n                css={() =>\n                  getHighPrecisionWrapperStyles({\n                    isDisabled: props.isDisabled,\n                  })\n                }\n              >\n                <Tooltip\n                  off={props.isDisabled}\n                  placement=\"top-end\"\n                  // we use negative margin to make up for the padding in the Tooltip Wrapper\n                  // so that the tooltip is flush with the component\n                  styles={{\n                    body: {\n                      margin: `${vars.spacingS} -${vars.spacingXs} ${vars.spacingS} 0`,\n                    },\n                  }}\n                  title={intl.formatMessage(messages.highPrecision)}\n                  components={{\n                    TooltipWrapperComponent: TooltipPortal,\n                    WrapperComponent: TooltipWrapper,\n                  }}\n                >\n                  <FractionDigitsIcon\n                    color={props.isDisabled ? 'neutral60' : 'info'}\n                  />\n                </Tooltip>\n              </div>\n            </>\n          )}\n        </div>\n      </div>\n    </Constraints.Horizontal>\n  );\n};\n\nMoneyInput.displayName = 'MoneyInput';\n\nMoneyInput.getAmountInputId = getAmountInputName;\n\nMoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;\n\nMoneyInput.convertToMoneyValue = (value, locale) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue, locale) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue, locale) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.propTypes = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id: PropTypes.string,\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete: PropTypes.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: PropTypes.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: PropTypes.shape({\n    amount: PropTypes.string.isRequired,\n    currencyCode: PropTypes.string.isRequired,\n  }).isRequired,\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: PropTypes.arrayOf(PropTypes.string).isRequired,\n  /**\n   * Placeholder text for the input\n   */\n  placeholder: PropTypes.string,\n  /**\n   * Called when input is blurred\n   */\n  onBlur: PropTypes.func,\n  /**\n   * Called when input is focused\n   */\n  onFocus: PropTypes.func,\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled: PropTypes.bool,\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly: PropTypes.bool,\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed: PropTypes.bool,\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  /**\n   * Dom element to portal the currency select menu to\n   */\n  menuPortalTarget: PropTypes.instanceOf(SafeHTMLElement),\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex: PropTypes.number,\n  /**\n   * whether the menu should block scroll while open\n   */\n  menuShouldBlockScroll: PropTypes.bool,\n  /**\n   * Indicates that input has errors\n   */\n  hasError: PropTypes.bool,\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning: PropTypes.bool,\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge: PropTypes.bool,\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint: PropTypes.oneOf([\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};\n\nMoneyInput.defaultProps = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nexport default MoneyInput;\n"]} */"],
|
|
1298
|
+
placeholder: props.placeholder,
|
|
1299
|
+
onChange: handleAmountChange,
|
|
1300
|
+
onBlur: handleAmountBlur,
|
|
1301
|
+
disabled: props.isDisabled,
|
|
1302
|
+
readOnly: props.isReadOnly,
|
|
1303
|
+
autoFocus: props.isAutofocussed
|
|
1304
|
+
}, filterDataAttributes(props))), props.hasHighPrecisionBadge && isHighPrecision && jsxs(Fragment, {
|
|
1305
|
+
children: [!props.isDisabled && jsx("div", {
|
|
1306
|
+
id: getPortalId(props.id)
|
|
1307
|
+
}), jsx("div", {
|
|
1308
|
+
css: function css() {
|
|
1309
|
+
return getHighPrecisionWrapperStyles({
|
|
1310
|
+
isDisabled: props.isDisabled
|
|
1311
|
+
});
|
|
1312
|
+
},
|
|
1313
|
+
children: jsx(Tooltip, {
|
|
1314
|
+
off: props.isDisabled,
|
|
1315
|
+
placement: "top-end" // we use negative margin to make up for the padding in the Tooltip Wrapper
|
|
1316
|
+
// so that the tooltip is flush with the component
|
|
1317
|
+
,
|
|
1318
|
+
styles: {
|
|
1319
|
+
body: {
|
|
1320
|
+
margin: _concatInstanceProperty(_context9 = _concatInstanceProperty(_context10 = "".concat(customProperties.spacingS, " -")).call(_context10, customProperties.spacingXs, " ")).call(_context9, customProperties.spacingS, " 0")
|
|
1321
|
+
}
|
|
1322
|
+
},
|
|
1323
|
+
title: intl.formatMessage(messages.highPrecision),
|
|
1324
|
+
components: {
|
|
1325
|
+
TooltipWrapperComponent: TooltipPortal,
|
|
1326
|
+
WrapperComponent: TooltipWrapper
|
|
1327
|
+
},
|
|
1328
|
+
children: jsx(FractionDigitsIcon, {
|
|
1329
|
+
color: props.isDisabled ? 'neutral60' : 'info'
|
|
1330
|
+
})
|
|
1331
|
+
})
|
|
1332
|
+
})]
|
|
1333
|
+
})]
|
|
1334
|
+
})]
|
|
1335
|
+
})
|
|
1336
|
+
});
|
|
1321
1337
|
};
|
|
1322
1338
|
|
|
1323
1339
|
MoneyInput.displayName = 'MoneyInput';
|
|
@@ -1476,7 +1492,7 @@ MoneyInput.defaultProps = {
|
|
|
1476
1492
|
};
|
|
1477
1493
|
var MoneyInput$1 = MoneyInput;
|
|
1478
1494
|
|
|
1479
|
-
// NOTE: This string will be replaced
|
|
1480
|
-
var version =
|
|
1495
|
+
// NOTE: This string will be replaced on build time with the package version.
|
|
1496
|
+
var version = "12.2.6";
|
|
1481
1497
|
|
|
1482
1498
|
export { MoneyInput$1 as default, version };
|