@commercetools-uikit/money-input 12.2.9 → 13.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +50 -23
- package/dist/commercetools-uikit-money-input.cjs.d.ts +2 -0
- package/dist/commercetools-uikit-money-input.cjs.dev.js +82 -152
- package/dist/commercetools-uikit-money-input.cjs.prod.js +44 -34
- package/dist/commercetools-uikit-money-input.esm.js +82 -151
- package/dist/declarations/src/index.d.ts +2 -0
- package/dist/declarations/src/messages.d.ts +8 -0
- package/dist/declarations/src/money-input.d.ts +95 -0
- package/dist/declarations/src/money-input.styles.d.ts +11 -0
- package/dist/declarations/src/version.d.ts +2 -0
- package/package.json +12 -12
|
@@ -10,11 +10,11 @@ var _forEachInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/in
|
|
|
10
10
|
var _Object$getOwnPropertyDescriptors = require('@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptors');
|
|
11
11
|
var _Object$defineProperties = require('@babel/runtime-corejs3/core-js-stable/object/define-properties');
|
|
12
12
|
var _Object$defineProperty = require('@babel/runtime-corejs3/core-js-stable/object/define-property');
|
|
13
|
-
var _typeof = require('@babel/runtime-corejs3/helpers/typeof');
|
|
14
13
|
var _slicedToArray = require('@babel/runtime-corejs3/helpers/slicedToArray');
|
|
15
14
|
var _defineProperty = require('@babel/runtime-corejs3/helpers/defineProperty');
|
|
16
15
|
var _objectWithoutProperties = require('@babel/runtime-corejs3/helpers/objectWithoutProperties');
|
|
17
16
|
var _styled = require('@emotion/styled/base');
|
|
17
|
+
var _pt = require('prop-types');
|
|
18
18
|
var _lastIndexOfInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/last-index-of');
|
|
19
19
|
var _parseFloat = require('@babel/runtime-corejs3/core-js-stable/parse-float');
|
|
20
20
|
var _Math$trunc = require('@babel/runtime-corejs3/core-js-stable/math/trunc');
|
|
@@ -26,9 +26,7 @@ var _findInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/insta
|
|
|
26
26
|
var _concatInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/concat');
|
|
27
27
|
var react$1 = require('react');
|
|
28
28
|
var ReactDOM = require('react-dom');
|
|
29
|
-
var PropTypes = require('prop-types');
|
|
30
29
|
var has = require('lodash/has');
|
|
31
|
-
var requiredIf = require('react-required-if');
|
|
32
30
|
var Select = require('react-select');
|
|
33
31
|
var reactIntl = require('react-intl');
|
|
34
32
|
var react = require('@emotion/react');
|
|
@@ -53,6 +51,7 @@ var _Object$getOwnPropertyDescriptors__default = /*#__PURE__*/_interopDefault(_O
|
|
|
53
51
|
var _Object$defineProperties__default = /*#__PURE__*/_interopDefault(_Object$defineProperties);
|
|
54
52
|
var _Object$defineProperty__default = /*#__PURE__*/_interopDefault(_Object$defineProperty);
|
|
55
53
|
var _styled__default = /*#__PURE__*/_interopDefault(_styled);
|
|
54
|
+
var _pt__default = /*#__PURE__*/_interopDefault(_pt);
|
|
56
55
|
var _lastIndexOfInstanceProperty__default = /*#__PURE__*/_interopDefault(_lastIndexOfInstanceProperty);
|
|
57
56
|
var _parseFloat__default = /*#__PURE__*/_interopDefault(_parseFloat);
|
|
58
57
|
var _Math$trunc__default = /*#__PURE__*/_interopDefault(_Math$trunc);
|
|
@@ -63,9 +62,7 @@ var _mapInstanceProperty__default = /*#__PURE__*/_interopDefault(_mapInstancePro
|
|
|
63
62
|
var _findInstanceProperty__default = /*#__PURE__*/_interopDefault(_findInstanceProperty);
|
|
64
63
|
var _concatInstanceProperty__default = /*#__PURE__*/_interopDefault(_concatInstanceProperty);
|
|
65
64
|
var ReactDOM__default = /*#__PURE__*/_interopDefault(ReactDOM);
|
|
66
|
-
var PropTypes__default = /*#__PURE__*/_interopDefault(PropTypes);
|
|
67
65
|
var has__default = /*#__PURE__*/_interopDefault(has);
|
|
68
|
-
var requiredIf__default = /*#__PURE__*/_interopDefault(requiredIf);
|
|
69
66
|
var Select__default = /*#__PURE__*/_interopDefault(Select);
|
|
70
67
|
var Tooltip__default = /*#__PURE__*/_interopDefault(Tooltip);
|
|
71
68
|
var Constraints__default = /*#__PURE__*/_interopDefault(Constraints);
|
|
@@ -734,16 +731,16 @@ var currencies = {
|
|
|
734
731
|
};
|
|
735
732
|
|
|
736
733
|
var getCurrencyLabelStyles = function getCurrencyLabelStyles() {
|
|
737
|
-
return /*#__PURE__*/react.css("display:flex;color:", designSystem.customProperties.fontColorForInputWhenDisabled, ";background-color:", designSystem.customProperties.backgroundColorForInputWhenDisabled, ";border-top-left-radius:", designSystem.customProperties.borderRadiusForInput, ";border-bottom-left-radius:", designSystem.customProperties.borderRadiusForInput, ";border:1px ", designSystem.customProperties.borderColorForInputWhenDisabled, " solid;border-right:0;padding:0 ", designSystem.customProperties.spacingS, ";align-items:center;font-size:", designSystem.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,
|
|
734
|
+
return /*#__PURE__*/react.css("display:flex;color:", designSystem.customProperties.fontColorForInputWhenDisabled, ";background-color:", designSystem.customProperties.backgroundColorForInputWhenDisabled, ";border-top-left-radius:", designSystem.customProperties.borderRadiusForInput, ";border-bottom-left-radius:", designSystem.customProperties.borderRadiusForInput, ";border:1px ", designSystem.customProperties.borderColorForInputWhenDisabled, " solid;border-right:0;padding:0 ", designSystem.customProperties.spacingS, ";align-items:center;font-size:", designSystem.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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLd0MiLCJmaWxlIjoibW9uZXktaW5wdXQuc3R5bGVzLnRzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgY3VzdG9tUHJvcGVydGllcyBhcyB2YXJzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBnZXRJbnB1dFN0eWxlcyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2lucHV0LXV0aWxzJztcbmltcG9ydCB0eXBlIHsgVElucHV0UHJvcHMgfSBmcm9tICcuL21vbmV5LWlucHV0JztcblxuY29uc3QgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyA9ICgpID0+IGNzc2BcbiAgZGlzcGxheTogZmxleDtcbiAgY29sb3I6ICR7dmFycy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJhY2tncm91bmQtY29sb3I6ICR7dmFycy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlcjogMXB4ICR7dmFycy5ib3JkZXJDb2xvckZvcklucHV0V2hlbkRpc2FibGVkfSBzb2xpZDtcbiAgYm9yZGVyLXJpZ2h0OiAwO1xuICBwYWRkaW5nOiAwICR7dmFycy5zcGFjaW5nU307XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGZvbnQtc2l6ZTogJHt2YXJzLmZvbnRTaXplRm9ySW5wdXR9O1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuYDtcblxudHlwZSBUR2V0QW1vdW50SW5wdXRTdHlsZXMgPSB7XG4gIGhhc0ZvY3VzOiBib29sZWFuO1xufSAmIFRJbnB1dFByb3BzO1xuY29uc3QgZ2V0QW1vdW50SW5wdXRTdHlsZXMgPSAocHJvcHM6IFRHZXRBbW91bnRJbnB1dFN0eWxlcykgPT4gW1xuICBnZXRJbnB1dFN0eWxlcyhwcm9wcyksXG4gIGNzc2BcbiAgICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAwO1xuICAgIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6IDA7XG4gICAgbWFyZ2luLWxlZnQ6IDA7XG5cbiAgICAmOjpwbGFjZWhvbGRlciB7XG4gICAgICBjb2xvcjogJHt2YXJzLnBsYWNlaG9sZGVyRm9udENvbG9yRm9ySW5wdXR9O1xuICAgIH1cbiAgYCxcbl07XG5cbnR5cGUgVEdldEhpZ2hQcmVjaXNpb25XcmFwcGVyU3R5bGVzID0ge1xuICBpc0Rpc2FibGVkPzogYm9vbGVhbjtcbn07XG5cbmNvbnN0IGdldEhpZ2hQcmVjaXNpb25XcmFwcGVyU3R5bGVzID0gKHtcbiAgaXNEaXNhYmxlZCxcbn06IFRHZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcykgPT4gY3NzYFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogMDtcbiAgcmlnaHQ6IDA7XG4gIG1hcmdpbi1yaWdodDogJHt2YXJzLnNwYWNpbmdYc307XG4gIGhlaWdodDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgY3Vyc29yOiAke2lzRGlzYWJsZWQgPyAnbm90LWFsbG93ZWQnIDogJ2RlZmF1bHQnfTtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG5gO1xuXG5leHBvcnQge1xuICBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyxcbiAgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyxcbiAgZ2V0QW1vdW50SW5wdXRTdHlsZXMsXG59O1xuIl19 */");
|
|
738
735
|
};
|
|
739
736
|
|
|
740
737
|
var getAmountInputStyles = function getAmountInputStyles(props) {
|
|
741
|
-
return [inputUtils.getInputStyles(props), /*#__PURE__*/react.css("border-top-left-radius:0;border-bottom-left-radius:0;margin-left:0;&::placeholder{color:", designSystem.customProperties.placeholderFontColorForInput, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:getAmountInputStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
738
|
+
return [inputUtils.getInputStyles(props), /*#__PURE__*/react.css("border-top-left-radius:0;border-bottom-left-radius:0;margin-left:0;&::placeholder{color:", designSystem.customProperties.placeholderFontColorForInput, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:getAmountInputStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF3QksiLCJmaWxlIjoibW9uZXktaW5wdXQuc3R5bGVzLnRzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgY3VzdG9tUHJvcGVydGllcyBhcyB2YXJzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBnZXRJbnB1dFN0eWxlcyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2lucHV0LXV0aWxzJztcbmltcG9ydCB0eXBlIHsgVElucHV0UHJvcHMgfSBmcm9tICcuL21vbmV5LWlucHV0JztcblxuY29uc3QgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyA9ICgpID0+IGNzc2BcbiAgZGlzcGxheTogZmxleDtcbiAgY29sb3I6ICR7dmFycy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJhY2tncm91bmQtY29sb3I6ICR7dmFycy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH07XG4gIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6ICR7dmFycy5ib3JkZXJSYWRpdXNGb3JJbnB1dH07XG4gIGJvcmRlcjogMXB4ICR7dmFycy5ib3JkZXJDb2xvckZvcklucHV0V2hlbkRpc2FibGVkfSBzb2xpZDtcbiAgYm9yZGVyLXJpZ2h0OiAwO1xuICBwYWRkaW5nOiAwICR7dmFycy5zcGFjaW5nU307XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGZvbnQtc2l6ZTogJHt2YXJzLmZvbnRTaXplRm9ySW5wdXR9O1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuYDtcblxudHlwZSBUR2V0QW1vdW50SW5wdXRTdHlsZXMgPSB7XG4gIGhhc0ZvY3VzOiBib29sZWFuO1xufSAmIFRJbnB1dFByb3BzO1xuY29uc3QgZ2V0QW1vdW50SW5wdXRTdHlsZXMgPSAocHJvcHM6IFRHZXRBbW91bnRJbnB1dFN0eWxlcykgPT4gW1xuICBnZXRJbnB1dFN0eWxlcyhwcm9wcyksXG4gIGNzc2BcbiAgICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAwO1xuICAgIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6IDA7XG4gICAgbWFyZ2luLWxlZnQ6IDA7XG5cbiAgICAmOjpwbGFjZWhvbGRlciB7XG4gICAgICBjb2xvcjogJHt2YXJzLnBsYWNlaG9sZGVyRm9udENvbG9yRm9ySW5wdXR9O1xuICAgIH1cbiAgYCxcbl07XG5cbnR5cGUgVEdldEhpZ2hQcmVjaXNpb25XcmFwcGVyU3R5bGVzID0ge1xuICBpc0Rpc2FibGVkPzogYm9vbGVhbjtcbn07XG5cbmNvbnN0IGdldEhpZ2hQcmVjaXNpb25XcmFwcGVyU3R5bGVzID0gKHtcbiAgaXNEaXNhYmxlZCxcbn06IFRHZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcykgPT4gY3NzYFxuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHRvcDogMDtcbiAgcmlnaHQ6IDA7XG4gIG1hcmdpbi1yaWdodDogJHt2YXJzLnNwYWNpbmdYc307XG4gIGhlaWdodDogMTAwJTtcbiAgZGlzcGxheTogZmxleDtcbiAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgY3Vyc29yOiAke2lzRGlzYWJsZWQgPyAnbm90LWFsbG93ZWQnIDogJ2RlZmF1bHQnfTtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG5gO1xuXG5leHBvcnQge1xuICBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyxcbiAgZ2V0Q3VycmVuY3lMYWJlbFN0eWxlcyxcbiAgZ2V0QW1vdW50SW5wdXRTdHlsZXMsXG59O1xuIl19 */")];
|
|
742
739
|
};
|
|
743
740
|
|
|
744
741
|
var getHighPrecisionWrapperStyles = function getHighPrecisionWrapperStyles(_ref) {
|
|
745
742
|
var isDisabled = _ref.isDisabled;
|
|
746
|
-
return /*#__PURE__*/react.css("position:absolute;top:0;right:0;margin-right:", designSystem.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,
|
|
743
|
+
return /*#__PURE__*/react.css("position:absolute;top:0;right:0;margin-right:", designSystem.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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm1vbmV5LWlucHV0LnN0eWxlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF5Q3lDIiwiZmlsZSI6Im1vbmV5LWlucHV0LnN0eWxlcy50cyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNzcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcbmltcG9ydCB7IGN1c3RvbVByb3BlcnRpZXMgYXMgdmFycyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2Rlc2lnbi1zeXN0ZW0nO1xuaW1wb3J0IHsgZ2V0SW5wdXRTdHlsZXMgfSBmcm9tICdAY29tbWVyY2V0b29scy11aWtpdC9pbnB1dC11dGlscyc7XG5pbXBvcnQgdHlwZSB7IFRJbnB1dFByb3BzIH0gZnJvbSAnLi9tb25leS1pbnB1dCc7XG5cbmNvbnN0IGdldEN1cnJlbmN5TGFiZWxTdHlsZXMgPSAoKSA9PiBjc3NgXG4gIGRpc3BsYXk6IGZsZXg7XG4gIGNvbG9yOiAke3ZhcnMuZm9udENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWR9O1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAke3ZhcnMuYmFja2dyb3VuZENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWR9O1xuICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAke3ZhcnMuYm9yZGVyUmFkaXVzRm9ySW5wdXR9O1xuICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAke3ZhcnMuYm9yZGVyUmFkaXVzRm9ySW5wdXR9O1xuICBib3JkZXI6IDFweCAke3ZhcnMuYm9yZGVyQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZH0gc29saWQ7XG4gIGJvcmRlci1yaWdodDogMDtcbiAgcGFkZGluZzogMCAke3ZhcnMuc3BhY2luZ1N9O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBmb250LXNpemU6ICR7dmFycy5mb250U2l6ZUZvcklucHV0fTtcbiAgYm94LXNpemluZzogYm9yZGVyLWJveDtcbmA7XG5cbnR5cGUgVEdldEFtb3VudElucHV0U3R5bGVzID0ge1xuICBoYXNGb2N1czogYm9vbGVhbjtcbn0gJiBUSW5wdXRQcm9wcztcbmNvbnN0IGdldEFtb3VudElucHV0U3R5bGVzID0gKHByb3BzOiBUR2V0QW1vdW50SW5wdXRTdHlsZXMpID0+IFtcbiAgZ2V0SW5wdXRTdHlsZXMocHJvcHMpLFxuICBjc3NgXG4gICAgYm9yZGVyLXRvcC1sZWZ0LXJhZGl1czogMDtcbiAgICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAwO1xuICAgIG1hcmdpbi1sZWZ0OiAwO1xuXG4gICAgJjo6cGxhY2Vob2xkZXIge1xuICAgICAgY29sb3I6ICR7dmFycy5wbGFjZWhvbGRlckZvbnRDb2xvckZvcklucHV0fTtcbiAgICB9XG4gIGAsXG5dO1xuXG50eXBlIFRHZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyA9IHtcbiAgaXNEaXNhYmxlZD86IGJvb2xlYW47XG59O1xuXG5jb25zdCBnZXRIaWdoUHJlY2lzaW9uV3JhcHBlclN0eWxlcyA9ICh7XG4gIGlzRGlzYWJsZWQsXG59OiBUR2V0SGlnaFByZWNpc2lvbldyYXBwZXJTdHlsZXMpID0+IGNzc2BcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICB0b3A6IDA7XG4gIHJpZ2h0OiAwO1xuICBtYXJnaW4tcmlnaHQ6ICR7dmFycy5zcGFjaW5nWHN9O1xuICBoZWlnaHQ6IDEwMCU7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGN1cnNvcjogJHtpc0Rpc2FibGVkID8gJ25vdC1hbGxvd2VkJyA6ICdkZWZhdWx0J307XG4gIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuYDtcblxuZXhwb3J0IHtcbiAgZ2V0SGlnaFByZWNpc2lvbldyYXBwZXJTdHlsZXMsXG4gIGdldEN1cnJlbmN5TGFiZWxTdHlsZXMsXG4gIGdldEFtb3VudElucHV0U3R5bGVzLFxufTtcbiJdfQ== */");
|
|
747
744
|
};
|
|
748
745
|
|
|
749
746
|
var messages = reactIntl.defineMessages({
|
|
@@ -756,16 +753,16 @@ var messages = reactIntl.defineMessages({
|
|
|
756
753
|
|
|
757
754
|
var _excluded = ["id"];
|
|
758
755
|
|
|
759
|
-
function ownKeys(object, enumerableOnly) { var keys = _Object$keys__default["default"](object); if (_Object$getOwnPropertySymbols__default["default"]) { var symbols = _Object$getOwnPropertySymbols__default["default"](object);
|
|
756
|
+
function ownKeys(object, enumerableOnly) { var keys = _Object$keys__default["default"](object); if (_Object$getOwnPropertySymbols__default["default"]) { var symbols = _Object$getOwnPropertySymbols__default["default"](object); enumerableOnly && (symbols = _filterInstanceProperty__default["default"](symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor__default["default"](object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
760
757
|
|
|
761
|
-
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]
|
|
758
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var _context14, _context15; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty__default["default"](_context14 = ownKeys(Object(source), !0)).call(_context14, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](target, _Object$getOwnPropertyDescriptors__default["default"](source)) : _forEachInstanceProperty__default["default"](_context15 = ownKeys(Object(source))).call(_context15, function (key) { _Object$defineProperty__default["default"](target, key, _Object$getOwnPropertyDescriptor__default["default"](source, key)); }); } return target; }
|
|
762
759
|
|
|
763
760
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
764
761
|
|
|
765
762
|
var TooltipWrapper = _styled__default["default"]("div", process.env.NODE_ENV === "production" ? {
|
|
766
|
-
target: "
|
|
763
|
+
target: "e1u90zjc0"
|
|
767
764
|
} : {
|
|
768
|
-
target: "
|
|
765
|
+
target: "e1u90zjc0",
|
|
769
766
|
label: "TooltipWrapper"
|
|
770
767
|
})(process.env.NODE_ENV === "production" ? {
|
|
771
768
|
name: "zjik7",
|
|
@@ -773,7 +770,7 @@ var TooltipWrapper = _styled__default["default"]("div", process.env.NODE_ENV ===
|
|
|
773
770
|
} : {
|
|
774
771
|
name: "zjik7",
|
|
775
772
|
styles: "display:flex",
|
|
776
|
-
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"]} */",
|
|
773
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAiCiC","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n} 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?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isDisabled?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n};\n\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(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          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          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(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\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            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) as string}\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              {\n                SingleValue: (innerProps) => (\n                  <SingleValue {...innerProps} id={id} />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(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: TValue, locale: string) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
|
|
777
774
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
778
775
|
});
|
|
779
776
|
|
|
@@ -787,7 +784,18 @@ var getPortalNode = function getPortalNode(id) {
|
|
|
787
784
|
|
|
788
785
|
var Portal = function Portal(props) {
|
|
789
786
|
var domNode = getPortalNode(props.id);
|
|
790
|
-
|
|
787
|
+
|
|
788
|
+
if (domNode) {
|
|
789
|
+
return /*#__PURE__*/ReactDOM__default["default"].createPortal(props.children, domNode);
|
|
790
|
+
}
|
|
791
|
+
|
|
792
|
+
return null;
|
|
793
|
+
};
|
|
794
|
+
|
|
795
|
+
Portal.propTypes = {
|
|
796
|
+
id: _pt__default["default"].string.isRequired,
|
|
797
|
+
children: _pt__default["default"].node,
|
|
798
|
+
isDisabled: _pt__default["default"].bool
|
|
791
799
|
};
|
|
792
800
|
|
|
793
801
|
var CurrencyLabel = function CurrencyLabel(props) {
|
|
@@ -798,11 +806,12 @@ var CurrencyLabel = function CurrencyLabel(props) {
|
|
|
798
806
|
});
|
|
799
807
|
};
|
|
800
808
|
|
|
801
|
-
CurrencyLabel.displayName = 'CurrencyLabel';
|
|
802
809
|
CurrencyLabel.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
803
|
-
id:
|
|
804
|
-
children:
|
|
810
|
+
id: _pt__default["default"].string.isRequired,
|
|
811
|
+
children: _pt__default["default"].node,
|
|
812
|
+
isDisabled: _pt__default["default"].bool
|
|
805
813
|
} : {};
|
|
814
|
+
CurrencyLabel.displayName = 'CurrencyLabel';
|
|
806
815
|
|
|
807
816
|
var _SingleValue = function SingleValue(_ref3) {
|
|
808
817
|
var id = _ref3.id,
|
|
@@ -816,12 +825,13 @@ var _SingleValue = function SingleValue(_ref3) {
|
|
|
816
825
|
}));
|
|
817
826
|
};
|
|
818
827
|
|
|
819
|
-
_SingleValue.displayName = 'SingleValue';
|
|
820
828
|
_SingleValue.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
821
|
-
id:
|
|
822
|
-
children:
|
|
823
|
-
} : {};
|
|
829
|
+
id: _pt__default["default"].string,
|
|
830
|
+
children: _pt__default["default"].node
|
|
831
|
+
} : {};
|
|
832
|
+
_SingleValue.displayName = 'SingleValue';
|
|
824
833
|
|
|
834
|
+
// overwrite styles of createSelectStyles
|
|
825
835
|
var createCurrencySelectStyles = function createCurrencySelectStyles(_ref4, theme) {
|
|
826
836
|
var hasWarning = _ref4.hasWarning,
|
|
827
837
|
hasError = _ref4.hasError,
|
|
@@ -967,7 +977,7 @@ var parseRawAmountToNumber = function parseRawAmountToNumber(rawAmount, locale)
|
|
|
967
977
|
var normalizedAmount = String(rawAmount).replace(new RegExp("[^0-9".concat(fractionsSeparator, "]"), 'g'), '') // we just keep the numbers and the fraction symbol
|
|
968
978
|
.replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float
|
|
969
979
|
|
|
970
|
-
return _parseFloat__default["default"](normalizedAmount
|
|
980
|
+
return _parseFloat__default["default"](normalizedAmount);
|
|
971
981
|
}; // Turns the user input into a value the MoneyInput can pass up through onChange
|
|
972
982
|
// In case the number of fraction digits contained in "amount" exceeds the
|
|
973
983
|
// number of fraction digits the currency uses, it will emit a price of
|
|
@@ -1063,13 +1073,19 @@ var getCurrencyDropdownName = function getCurrencyDropdownName(name) {
|
|
|
1063
1073
|
return name ? "".concat(name, ".currencyCode") : undefined;
|
|
1064
1074
|
};
|
|
1065
1075
|
|
|
1076
|
+
var defaultProps = {
|
|
1077
|
+
currencies: [],
|
|
1078
|
+
horizontalConstraint: 'scale',
|
|
1079
|
+
menuPortalZIndex: 1
|
|
1080
|
+
};
|
|
1081
|
+
|
|
1066
1082
|
var _ref = process.env.NODE_ENV === "production" ? {
|
|
1067
1083
|
name: "pw7jst",
|
|
1068
1084
|
styles: "position:relative;width:100%"
|
|
1069
1085
|
} : {
|
|
1070
1086
|
name: "h5hvsa-MoneyInput",
|
|
1071
1087
|
styles: "position:relative;width:100%;label:MoneyInput;",
|
|
1072
|
-
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"]} */",
|
|
1088
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAgwBkB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n} 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?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isDisabled?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n};\n\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(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          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          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(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\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            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) as string}\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              {\n                SingleValue: (innerProps) => (\n                  <SingleValue {...innerProps} id={id} />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(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: TValue, locale: string) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
|
|
1073
1089
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1074
1090
|
};
|
|
1075
1091
|
|
|
@@ -1079,7 +1095,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
|
1079
1095
|
} : {
|
|
1080
1096
|
name: "1w49f4-MoneyInput",
|
|
1081
1097
|
styles: "font-family:inherit;width:100%;position:relative;display:flex;label:MoneyInput;",
|
|
1082
|
-
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"]} */",
|
|
1098
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAitBgB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n} 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?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isDisabled?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n};\n\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(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          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          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(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\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            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) as string}\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              {\n                SingleValue: (innerProps) => (\n                  <SingleValue {...innerProps} id={id} />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(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: TValue, locale: string) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */",
|
|
1083
1099
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
1084
1100
|
};
|
|
1085
1101
|
|
|
@@ -1098,8 +1114,13 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1098
1114
|
amountHasFocus = _useToggleState4[0],
|
|
1099
1115
|
toggleAmountHasFocus = _useToggleState4[1];
|
|
1100
1116
|
|
|
1101
|
-
var containerRef = react$1.useRef();
|
|
1102
|
-
var amountInputRef = react$1.useRef();
|
|
1117
|
+
var containerRef = react$1.useRef(null);
|
|
1118
|
+
var amountInputRef = react$1.useRef(null);
|
|
1119
|
+
|
|
1120
|
+
if (!props.isReadOnly) {
|
|
1121
|
+
process.env.NODE_ENV !== "production" ? utils.warning(typeof props.onChange === 'function', 'MoneyInput: "onChange" is required when is not read only.') : void 0;
|
|
1122
|
+
}
|
|
1123
|
+
|
|
1103
1124
|
var onFocus = props.onFocus;
|
|
1104
1125
|
var handleAmountFocus = react$1.useCallback(function () {
|
|
1105
1126
|
if (onFocus) onFocus({
|
|
@@ -1126,7 +1147,6 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1126
1147
|
if (String(formattedAmount) !== amount) {
|
|
1127
1148
|
// We need to emit an event with the now formatted value
|
|
1128
1149
|
var fakeEvent = {
|
|
1129
|
-
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
1130
1150
|
persist: function persist() {},
|
|
1131
1151
|
target: {
|
|
1132
1152
|
id: MoneyInput.getAmountInputId(props.id),
|
|
@@ -1141,7 +1161,6 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1141
1161
|
var handleAmountChange = react$1.useCallback(function (event) {
|
|
1142
1162
|
if (utils.isNumberish(event.target.value)) {
|
|
1143
1163
|
onChange({
|
|
1144
|
-
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
1145
1164
|
persist: function persist() {},
|
|
1146
1165
|
target: {
|
|
1147
1166
|
id: MoneyInput.getAmountInputId(props.id),
|
|
@@ -1155,7 +1174,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1155
1174
|
var currencyCode = option.value;
|
|
1156
1175
|
|
|
1157
1176
|
if (props.value.currencyCode !== currencyCode) {
|
|
1158
|
-
var _context6;
|
|
1177
|
+
var _context6, _amountInputRef$curre;
|
|
1159
1178
|
|
|
1160
1179
|
// When the user changes from a currency with 3 fraction digits to
|
|
1161
1180
|
// a currency with 2 fraction digits, and when the input value was
|
|
@@ -1167,10 +1186,9 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1167
1186
|
// or while the amount is invalid. In these cases, we don't attempt to
|
|
1168
1187
|
// format the amount.
|
|
1169
1188
|
|
|
1170
|
-
var nextAmount = isNaN(formattedAmount) ? props.value.amount : formattedAmount; // change currency code
|
|
1189
|
+
var nextAmount = isNaN(Number(formattedAmount)) ? props.value.amount : formattedAmount; // change currency code
|
|
1171
1190
|
|
|
1172
1191
|
var fakeCurrencyEvent = {
|
|
1173
|
-
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
1174
1192
|
persist: function persist() {},
|
|
1175
1193
|
target: {
|
|
1176
1194
|
id: MoneyInput.getCurrencyDropdownId(props.id),
|
|
@@ -1182,7 +1200,6 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1182
1200
|
|
|
1183
1201
|
if (props.value.amount !== nextAmount) {
|
|
1184
1202
|
onChange({
|
|
1185
|
-
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
1186
1203
|
persist: function persist() {},
|
|
1187
1204
|
target: {
|
|
1188
1205
|
id: MoneyInput.getAmountInputId(props.id),
|
|
@@ -1192,7 +1209,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1192
1209
|
});
|
|
1193
1210
|
}
|
|
1194
1211
|
|
|
1195
|
-
amountInputRef.current.focus();
|
|
1212
|
+
(_amountInputRef$curre = amountInputRef.current) === null || _amountInputRef$curre === void 0 ? void 0 : _amountInputRef$curre.focus();
|
|
1196
1213
|
}
|
|
1197
1214
|
}, [intl.locale, onChange, props.id, props.name, props.value.amount, props.value.currencyCode]);
|
|
1198
1215
|
var handleCurrencyFocus = react$1.useCallback(function () {
|
|
@@ -1247,9 +1264,11 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1247
1264
|
var isHighPrecision = !MoneyInput.isEmpty(props.value) && MoneyInput.isHighPrecision(props.value, intl.locale);
|
|
1248
1265
|
var onBlur = props.onBlur;
|
|
1249
1266
|
var handleContainerBlur = react$1.useCallback(function (event) {
|
|
1267
|
+
var _containerRef$current;
|
|
1268
|
+
|
|
1250
1269
|
// ensures that both fields are marked as touched when one of them
|
|
1251
1270
|
// is blurred
|
|
1252
|
-
if (typeof onBlur === 'function' && !containerRef.current.contains(event.relatedTarget)) {
|
|
1271
|
+
if (typeof onBlur === 'function' && !((_containerRef$current = containerRef.current) !== null && _containerRef$current !== void 0 && _containerRef$current.contains(event.relatedTarget))) {
|
|
1253
1272
|
onBlur({
|
|
1254
1273
|
target: {
|
|
1255
1274
|
id: MoneyInput.getCurrencyDropdownId(props.id),
|
|
@@ -1287,19 +1306,15 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1287
1306
|
isDisabled: props.isDisabled,
|
|
1288
1307
|
isSearchable: false,
|
|
1289
1308
|
components: {
|
|
1290
|
-
// eslint-disable-next-line react/display-name
|
|
1291
1309
|
SingleValue: function SingleValue(innerProps) {
|
|
1292
1310
|
return jsxRuntime.jsx(_SingleValue, _objectSpread(_objectSpread({}, innerProps), {}, {
|
|
1293
1311
|
id: id
|
|
1294
1312
|
}));
|
|
1295
1313
|
},
|
|
1296
|
-
// eslint-disable-next-line react/display-name
|
|
1297
1314
|
Input: function Input(ownProps) {
|
|
1298
|
-
return (
|
|
1299
|
-
|
|
1300
|
-
|
|
1301
|
-
}))
|
|
1302
|
-
);
|
|
1315
|
+
return jsxRuntime.jsx(Select.components.Input, _objectSpread(_objectSpread({}, ownProps), {}, {
|
|
1316
|
+
readOnly: props.isReadOnly
|
|
1317
|
+
}));
|
|
1303
1318
|
},
|
|
1304
1319
|
DropdownIndicator: selectUtils.DropdownIndicator
|
|
1305
1320
|
},
|
|
@@ -1326,7 +1341,7 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1326
1341
|
css: [getAmountInputStyles(_objectSpread(_objectSpread({}, props), {}, {
|
|
1327
1342
|
hasFocus: hasFocus
|
|
1328
1343
|
})), // accounts for size of icon
|
|
1329
|
-
props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.customProperties.spacingL, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.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"]} */"],
|
|
1344
|
+
props.hasHighPrecisionBadge && isHighPrecision && /*#__PURE__*/react.css("padding-right:", designSystem.customProperties.spacingL, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AAkxBmB","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n} 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?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isDisabled?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n};\n\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(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          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          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(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\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            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) as string}\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              {\n                SingleValue: (innerProps) => (\n                  <SingleValue {...innerProps} id={id} />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(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: TValue, locale: string) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:MoneyInput;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["money-input.tsx"],"names":[],"mappings":"AA6wBY","file":"money-input.tsx","sourcesContent":["import { useRef, useCallback, type ReactNode } from 'react';\nimport ReactDOM from 'react-dom';\nimport has from 'lodash/has';\nimport Select, {\n  components,\n  type SingleValueProps,\n  type Props as ReactSelectProps,\n} from 'react-select';\nimport { useIntl } from 'react-intl';\nimport { css, useTheme, type Theme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { customProperties as vars } from '@commercetools-uikit/design-system';\nimport {\n  warning,\n  isNumberish,\n  filterDataAttributes,\n} 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?: string) => `portal-${id}`;\nconst getPortalNode = (id?: string) =>\n  document.querySelector(`#${getPortalId(id)}`);\n\ntype TLabel = {\n  id: string;\n  children?: ReactNode;\n  isDisabled?: boolean;\n};\n\nconst Portal = (props: TLabel) => {\n  const domNode = getPortalNode(props.id);\n  if (domNode) {\n    return ReactDOM.createPortal(props.children, domNode);\n  }\n  return null;\n};\n\nconst CurrencyLabel = (props: TLabel) => (\n  <label htmlFor={props.id} css={getCurrencyLabelStyles()}>\n    {props.children}\n  </label>\n);\n\nCurrencyLabel.displayName = 'CurrencyLabel';\n\ntype TSingleValue = {\n  id?: string;\n  children?: ReactNode;\n} & SingleValueProps;\n\nconst SingleValue = ({ id, ...props }: TSingleValue) => (\n  <components.SingleValue {...props}>\n    <label htmlFor={id}>{props.children}</label>\n  </components.SingleValue>\n);\n\nSingleValue.displayName = 'SingleValue';\n\ntype TCreateCurrencySelectStyles = (input: TInputProps, theme: Theme) => void;\n\nexport type TInputProps = {\n  isDisabled?: boolean;\n  hasError?: boolean;\n  hasWarning?: boolean;\n  isReadOnly?: boolean;\n  hasFocus?: boolean;\n  menuPortalZIndex?: number;\n  theme?: Theme;\n};\n\ntype TBase = {\n  backgroundColor?: string;\n  color?: string;\n};\n// overwrite styles of createSelectStyles\nconst createCurrencySelectStyles: TCreateCurrencySelectStyles = (\n  { hasWarning, hasError, isDisabled, isReadOnly, hasFocus, menuPortalZIndex },\n  theme\n) => {\n  const selectStyles = createSelectStyles(\n    {\n      hasWarning,\n      hasError,\n      menuPortalZIndex,\n    },\n    theme\n  );\n  return {\n    ...selectStyles,\n    control: (base: TBase, state: ReactSelectProps) => ({\n      ...selectStyles.control(base, state),\n      borderTopRightRadius: '0',\n      borderBottomRightRadius: '0',\n      borderRight: '0',\n      minWidth: '72px',\n      borderColor: (() => {\n        if (isDisabled)\n          return `${vars.borderColorForInputWhenDisabled} !important`;\n        if (hasError) return vars.borderColorForInputWhenError;\n        if (hasWarning) return vars.borderColorForInputWhenWarning;\n        if (hasFocus) return vars.borderColorForInputWhenFocused;\n        if (isReadOnly)\n          return `${vars.borderColorForInputWhenReadonly} !important`;\n        return vars.borderColorForInput;\n      })(),\n      cursor: (() => {\n        if (isDisabled) return 'not-allowed';\n        if (isReadOnly) return `default`;\n        return 'pointer';\n      })(),\n      backgroundColor: (() => {\n        if (isReadOnly) return vars.backgroundColorForInput;\n        return base.backgroundColor;\n      })(),\n    }),\n    singleValue: (base: TBase) => ({\n      ...base,\n      marginLeft: 0,\n      maxWidth: 'initial',\n      color: (() => {\n        if (isDisabled) return vars.fontColorForInputWhenDisabled;\n        if (hasError) return vars.fontColorForInputWhenError;\n        if (hasWarning) return vars.fontColorForInputWhenWarning;\n        if (isReadOnly) return vars.fontColorForInputWhenReadonly;\n        return base.color;\n      })(),\n    }),\n    dropdownIndicator: () => ({\n      fill: isReadOnly ? vars.fontColorForInputWhenReadonly : '',\n    }),\n  };\n};\n\n// The MoneyInput component always operates on a value consisting of:\n//   { amount: String, currencyCode: String }\n//\n// The amount may only use a dot as the decimal separator.\n// The currencyCode must be supported by the API.\n//\n// The MoneyInput does not do any validation on its own. It only serves as a way\n// to get the amount and currencyCode input from the user. Validation is always\n// up to the parent.\n//\n// The CTP API supports prices two types of prices: centPrecision and\n// highPrecision. The MoneyInput itself does not know about these. However,\n// it has two static methods defined (convertToMoneyValue and parseMoneyValue),\n// which can be used to convert between MoneyInput value and the MoneyValue\n// supported by the API.\n// Some places in the API do not support highPrecision prices, but the\n// convertToMoneyValue will always return either a centPrecision or a\n// highPrecision price. It's up the MoneyInput's parent to show a validation\n// error in case a highPrecision price is used.\n//\n// A value is considered as to have highPrecision when the number of supplied\n// fraction digits exceed the number of fraction digits the currency uses. For\n// example, 42.00 € is always a centPrecision price, while 42.001 € is always a\n// highPrecision price. It is not possible to hae 42.00 € as a highPrecision\n// price.\n//\n// The first time the component renders, we want to try to show the centAmount\n// as a formatted number. To achieve this, the parseMoneyValue function can\n// be used to turn the API value into a value the MoneyInput understands.\n// During this transformation, the money value will get formatted into \"amount\".\n//\n// When the user changes the value, we don't want to format again. We only format\n// in case the user blurs the field. This avoids many edge cases where the\n// formatting would mess with the user's input.\n//\n//\n// A full example of an MoneyValue with centPrecision would be\n// {\n//   \"type\": \"centPrecision\",\n//   \"currencyCode\": \"EUR\",\n//   \"centAmount\": 4200,\n//   \"fractionDigits\": 2\n// }\n// which equals 42.00 €\n//\n// A full example of an MoneyValue with highPrecision would be\n// {\n//  \"type\": \"highPrecision\",\n//  \"currencyCode\": \"EUR\",\n//  \"centAmount\": 1,\n//  \"preciseAmount\": 123456,\n//  \"fractionDigits\": 7\n// }\n// which equals 0.0123456 €\n\n// Parsing\n// Since most users are not careful about how they enter values, we will parse\n// both `.` and `,` as decimal separators.\n// When a value is `1.000,00` we parse it as `1000`.\n// When a value is `1,000.00` we also parse it as `1000`.\n//\n// This means the highest amount always wins. We do this by comparing the last\n// position of `.` and `,`. Whatever occurs later is used as the decimal\n// separator.\nexport const parseRawAmountToNumber = (rawAmount: string, locale: string) => {\n  let fractionsSeparator;\n\n  if (locale) {\n    fractionsSeparator = (2.5) // we need any number with fractions, so that we know what is the fraction\n      .toLocaleString(locale) // \"symbol\" for the provided locale\n      .replace(/\\d/g, ''); // then we remove the numbers and keep the \"symbol\"\n  } else {\n    const lastDot = String(rawAmount).lastIndexOf('.');\n    const lastComma = String(rawAmount).lastIndexOf(',');\n    fractionsSeparator = lastComma > lastDot ? ',' : '.';\n  }\n\n  fractionsSeparator = fractionsSeparator === '.' ? '\\\\.' : fractionsSeparator; // here we escape the '.' to use it as regex\n  // The raw amount with only one sparator\n  const normalizedAmount = String(rawAmount)\n    .replace(new RegExp(`[^0-9${fractionsSeparator}]`, 'g'), '') // we just keep the numbers and the fraction symbol\n    .replace(fractionsSeparator, '.'); // then we change whatever `fractionsSeparator` was to `.` so we can parse it as float\n\n  return parseFloat(normalizedAmount);\n};\n\n// Turns the user input into a value the MoneyInput can pass up through onChange\n// In case the number of fraction digits contained in \"amount\" exceeds the\n// number of fraction digits the currency uses, it will emit a price of\n// type \"highPrecision\" instead of the regular \"centPrecision\".\n// It will return \"null\" in case an invalid value is entered.\n// The value is invalid when\n//  - no amount was entered\n//  - an invalid amount was entered\n//  - no currency was selected\n//\n// This function expects the \"amount\" to be a trimmed value.\n\ntype TCurrencyCode = keyof typeof currencies;\n\nexport const createMoneyValue = (\n  currencyCode: TCurrencyCode,\n  rawAmount: string,\n  locale: string\n) => {\n  if (!currencyCode) return null;\n\n  const currency = currencies[currencyCode];\n  if (!currency) return null;\n\n  if (rawAmount.length === 0 || !isNumberish(rawAmount)) return null;\n\n  warning(\n    locale || currency.fractionDigits !== 0,\n    `A locale must be provided when currency has no fraction digits (${currencyCode})`\n  );\n  const amountAsNumber = parseRawAmountToNumber(rawAmount, locale);\n  if (isNaN(amountAsNumber)) return null;\n\n  // The cent amount is rounded to the currencie's default number\n  // of fraction digits for prices with high precision.\n  //\n  // Additionally, JavaScript is sometimes incorrect when multiplying floats,\n  //   e.g. 2.49 * 100 -> 249.00000000000003\n  // While inaccuracy from multiplying floating point numbers is a\n  // general problem in JS, we can avoid it by cutting off all\n  // decimals. This is possible since cents is the base unit, so we\n  // operate on integers anyways\n  // Also we should the round the value to ensure that we come close\n  // to the nearest decimal value\n  // ref: https://github.com/commercetools/merchant-center-frontend/pull/770\n  const centAmount = Math.trunc(\n    Math.round(amountAsNumber * 10 ** currency.fractionDigits)\n  );\n\n  const fractionDigitsOfAmount =\n    // The conversion to a string will always use a dot as the separator.\n    // That means we don't have to handle a comma.\n    String(amountAsNumber).indexOf('.') === -1\n      ? 0\n      : String(amountAsNumber).length - String(amountAsNumber).indexOf('.') - 1;\n\n  if (fractionDigitsOfAmount > currency.fractionDigits) {\n    return {\n      type: 'highPrecision',\n      currencyCode,\n      centAmount,\n      preciseAmount: parseInt(\n        // Here we need to convert  a number like 8.066652 to its centamount\n        // We could do that by multiplying it with 10 ** number-of-fraction-digits\n        // but then we'll run into problems with JavaScript's floating point\n        // number precision and end up with 8066651.9999999, and then parseInt\n        // cuts off the remainder.\n        // So instead of using maths to convert the number, we just replace\n        // the dot inside the number which does the same thing.\n        // We don't need to replace \",\" as well, as numbers always us a dot\n        // when converted using String().\n        //\n        // The mathematical way: amountAsNumber * 10 ** fractionDigitsOfAmount,\n        String(amountAsNumber).replace('.', ''),\n        10\n      ),\n      fractionDigits: fractionDigitsOfAmount,\n    };\n  }\n\n  return {\n    type: 'centPrecision',\n    currencyCode,\n    centAmount,\n    fractionDigits: currency.fractionDigits,\n  };\n};\n\ntype TMoneyValue = {\n  type: string;\n  currencyCode: TCurrencyCode;\n  centAmount: number;\n  preciseAmount: number;\n  fractionDigits: number;\n};\n\nconst getAmountAsNumberFromMoneyValue = (moneyValue: TMoneyValue) =>\n  moneyValue.type === 'highPrecision'\n    ? moneyValue.preciseAmount / 10 ** moneyValue.fractionDigits\n    : moneyValue.centAmount /\n      10 ** currencies[moneyValue.currencyCode].fractionDigits;\n\n// gets called with a string and should return a formatted string\nconst formatAmount = (\n  rawAmount: string,\n  currencyCode: TCurrencyCode,\n  locale: string\n) => {\n  // fallback in case the user didn't enter an amount yet (or it's invalid)\n  const moneyValue = (createMoneyValue(currencyCode, rawAmount, locale) || {\n    currencyCode,\n    centAmount: NaN,\n  }) as TMoneyValue;\n\n  const amount = getAmountAsNumberFromMoneyValue(moneyValue);\n\n  const fractionDigits = moneyValue.preciseAmount\n    ? moneyValue.fractionDigits\n    : currencies[moneyValue.currencyCode].fractionDigits;\n\n  return isNaN(amount)\n    ? ''\n    : amount.toLocaleString(locale, { minimumFractionDigits: fractionDigits });\n};\n\nconst getAmountInputName = (name?: string) =>\n  name ? `${name}.amount` : undefined;\nconst getCurrencyDropdownName = (name?: string) =>\n  name ? `${name}.currencyCode` : undefined;\n\ntype TValue = {\n  amount: string;\n  currencyCode: TCurrencyCode;\n};\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    value?: string | string[] | null;\n  };\n  persist?: () => void;\n};\n\ntype TMoneyInputProps = {\n  /**\n   * Used as HTML id property. An id is auto-generated when it is not specified.\n   */\n  id?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  /**\n   * The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).\n   */\n  name?: string;\n  /**\n   * Value of the input. Consists of the currency code and an amount. `amount` is a string representing the amount. A dot has to be used as the decimal separator.\n   */\n  value: TValue;\n  /**\n   * List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.\n   */\n  currencies: string[];\n  /**\n   * Placeholder text for the input\n   */\n  placeholder?: string;\n  /**\n   * Called when input is blurred\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when input is focused\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).\n   */\n  isDisabled?: boolean;\n  /**\n   * Indicates that the field is displaying read-only content\n   */\n  isReadOnly?: boolean;\n  /**\n   * Focus the input on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Called with the event of the input or dropdown when either the currency or the amount have changed.\n   * <br />\n   * Signature: `(event) => void`\n   */\n  onChange: (event: TEvent) => void;\n  /**\n   * Dom element to portal the currency select menu to\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuPortalTarget?: ReactSelectProps['menuPortalTarget'];\n  /**\n   * z-index value for the currency select menu portal\n   */\n  menuPortalZIndex?: number;\n  /**\n   * whether the menu should block scroll while open\n   * <br>\n   * [Props from React select was used](https://react-select.com/props)\n   */\n  menuShouldBlockScroll: ReactSelectProps['menuShouldBlockScroll'];\n  /**\n   * Indicates that input has errors\n   */\n  hasError?: boolean;\n  /**\n   * Control to indicate on the input if there are selected values that are potentially invalid\n   */\n  hasWarning?: boolean;\n  /**\n   * Shows high precision badge in case current value uses high precision.\n   */\n  hasHighPrecisionBadge?: boolean;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n};\n\nconst defaultProps: Pick<\n  TMoneyInputProps,\n  'currencies' | 'horizontalConstraint' | 'menuPortalZIndex'\n> = {\n  currencies: [],\n  horizontalConstraint: 'scale',\n  menuPortalZIndex: 1,\n};\n\nconst MoneyInput = (props: TMoneyInputProps) => {\n  const intl = useIntl();\n  const [currencyHasFocus, toggleCurrencyHasFocus] = useToggleState(false);\n  const [amountHasFocus, toggleAmountHasFocus] = useToggleState(false);\n\n  const containerRef = useRef<HTMLDivElement>(null);\n  const amountInputRef = useRef<HTMLInputElement>(null);\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'MoneyInput: \"onChange\" is required when is not read only.'\n    );\n  }\n\n  const { onFocus } = props;\n  const handleAmountFocus = useCallback(() => {\n    if (onFocus)\n      onFocus({\n        target: {\n          id: MoneyInput.getAmountInputId(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          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          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(Number(formattedAmount))\n          ? props.value.amount\n          : formattedAmount;\n\n        // change currency code\n        const fakeCurrencyEvent = {\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            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) as string}\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              {\n                SingleValue: (innerProps) => (\n                  <SingleValue {...innerProps} id={id} />\n                ),\n                Input: (ownProps) => (\n                  <components.Input {...ownProps} readOnly={props.isReadOnly} />\n                ),\n                DropdownIndicator,\n              } as ReactSelectProps['components']\n            }\n            options={options}\n            menuIsOpen={props.isReadOnly ? false : undefined}\n            placeholder=\"\"\n            styles={currencySelectStyles as ReactSelectProps['styles']}\n            onFocus={handleCurrencyFocus}\n            menuPortalTarget={props.menuPortalTarget}\n            menuShouldBlockScroll={props.menuShouldBlockScroll}\n            onBlur={handleCurrencyBlur}\n            onChange={handleCurrencyChange}\n            data-testid=\"currency-dropdown\"\n          />\n        )}\n        <div\n          css={css`\n            position: relative;\n            width: 100%;\n          `}\n        >\n          <input\n            ref={amountInputRef}\n            id={MoneyInput.getAmountInputId(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: TValue, locale: string) =>\n  createMoneyValue(\n    value.currencyCode,\n    typeof value.amount === 'string' ? value.amount.trim() : '',\n    locale\n  );\n\nMoneyInput.parseMoneyValue = (moneyValue: TMoneyValue, locale: string) => {\n  if (!moneyValue) return { currencyCode: '', amount: '' };\n\n  warning(\n    typeof locale === 'string',\n    'MoneyInput.parseMoneyValue: A locale must be passed as the second argument'\n  );\n\n  warning(\n    typeof moneyValue === 'object',\n    'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined'\n  );\n\n  warning(\n    typeof moneyValue.currencyCode === 'string',\n    'MoneyInput.parseMoneyValue: Value must contain \"currencyCode\"'\n  );\n\n  warning(\n    has(currencies, moneyValue.currencyCode),\n    'MoneyInput.parseMoneyValue: Value must use known currency code'\n  );\n\n  warning(\n    // highPrecision or centPrecision values must be set\n    typeof moneyValue.centAmount === 'number' ||\n      (typeof moneyValue.preciseAmount === 'number' &&\n        typeof moneyValue.fractionDigits === 'number'),\n    'MoneyInput.parseMoneyValue: Value must contain \"amount\"'\n  );\n\n  const amount = formatAmount(\n    getAmountAsNumberFromMoneyValue(moneyValue).toLocaleString(locale, {\n      minimumFractionDigits: moneyValue.fractionDigits,\n    }),\n    moneyValue.currencyCode,\n    locale\n  );\n\n  return { amount, currencyCode: moneyValue.currencyCode };\n};\n\nMoneyInput.isEmpty = (formValue: TValue) =>\n  !formValue ||\n  formValue.amount.trim() === '' ||\n  formValue.currencyCode.trim() === '';\n\nMoneyInput.isHighPrecision = (formValue: TValue, locale: string) => {\n  warning(\n    !MoneyInput.isEmpty(formValue),\n    'MoneyValue.isHighPrecision may not be called with an empty money value.'\n  );\n  const moneyValue = MoneyInput.convertToMoneyValue(formValue, locale);\n  return moneyValue && moneyValue.type === 'highPrecision';\n};\n\nMoneyInput.isTouched = (touched: TValue) =>\n  Boolean(touched && touched.currencyCode && touched.amount);\n\nMoneyInput.defaultProps = defaultProps;\n\nexport default MoneyInput;\n"]} */"],
|
|
1330
1345
|
placeholder: props.placeholder,
|
|
1331
1346
|
onChange: handleAmountChange,
|
|
1332
1347
|
onBlur: handleAmountBlur,
|
|
@@ -1368,6 +1383,28 @@ var MoneyInput = function MoneyInput(props) {
|
|
|
1368
1383
|
});
|
|
1369
1384
|
};
|
|
1370
1385
|
|
|
1386
|
+
MoneyInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
1387
|
+
id: _pt__default["default"].string,
|
|
1388
|
+
autoComplete: _pt__default["default"].string,
|
|
1389
|
+
name: _pt__default["default"].string,
|
|
1390
|
+
value: _pt__default["default"].shape({
|
|
1391
|
+
amount: _pt__default["default"].string.isRequired,
|
|
1392
|
+
currencyCode: _pt__default["default"].any.isRequired
|
|
1393
|
+
}).isRequired,
|
|
1394
|
+
currencies: _pt__default["default"].arrayOf(_pt__default["default"].string).isRequired,
|
|
1395
|
+
placeholder: _pt__default["default"].string,
|
|
1396
|
+
onBlur: _pt__default["default"].func,
|
|
1397
|
+
onFocus: _pt__default["default"].func,
|
|
1398
|
+
isDisabled: _pt__default["default"].bool,
|
|
1399
|
+
isReadOnly: _pt__default["default"].bool,
|
|
1400
|
+
isAutofocussed: _pt__default["default"].bool,
|
|
1401
|
+
onChange: _pt__default["default"].func.isRequired,
|
|
1402
|
+
menuPortalZIndex: _pt__default["default"].number,
|
|
1403
|
+
hasError: _pt__default["default"].bool,
|
|
1404
|
+
hasWarning: _pt__default["default"].bool,
|
|
1405
|
+
hasHighPrecisionBadge: _pt__default["default"].bool,
|
|
1406
|
+
horizontalConstraint: _pt__default["default"].oneOf([3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto'])
|
|
1407
|
+
} : {};
|
|
1371
1408
|
MoneyInput.displayName = 'MoneyInput';
|
|
1372
1409
|
MoneyInput.getAmountInputId = getAmountInputName;
|
|
1373
1410
|
MoneyInput.getCurrencyDropdownId = getCurrencyDropdownName;
|
|
@@ -1384,7 +1421,7 @@ MoneyInput.parseMoneyValue = function (moneyValue, locale) {
|
|
|
1384
1421
|
amount: ''
|
|
1385
1422
|
};
|
|
1386
1423
|
process.env.NODE_ENV !== "production" ? utils.warning(typeof locale === 'string', 'MoneyInput.parseMoneyValue: A locale must be passed as the second argument') : void 0;
|
|
1387
|
-
process.env.NODE_ENV !== "production" ? utils.warning(
|
|
1424
|
+
process.env.NODE_ENV !== "production" ? utils.warning(typeof moneyValue === 'object', 'MoneyInput.parseMoneyValue: Value must be passed as an object or be undefined') : void 0;
|
|
1388
1425
|
process.env.NODE_ENV !== "production" ? utils.warning(typeof moneyValue.currencyCode === 'string', 'MoneyInput.parseMoneyValue: Value must contain "currencyCode"') : void 0;
|
|
1389
1426
|
process.env.NODE_ENV !== "production" ? utils.warning(has__default["default"](currencies, moneyValue.currencyCode), 'MoneyInput.parseMoneyValue: Value must use known currency code') : void 0;
|
|
1390
1427
|
process.env.NODE_ENV !== "production" ? utils.warning( // highPrecision or centPrecision values must be set
|
|
@@ -1414,118 +1451,11 @@ MoneyInput.isTouched = function (touched) {
|
|
|
1414
1451
|
return Boolean(touched && touched.currencyCode && touched.amount);
|
|
1415
1452
|
};
|
|
1416
1453
|
|
|
1417
|
-
MoneyInput.
|
|
1418
|
-
/**
|
|
1419
|
-
* Used as HTML id property. An id is auto-generated when it is not specified.
|
|
1420
|
-
*/
|
|
1421
|
-
id: PropTypes__default["default"].string,
|
|
1422
|
-
|
|
1423
|
-
/**
|
|
1424
|
-
* Used as HTML `autocomplete` property
|
|
1425
|
-
*/
|
|
1426
|
-
autoComplete: PropTypes__default["default"].string,
|
|
1427
|
-
|
|
1428
|
-
/**
|
|
1429
|
-
* The prefix used to create a HTML `name` property for the amount input field (`${name}.amount`) and the currency dropdown (`${name}.currencyCode`).
|
|
1430
|
-
*/
|
|
1431
|
-
name: PropTypes__default["default"].string,
|
|
1432
|
-
|
|
1433
|
-
/**
|
|
1434
|
-
* 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.
|
|
1435
|
-
*/
|
|
1436
|
-
value: PropTypes__default["default"].shape({
|
|
1437
|
-
amount: PropTypes__default["default"].string.isRequired,
|
|
1438
|
-
currencyCode: PropTypes__default["default"].string.isRequired
|
|
1439
|
-
}).isRequired,
|
|
1440
|
-
|
|
1441
|
-
/**
|
|
1442
|
-
* List of possible currencies. When not provided or empty, the component renders a label with the value's currency instead of a dropdown.
|
|
1443
|
-
*/
|
|
1444
|
-
currencies: PropTypes__default["default"].arrayOf(PropTypes__default["default"].string).isRequired,
|
|
1445
|
-
|
|
1446
|
-
/**
|
|
1447
|
-
* Placeholder text for the input
|
|
1448
|
-
*/
|
|
1449
|
-
placeholder: PropTypes__default["default"].string,
|
|
1450
|
-
|
|
1451
|
-
/**
|
|
1452
|
-
* Called when input is blurred
|
|
1453
|
-
*/
|
|
1454
|
-
onBlur: PropTypes__default["default"].func,
|
|
1455
|
-
|
|
1456
|
-
/**
|
|
1457
|
-
* Called when input is focused
|
|
1458
|
-
*/
|
|
1459
|
-
onFocus: PropTypes__default["default"].func,
|
|
1460
|
-
|
|
1461
|
-
/**
|
|
1462
|
-
* Indicates that the input cannot be modified (e.g not authorized, or changes currently saving).
|
|
1463
|
-
*/
|
|
1464
|
-
isDisabled: PropTypes__default["default"].bool,
|
|
1465
|
-
|
|
1466
|
-
/**
|
|
1467
|
-
* Indicates that the field is displaying read-only content
|
|
1468
|
-
*/
|
|
1469
|
-
isReadOnly: PropTypes__default["default"].bool,
|
|
1470
|
-
|
|
1471
|
-
/**
|
|
1472
|
-
* Focus the input on initial render
|
|
1473
|
-
*/
|
|
1474
|
-
isAutofocussed: PropTypes__default["default"].bool,
|
|
1475
|
-
|
|
1476
|
-
/**
|
|
1477
|
-
* Called with the event of the input or dropdown when either the currency or the amount have changed.
|
|
1478
|
-
* <br />
|
|
1479
|
-
* Signature: `(event) => void`
|
|
1480
|
-
*/
|
|
1481
|
-
onChange: requiredIf__default["default"](PropTypes__default["default"].func, function (props) {
|
|
1482
|
-
return !props.isReadOnly;
|
|
1483
|
-
}),
|
|
1484
|
-
|
|
1485
|
-
/**
|
|
1486
|
-
* Dom element to portal the currency select menu to
|
|
1487
|
-
*/
|
|
1488
|
-
menuPortalTarget: PropTypes__default["default"].instanceOf(utils.SafeHTMLElement),
|
|
1489
|
-
|
|
1490
|
-
/**
|
|
1491
|
-
* z-index value for the currency select menu portal
|
|
1492
|
-
*/
|
|
1493
|
-
menuPortalZIndex: PropTypes__default["default"].number,
|
|
1494
|
-
|
|
1495
|
-
/**
|
|
1496
|
-
* whether the menu should block scroll while open
|
|
1497
|
-
*/
|
|
1498
|
-
menuShouldBlockScroll: PropTypes__default["default"].bool,
|
|
1499
|
-
|
|
1500
|
-
/**
|
|
1501
|
-
* Indicates that input has errors
|
|
1502
|
-
*/
|
|
1503
|
-
hasError: PropTypes__default["default"].bool,
|
|
1504
|
-
|
|
1505
|
-
/**
|
|
1506
|
-
* Control to indicate on the input if there are selected values that are potentially invalid
|
|
1507
|
-
*/
|
|
1508
|
-
hasWarning: PropTypes__default["default"].bool,
|
|
1509
|
-
|
|
1510
|
-
/**
|
|
1511
|
-
* Shows high precision badge in case current value uses high precision.
|
|
1512
|
-
*/
|
|
1513
|
-
hasHighPrecisionBadge: PropTypes__default["default"].bool,
|
|
1514
|
-
|
|
1515
|
-
/**
|
|
1516
|
-
* Horizontal size limit of the input fields.
|
|
1517
|
-
*/
|
|
1518
|
-
horizontalConstraint: PropTypes__default["default"].oneOf([3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto'])
|
|
1519
|
-
} : {};
|
|
1520
|
-
MoneyInput.defaultProps = {
|
|
1521
|
-
currencies: [],
|
|
1522
|
-
horizontalConstraint: 'scale',
|
|
1523
|
-
menuPortalZIndex: 1
|
|
1524
|
-
};
|
|
1454
|
+
MoneyInput.defaultProps = defaultProps;
|
|
1525
1455
|
var MoneyInput$1 = MoneyInput;
|
|
1526
1456
|
|
|
1527
1457
|
// NOTE: This string will be replaced on build time with the package version.
|
|
1528
|
-
var version = "
|
|
1458
|
+
var version = "13.0.0";
|
|
1529
1459
|
|
|
1530
1460
|
exports["default"] = MoneyInput$1;
|
|
1531
1461
|
exports.version = version;
|