@commercetools-uikit/localized-rich-text-input 13.0.1 → 13.0.4
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 +75 -45
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.d.ts +2 -0
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.dev.js +179 -175
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.prod.js +78 -53
- package/dist/commercetools-uikit-localized-rich-text-input.esm.js +178 -173
- package/dist/declarations/src/editor.d.ts +48 -0
- package/dist/declarations/src/editor.styles.d.ts +15 -0
- package/dist/declarations/src/editor.types.d.ts +103 -0
- package/dist/declarations/src/index.d.ts +2 -0
- package/dist/declarations/src/localized-rich-text-input.d.ts +55 -0
- package/dist/declarations/src/required-value-error-message.d.ts +5 -0
- package/dist/declarations/src/rich-text-input.d.ts +65 -0
- package/dist/declarations/src/version.d.ts +2 -0
- package/package.json +16 -18
|
@@ -5,17 +5,13 @@ import _forEachInstanceProperty from '@babel/runtime-corejs3/core-js-stable/inst
|
|
|
5
5
|
import _Object$getOwnPropertyDescriptors from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptors';
|
|
6
6
|
import _Object$defineProperties from '@babel/runtime-corejs3/core-js-stable/object/define-properties';
|
|
7
7
|
import _Object$defineProperty from '@babel/runtime-corejs3/core-js-stable/object/define-property';
|
|
8
|
-
import _taggedTemplateLiteral from '@babel/runtime-corejs3/helpers/esm/taggedTemplateLiteral';
|
|
9
8
|
import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
|
|
10
9
|
import _defineProperty from '@babel/runtime-corejs3/helpers/esm/defineProperty';
|
|
10
|
+
import _pt from 'prop-types';
|
|
11
11
|
import _reduceInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/reduce';
|
|
12
12
|
import _Object$keys from '@babel/runtime-corejs3/core-js-stable/object/keys';
|
|
13
13
|
import _mapInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/map';
|
|
14
|
-
import _concatInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/concat';
|
|
15
14
|
import { cloneElement, useRef, useState, useCallback, useEffect, PureComponent, useReducer } from 'react';
|
|
16
|
-
import PropTypes from 'prop-types';
|
|
17
|
-
import requiredIf from 'react-required-if';
|
|
18
|
-
import { oneLine } from 'common-tags';
|
|
19
15
|
import { css, useTheme } from '@emotion/react';
|
|
20
16
|
import Stack from '@commercetools-uikit/spacings-stack';
|
|
21
17
|
import Constraints from '@commercetools-uikit/constraints';
|
|
@@ -23,16 +19,17 @@ import { usePrevious, useToggleState } from '@commercetools-uikit/hooks';
|
|
|
23
19
|
import { sortLanguages, getHasErrorOnRemainingLanguages, getHasWarningOnRemainingLanguages, createLocalizedDataAttributes, getId, getName, isTouched } from '@commercetools-uikit/localized-utils';
|
|
24
20
|
import { messagesMultilineInput, messagesLocalizedInput, LocalizedInputToggle } from '@commercetools-uikit/input-utils';
|
|
25
21
|
import { HiddenInput, RichTextBody, html, richTextPlugins, localized } from '@commercetools-uikit/rich-text-utils';
|
|
22
|
+
import { warning, filterDataAttributes } from '@commercetools-uikit/utils';
|
|
26
23
|
import _Reflect$construct from '@babel/runtime-corejs3/core-js-stable/reflect/construct';
|
|
27
24
|
import _classCallCheck from '@babel/runtime-corejs3/helpers/esm/classCallCheck';
|
|
28
25
|
import _createClass from '@babel/runtime-corejs3/helpers/esm/createClass';
|
|
29
26
|
import _inherits from '@babel/runtime-corejs3/helpers/esm/inherits';
|
|
30
27
|
import _possibleConstructorReturn from '@babel/runtime-corejs3/helpers/esm/possibleConstructorReturn';
|
|
31
28
|
import _getPrototypeOf from '@babel/runtime-corejs3/helpers/esm/getPrototypeOf';
|
|
29
|
+
import _concatInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/concat';
|
|
32
30
|
import _setTimeout from '@babel/runtime-corejs3/core-js-stable/set-timeout';
|
|
33
31
|
import pick from 'lodash/pick';
|
|
34
|
-
import { Editor as Editor$
|
|
35
|
-
import { filterDataAttributes } from '@commercetools-uikit/utils';
|
|
32
|
+
import { Editor as Editor$2 } from 'slate-react';
|
|
36
33
|
import _styled from '@emotion/styled/base';
|
|
37
34
|
import { useIntl, FormattedMessage } from 'react-intl';
|
|
38
35
|
import { customProperties, designTokens } from '@commercetools-uikit/design-system';
|
|
@@ -50,29 +47,29 @@ function _objectSpread$4(target) { for (var i = 1; i < arguments.length; i++) {
|
|
|
50
47
|
function _EMOTION_STRINGIFIED_CSS_ERROR__$2() { 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)."; }
|
|
51
48
|
|
|
52
49
|
var EditorLanguageLabel = _styled("label", process.env.NODE_ENV === "production" ? {
|
|
53
|
-
target: "
|
|
50
|
+
target: "ew063c2"
|
|
54
51
|
} : {
|
|
55
|
-
target: "
|
|
52
|
+
target: "ew063c2",
|
|
56
53
|
label: "EditorLanguageLabel"
|
|
57
54
|
})(function (props) {
|
|
58
55
|
var overwrittenVars = _objectSpread$4(_objectSpread$4({}, customProperties), props.theme);
|
|
59
56
|
|
|
60
|
-
return /*#__PURE__*/css("white-space:nowrap;flex:0;color:", overwrittenVars[designTokens.fontColorForInputWhenDisabled], ";line-height:calc(\n ", customProperties.sizeHeightInput, " - 2 *\n ", customProperties.borderRadius1, "\n );background-color:", overwrittenVars[designTokens.backgroundColorForInputWhenDisabled], ";border-top-left-radius:", overwrittenVars[designTokens.borderRadiusForInput], ";border-bottom-left-radius:", overwrittenVars[designTokens.borderRadiusForInput], ";border:1px ", overwrittenVars[designTokens.borderColorForInputWhenDisabled], " solid;padding:0 ", customProperties.spacingS, ";transition:border-color ", customProperties.transitionStandard, ",background-color ", customProperties.transitionStandard, ",color ", customProperties.transitionStandard, ";border-right:0;box-shadow:none;appearance:none;cursor:inherit;" + (process.env.NODE_ENV === "production" ? "" : ";label:EditorLanguageLabel;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
61
|
-
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
57
|
+
return /*#__PURE__*/css("white-space:nowrap;flex:0;color:", overwrittenVars[designTokens.fontColorForInputWhenDisabled], ";line-height:calc(\n ", customProperties.sizeHeightInput, " - 2 *\n ", customProperties.borderRadius1, "\n );background-color:", overwrittenVars[designTokens.backgroundColorForInputWhenDisabled], ";border-top-left-radius:", overwrittenVars[designTokens.borderRadiusForInput], ";border-bottom-left-radius:", overwrittenVars[designTokens.borderRadiusForInput], ";border:1px ", overwrittenVars[designTokens.borderColorForInputWhenDisabled], " solid;padding:0 ", customProperties.spacingS, ";transition:border-color ", customProperties.transitionStandard, ",background-color ", customProperties.transitionStandard, ",color ", customProperties.transitionStandard, ";border-right:0;box-shadow:none;appearance:none;cursor:inherit;" + (process.env.NODE_ENV === "production" ? "" : ";label:EditorLanguageLabel;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImVkaXRvci5zdHlsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY1kiLCJmaWxlIjoiZWRpdG9yLnN0eWxlcy50cyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNzcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7XG4gIGN1c3RvbVByb3BlcnRpZXMsXG4gIGRlc2lnblRva2Vucyxcbn0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgdHlwZSB7IFRFZGl0b3JQcm9wcyB9IGZyb20gJy4vZWRpdG9yJztcblxuY29uc3QgRWRpdG9yTGFuZ3VhZ2VMYWJlbCA9IHN0eWxlZC5sYWJlbCgocHJvcHMpID0+IHtcbiAgY29uc3Qgb3ZlcndyaXR0ZW5WYXJzID0ge1xuICAgIC4uLmN1c3RvbVByb3BlcnRpZXMsXG4gICAgLi4ucHJvcHMudGhlbWUsXG4gIH07XG5cbiAgcmV0dXJuIGNzc2BcbiAgICAvKiBhdm9pZCB3cmFwcGluZyBsYWJlbCBvbnRvIG5ldyBsaW5lcyAqL1xuICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gICAgZmxleDogMDtcbiAgICBjb2xvcjogJHtvdmVyd3JpdHRlblZhcnNbZGVzaWduVG9rZW5zLmZvbnRDb2xvckZvcklucHV0V2hlbkRpc2FibGVkXX07XG4gICAgbGluZS1oZWlnaHQ6IGNhbGMoXG4gICAgICAke2N1c3RvbVByb3BlcnRpZXMuc2l6ZUhlaWdodElucHV0fSAtIDIgKlxuICAgICAgICAke2N1c3RvbVByb3BlcnRpZXMuYm9yZGVyUmFkaXVzMX1cbiAgICApO1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR7b3ZlcndyaXR0ZW5WYXJzW1xuICAgICAgZGVzaWduVG9rZW5zLmJhY2tncm91bmRDb2xvckZvcklucHV0V2hlbkRpc2FibGVkXG4gICAgXX07XG4gICAgYm9yZGVyLXRvcC1sZWZ0LXJhZGl1czogJHtvdmVyd3JpdHRlblZhcnNbXG4gICAgICBkZXNpZ25Ub2tlbnMuYm9yZGVyUmFkaXVzRm9ySW5wdXRcbiAgICBdfTtcbiAgICBib3JkZXItYm90dG9tLWxlZnQtcmFkaXVzOiAke292ZXJ3cml0dGVuVmFyc1tcbiAgICAgIGRlc2lnblRva2Vucy5ib3JkZXJSYWRpdXNGb3JJbnB1dFxuICAgIF19O1xuICAgIGJvcmRlcjogMXB4ICR7b3ZlcndyaXR0ZW5WYXJzW2Rlc2lnblRva2Vucy5ib3JkZXJDb2xvckZvcklucHV0V2hlbkRpc2FibGVkXX1cbiAgICAgIHNvbGlkO1xuICAgIHBhZGRpbmc6IDAgJHtjdXN0b21Qcm9wZXJ0aWVzLnNwYWNpbmdTfTtcbiAgICB0cmFuc2l0aW9uOiBib3JkZXItY29sb3IgJHtjdXN0b21Qcm9wZXJ0aWVzLnRyYW5zaXRpb25TdGFuZGFyZH0sXG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yICR7Y3VzdG9tUHJvcGVydGllcy50cmFuc2l0aW9uU3RhbmRhcmR9LFxuICAgICAgY29sb3IgJHtjdXN0b21Qcm9wZXJ0aWVzLnRyYW5zaXRpb25TdGFuZGFyZH07XG4gICAgYm9yZGVyLXJpZ2h0OiAwO1xuICAgIGJveC1zaGFkb3c6IG5vbmU7XG4gICAgYXBwZWFyYW5jZTogbm9uZTtcblxuICAgIC8qIGN1cnNvciBzaG91bGQgYmUgaW5oZXJpdGVkIGZyb20gcGFyZW50LFxuICAgICogR0lWRU4gcGFyZW50IGhhcyAnbm90LWFsbG93ZWQnIGN1cnNvclxuICAgICogVEhFTiB0aGUgbGFuZ3VhZ2UgbGFiZWwgc2hvdWxkIGFsc28gaGF2ZSB0aGF0IChpbnN0ZWFkIG9mIGxhYmVsJ3MgZGVmYXVsdCBjdXJzb3IpXG4gICAgKi9cbiAgICBjdXJzb3I6IGluaGVyaXQ7XG4gIGA7XG59KTtcblxuY29uc3QgRWRpdG9yV3JhcHBlciA9IHN0eWxlZC5kaXY8XG4gIFBpY2s8VEVkaXRvclByb3BzLCAnaXNEaXNhYmxlZCcgfCAnaXNSZWFkT25seSc+XG4+YFxuICB3aWR0aDogMTAwJTtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICBkaXNwbGF5OiBmbGV4O1xuICBjdXJzb3I6ICR7KHByb3BzKSA9PlxuICAgIHByb3BzLmlzRGlzYWJsZWQgfHwgcHJvcHMuaXNSZWFkT25seSA/ICdub3QtYWxsb3dlZCcgOiAnaW5oZXJpdCd9O1xuYDtcblxuY29uc3QgVG9nZ2xlQnV0dG9uV3JhcHBlciA9IHN0eWxlZC5kaXZgXG4gIGZsZXg6IDA7XG4gIGRpc3BsYXk6IGZsZXg7XG5gO1xuXG5leHBvcnQgeyBFZGl0b3JMYW5ndWFnZUxhYmVsLCBFZGl0b3JXcmFwcGVyLCBUb2dnbGVCdXR0b25XcmFwcGVyIH07XG4iXX0= */");
|
|
58
|
+
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImVkaXRvci5zdHlsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBUTRCIiwiZmlsZSI6ImVkaXRvci5zdHlsZXMudHMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQge1xuICBjdXN0b21Qcm9wZXJ0aWVzLFxuICBkZXNpZ25Ub2tlbnMsXG59IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2Rlc2lnbi1zeXN0ZW0nO1xuaW1wb3J0IHR5cGUgeyBURWRpdG9yUHJvcHMgfSBmcm9tICcuL2VkaXRvcic7XG5cbmNvbnN0IEVkaXRvckxhbmd1YWdlTGFiZWwgPSBzdHlsZWQubGFiZWwoKHByb3BzKSA9PiB7XG4gIGNvbnN0IG92ZXJ3cml0dGVuVmFycyA9IHtcbiAgICAuLi5jdXN0b21Qcm9wZXJ0aWVzLFxuICAgIC4uLnByb3BzLnRoZW1lLFxuICB9O1xuXG4gIHJldHVybiBjc3NgXG4gICAgLyogYXZvaWQgd3JhcHBpbmcgbGFiZWwgb250byBuZXcgbGluZXMgKi9cbiAgICB3aGl0ZS1zcGFjZTogbm93cmFwO1xuICAgIGZsZXg6IDA7XG4gICAgY29sb3I6ICR7b3ZlcndyaXR0ZW5WYXJzW2Rlc2lnblRva2Vucy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZF19O1xuICAgIGxpbmUtaGVpZ2h0OiBjYWxjKFxuICAgICAgJHtjdXN0b21Qcm9wZXJ0aWVzLnNpemVIZWlnaHRJbnB1dH0gLSAyICpcbiAgICAgICAgJHtjdXN0b21Qcm9wZXJ0aWVzLmJvcmRlclJhZGl1czF9XG4gICAgKTtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAke292ZXJ3cml0dGVuVmFyc1tcbiAgICAgIGRlc2lnblRva2Vucy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZFxuICAgIF19O1xuICAgIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7b3ZlcndyaXR0ZW5WYXJzW1xuICAgICAgZGVzaWduVG9rZW5zLmJvcmRlclJhZGl1c0ZvcklucHV0XG4gICAgXX07XG4gICAgYm9yZGVyLWJvdHRvbS1sZWZ0LXJhZGl1czogJHtvdmVyd3JpdHRlblZhcnNbXG4gICAgICBkZXNpZ25Ub2tlbnMuYm9yZGVyUmFkaXVzRm9ySW5wdXRcbiAgICBdfTtcbiAgICBib3JkZXI6IDFweCAke292ZXJ3cml0dGVuVmFyc1tkZXNpZ25Ub2tlbnMuYm9yZGVyQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZF19XG4gICAgICBzb2xpZDtcbiAgICBwYWRkaW5nOiAwICR7Y3VzdG9tUHJvcGVydGllcy5zcGFjaW5nU307XG4gICAgdHJhbnNpdGlvbjogYm9yZGVyLWNvbG9yICR7Y3VzdG9tUHJvcGVydGllcy50cmFuc2l0aW9uU3RhbmRhcmR9LFxuICAgICAgYmFja2dyb3VuZC1jb2xvciAke2N1c3RvbVByb3BlcnRpZXMudHJhbnNpdGlvblN0YW5kYXJkfSxcbiAgICAgIGNvbG9yICR7Y3VzdG9tUHJvcGVydGllcy50cmFuc2l0aW9uU3RhbmRhcmR9O1xuICAgIGJvcmRlci1yaWdodDogMDtcbiAgICBib3gtc2hhZG93OiBub25lO1xuICAgIGFwcGVhcmFuY2U6IG5vbmU7XG5cbiAgICAvKiBjdXJzb3Igc2hvdWxkIGJlIGluaGVyaXRlZCBmcm9tIHBhcmVudCxcbiAgICAqIEdJVkVOIHBhcmVudCBoYXMgJ25vdC1hbGxvd2VkJyBjdXJzb3JcbiAgICAqIFRIRU4gdGhlIGxhbmd1YWdlIGxhYmVsIHNob3VsZCBhbHNvIGhhdmUgdGhhdCAoaW5zdGVhZCBvZiBsYWJlbCdzIGRlZmF1bHQgY3Vyc29yKVxuICAgICovXG4gICAgY3Vyc29yOiBpbmhlcml0O1xuICBgO1xufSk7XG5cbmNvbnN0IEVkaXRvcldyYXBwZXIgPSBzdHlsZWQuZGl2PFxuICBQaWNrPFRFZGl0b3JQcm9wcywgJ2lzRGlzYWJsZWQnIHwgJ2lzUmVhZE9ubHknPlxuPmBcbiAgd2lkdGg6IDEwMCU7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgZGlzcGxheTogZmxleDtcbiAgY3Vyc29yOiAkeyhwcm9wcykgPT5cbiAgICBwcm9wcy5pc0Rpc2FibGVkIHx8IHByb3BzLmlzUmVhZE9ubHkgPyAnbm90LWFsbG93ZWQnIDogJ2luaGVyaXQnfTtcbmA7XG5cbmNvbnN0IFRvZ2dsZUJ1dHRvbldyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBmbGV4OiAwO1xuICBkaXNwbGF5OiBmbGV4O1xuYDtcblxuZXhwb3J0IHsgRWRpdG9yTGFuZ3VhZ2VMYWJlbCwgRWRpdG9yV3JhcHBlciwgVG9nZ2xlQnV0dG9uV3JhcHBlciB9O1xuIl19 */");
|
|
62
59
|
|
|
63
60
|
var EditorWrapper = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
64
|
-
target: "
|
|
61
|
+
target: "ew063c1"
|
|
65
62
|
} : {
|
|
66
|
-
target: "
|
|
63
|
+
target: "ew063c1",
|
|
67
64
|
label: "EditorWrapper"
|
|
68
65
|
})("width:100%;position:relative;display:flex;cursor:", function (props) {
|
|
69
66
|
return props.isDisabled || props.isReadOnly ? 'not-allowed' : 'inherit';
|
|
70
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
67
|
+
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImVkaXRvci5zdHlsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBb0RDIiwiZmlsZSI6ImVkaXRvci5zdHlsZXMudHMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQge1xuICBjdXN0b21Qcm9wZXJ0aWVzLFxuICBkZXNpZ25Ub2tlbnMsXG59IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2Rlc2lnbi1zeXN0ZW0nO1xuaW1wb3J0IHR5cGUgeyBURWRpdG9yUHJvcHMgfSBmcm9tICcuL2VkaXRvcic7XG5cbmNvbnN0IEVkaXRvckxhbmd1YWdlTGFiZWwgPSBzdHlsZWQubGFiZWwoKHByb3BzKSA9PiB7XG4gIGNvbnN0IG92ZXJ3cml0dGVuVmFycyA9IHtcbiAgICAuLi5jdXN0b21Qcm9wZXJ0aWVzLFxuICAgIC4uLnByb3BzLnRoZW1lLFxuICB9O1xuXG4gIHJldHVybiBjc3NgXG4gICAgLyogYXZvaWQgd3JhcHBpbmcgbGFiZWwgb250byBuZXcgbGluZXMgKi9cbiAgICB3aGl0ZS1zcGFjZTogbm93cmFwO1xuICAgIGZsZXg6IDA7XG4gICAgY29sb3I6ICR7b3ZlcndyaXR0ZW5WYXJzW2Rlc2lnblRva2Vucy5mb250Q29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZF19O1xuICAgIGxpbmUtaGVpZ2h0OiBjYWxjKFxuICAgICAgJHtjdXN0b21Qcm9wZXJ0aWVzLnNpemVIZWlnaHRJbnB1dH0gLSAyICpcbiAgICAgICAgJHtjdXN0b21Qcm9wZXJ0aWVzLmJvcmRlclJhZGl1czF9XG4gICAgKTtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAke292ZXJ3cml0dGVuVmFyc1tcbiAgICAgIGRlc2lnblRva2Vucy5iYWNrZ3JvdW5kQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZFxuICAgIF19O1xuICAgIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6ICR7b3ZlcndyaXR0ZW5WYXJzW1xuICAgICAgZGVzaWduVG9rZW5zLmJvcmRlclJhZGl1c0ZvcklucHV0XG4gICAgXX07XG4gICAgYm9yZGVyLWJvdHRvbS1sZWZ0LXJhZGl1czogJHtvdmVyd3JpdHRlblZhcnNbXG4gICAgICBkZXNpZ25Ub2tlbnMuYm9yZGVyUmFkaXVzRm9ySW5wdXRcbiAgICBdfTtcbiAgICBib3JkZXI6IDFweCAke292ZXJ3cml0dGVuVmFyc1tkZXNpZ25Ub2tlbnMuYm9yZGVyQ29sb3JGb3JJbnB1dFdoZW5EaXNhYmxlZF19XG4gICAgICBzb2xpZDtcbiAgICBwYWRkaW5nOiAwICR7Y3VzdG9tUHJvcGVydGllcy5zcGFjaW5nU307XG4gICAgdHJhbnNpdGlvbjogYm9yZGVyLWNvbG9yICR7Y3VzdG9tUHJvcGVydGllcy50cmFuc2l0aW9uU3RhbmRhcmR9LFxuICAgICAgYmFja2dyb3VuZC1jb2xvciAke2N1c3RvbVByb3BlcnRpZXMudHJhbnNpdGlvblN0YW5kYXJkfSxcbiAgICAgIGNvbG9yICR7Y3VzdG9tUHJvcGVydGllcy50cmFuc2l0aW9uU3RhbmRhcmR9O1xuICAgIGJvcmRlci1yaWdodDogMDtcbiAgICBib3gtc2hhZG93OiBub25lO1xuICAgIGFwcGVhcmFuY2U6IG5vbmU7XG5cbiAgICAvKiBjdXJzb3Igc2hvdWxkIGJlIGluaGVyaXRlZCBmcm9tIHBhcmVudCxcbiAgICAqIEdJVkVOIHBhcmVudCBoYXMgJ25vdC1hbGxvd2VkJyBjdXJzb3JcbiAgICAqIFRIRU4gdGhlIGxhbmd1YWdlIGxhYmVsIHNob3VsZCBhbHNvIGhhdmUgdGhhdCAoaW5zdGVhZCBvZiBsYWJlbCdzIGRlZmF1bHQgY3Vyc29yKVxuICAgICovXG4gICAgY3Vyc29yOiBpbmhlcml0O1xuICBgO1xufSk7XG5cbmNvbnN0IEVkaXRvcldyYXBwZXIgPSBzdHlsZWQuZGl2PFxuICBQaWNrPFRFZGl0b3JQcm9wcywgJ2lzRGlzYWJsZWQnIHwgJ2lzUmVhZE9ubHknPlxuPmBcbiAgd2lkdGg6IDEwMCU7XG4gIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgZGlzcGxheTogZmxleDtcbiAgY3Vyc29yOiAkeyhwcm9wcykgPT5cbiAgICBwcm9wcy5pc0Rpc2FibGVkIHx8IHByb3BzLmlzUmVhZE9ubHkgPyAnbm90LWFsbG93ZWQnIDogJ2luaGVyaXQnfTtcbmA7XG5cbmNvbnN0IFRvZ2dsZUJ1dHRvbldyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBmbGV4OiAwO1xuICBkaXNwbGF5OiBmbGV4O1xuYDtcblxuZXhwb3J0IHsgRWRpdG9yTGFuZ3VhZ2VMYWJlbCwgRWRpdG9yV3JhcHBlciwgVG9nZ2xlQnV0dG9uV3JhcHBlciB9O1xuIl19 */"));
|
|
71
68
|
|
|
72
69
|
var ToggleButtonWrapper = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
73
|
-
target: "
|
|
70
|
+
target: "ew063c0"
|
|
74
71
|
} : {
|
|
75
|
-
target: "
|
|
72
|
+
target: "ew063c0",
|
|
76
73
|
label: "ToggleButtonWrapper"
|
|
77
74
|
})(process.env.NODE_ENV === "production" ? {
|
|
78
75
|
name: "ejz79s",
|
|
@@ -80,7 +77,7 @@ var ToggleButtonWrapper = _styled("div", process.env.NODE_ENV === "production" ?
|
|
|
80
77
|
} : {
|
|
81
78
|
name: "ejz79s",
|
|
82
79
|
styles: "flex:0;display:flex",
|
|
83
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
80
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImVkaXRvci5zdHlsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNERzQyIsImZpbGUiOiJlZGl0b3Iuc3R5bGVzLnRzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHtcbiAgY3VzdG9tUHJvcGVydGllcyxcbiAgZGVzaWduVG9rZW5zLFxufSBmcm9tICdAY29tbWVyY2V0b29scy11aWtpdC9kZXNpZ24tc3lzdGVtJztcbmltcG9ydCB0eXBlIHsgVEVkaXRvclByb3BzIH0gZnJvbSAnLi9lZGl0b3InO1xuXG5jb25zdCBFZGl0b3JMYW5ndWFnZUxhYmVsID0gc3R5bGVkLmxhYmVsKChwcm9wcykgPT4ge1xuICBjb25zdCBvdmVyd3JpdHRlblZhcnMgPSB7XG4gICAgLi4uY3VzdG9tUHJvcGVydGllcyxcbiAgICAuLi5wcm9wcy50aGVtZSxcbiAgfTtcblxuICByZXR1cm4gY3NzYFxuICAgIC8qIGF2b2lkIHdyYXBwaW5nIGxhYmVsIG9udG8gbmV3IGxpbmVzICovXG4gICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICBmbGV4OiAwO1xuICAgIGNvbG9yOiAke292ZXJ3cml0dGVuVmFyc1tkZXNpZ25Ub2tlbnMuZm9udENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWRdfTtcbiAgICBsaW5lLWhlaWdodDogY2FsYyhcbiAgICAgICR7Y3VzdG9tUHJvcGVydGllcy5zaXplSGVpZ2h0SW5wdXR9IC0gMiAqXG4gICAgICAgICR7Y3VzdG9tUHJvcGVydGllcy5ib3JkZXJSYWRpdXMxfVxuICAgICk7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHtvdmVyd3JpdHRlblZhcnNbXG4gICAgICBkZXNpZ25Ub2tlbnMuYmFja2dyb3VuZENvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWRcbiAgICBdfTtcbiAgICBib3JkZXItdG9wLWxlZnQtcmFkaXVzOiAke292ZXJ3cml0dGVuVmFyc1tcbiAgICAgIGRlc2lnblRva2Vucy5ib3JkZXJSYWRpdXNGb3JJbnB1dFxuICAgIF19O1xuICAgIGJvcmRlci1ib3R0b20tbGVmdC1yYWRpdXM6ICR7b3ZlcndyaXR0ZW5WYXJzW1xuICAgICAgZGVzaWduVG9rZW5zLmJvcmRlclJhZGl1c0ZvcklucHV0XG4gICAgXX07XG4gICAgYm9yZGVyOiAxcHggJHtvdmVyd3JpdHRlblZhcnNbZGVzaWduVG9rZW5zLmJvcmRlckNvbG9yRm9ySW5wdXRXaGVuRGlzYWJsZWRdfVxuICAgICAgc29saWQ7XG4gICAgcGFkZGluZzogMCAke2N1c3RvbVByb3BlcnRpZXMuc3BhY2luZ1N9O1xuICAgIHRyYW5zaXRpb246IGJvcmRlci1jb2xvciAke2N1c3RvbVByb3BlcnRpZXMudHJhbnNpdGlvblN0YW5kYXJkfSxcbiAgICAgIGJhY2tncm91bmQtY29sb3IgJHtjdXN0b21Qcm9wZXJ0aWVzLnRyYW5zaXRpb25TdGFuZGFyZH0sXG4gICAgICBjb2xvciAke2N1c3RvbVByb3BlcnRpZXMudHJhbnNpdGlvblN0YW5kYXJkfTtcbiAgICBib3JkZXItcmlnaHQ6IDA7XG4gICAgYm94LXNoYWRvdzogbm9uZTtcbiAgICBhcHBlYXJhbmNlOiBub25lO1xuXG4gICAgLyogY3Vyc29yIHNob3VsZCBiZSBpbmhlcml0ZWQgZnJvbSBwYXJlbnQsXG4gICAgKiBHSVZFTiBwYXJlbnQgaGFzICdub3QtYWxsb3dlZCcgY3Vyc29yXG4gICAgKiBUSEVOIHRoZSBsYW5ndWFnZSBsYWJlbCBzaG91bGQgYWxzbyBoYXZlIHRoYXQgKGluc3RlYWQgb2YgbGFiZWwncyBkZWZhdWx0IGN1cnNvcilcbiAgICAqL1xuICAgIGN1cnNvcjogaW5oZXJpdDtcbiAgYDtcbn0pO1xuXG5jb25zdCBFZGl0b3JXcmFwcGVyID0gc3R5bGVkLmRpdjxcbiAgUGljazxURWRpdG9yUHJvcHMsICdpc0Rpc2FibGVkJyB8ICdpc1JlYWRPbmx5Jz5cbj5gXG4gIHdpZHRoOiAxMDAlO1xuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGN1cnNvcjogJHsocHJvcHMpID0+XG4gICAgcHJvcHMuaXNEaXNhYmxlZCB8fCBwcm9wcy5pc1JlYWRPbmx5ID8gJ25vdC1hbGxvd2VkJyA6ICdpbmhlcml0J307XG5gO1xuXG5jb25zdCBUb2dnbGVCdXR0b25XcmFwcGVyID0gc3R5bGVkLmRpdmBcbiAgZmxleDogMDtcbiAgZGlzcGxheTogZmxleDtcbmA7XG5cbmV4cG9ydCB7IEVkaXRvckxhbmd1YWdlTGFiZWwsIEVkaXRvcldyYXBwZXIsIFRvZ2dsZUJ1dHRvbldyYXBwZXIgfTtcbiJdfQ== */",
|
|
84
81
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$2
|
|
85
82
|
});
|
|
86
83
|
|
|
@@ -92,9 +89,9 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__$1() { return "You have tried to string
|
|
|
92
89
|
var COLLAPSED_HEIGHT = 32;
|
|
93
90
|
|
|
94
91
|
var LeftColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
95
|
-
target: "
|
|
92
|
+
target: "el9zors2"
|
|
96
93
|
} : {
|
|
97
|
-
target: "
|
|
94
|
+
target: "el9zors2",
|
|
98
95
|
label: "LeftColumn"
|
|
99
96
|
})(process.env.NODE_ENV === "production" ? {
|
|
100
97
|
name: "147rp59",
|
|
@@ -102,14 +99,14 @@ var LeftColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
102
99
|
} : {
|
|
103
100
|
name: "147rp59",
|
|
104
101
|
styles: "flex:1;display:flex;align-items:flex-start",
|
|
105
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AA4B6B","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */",
|
|
102
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAsC6B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
106
103
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
107
104
|
});
|
|
108
105
|
|
|
109
106
|
var RightColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
110
|
-
target: "
|
|
107
|
+
target: "el9zors1"
|
|
111
108
|
} : {
|
|
112
|
-
target: "
|
|
109
|
+
target: "el9zors1",
|
|
113
110
|
label: "RightColumn"
|
|
114
111
|
})(process.env.NODE_ENV === "production" ? {
|
|
115
112
|
name: "1m04uhl",
|
|
@@ -117,14 +114,14 @@ var RightColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
117
114
|
} : {
|
|
118
115
|
name: "1m04uhl",
|
|
119
116
|
styles: "position:relative;flex:0;display:flex;align-items:flex-start",
|
|
120
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AAkC8B","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */",
|
|
117
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA4C8B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
121
118
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
122
119
|
});
|
|
123
120
|
|
|
124
121
|
var Row = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
125
|
-
target: "
|
|
122
|
+
target: "el9zors0"
|
|
126
123
|
} : {
|
|
127
|
-
target: "
|
|
124
|
+
target: "el9zors0",
|
|
128
125
|
label: "Row"
|
|
129
126
|
})(process.env.NODE_ENV === "production" ? {
|
|
130
127
|
name: "skgbeu",
|
|
@@ -132,7 +129,7 @@ var Row = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
132
129
|
} : {
|
|
133
130
|
name: "skgbeu",
|
|
134
131
|
styles: "display:flex;justify-content:flex-end",
|
|
135
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AAyCsB","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */",
|
|
132
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAmDsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
136
133
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
137
134
|
});
|
|
138
135
|
|
|
@@ -142,14 +139,20 @@ var _ref$1 = process.env.NODE_ENV === "production" ? {
|
|
|
142
139
|
} : {
|
|
143
140
|
name: "16ohn6x-container",
|
|
144
141
|
styles: "flex:auto;width:0;border-top-left-radius:0;border-bottom-left-radius:0;label:container;",
|
|
145
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AA6HgC","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */",
|
|
142
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA6KgC","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
146
143
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
147
144
|
};
|
|
148
145
|
|
|
149
|
-
var Editor = function Editor(props) {
|
|
146
|
+
var Editor$1 = function Editor(props) {
|
|
147
|
+
var _props$editor, _props$editor2, _props$editor3, _props$editor4;
|
|
148
|
+
|
|
150
149
|
var intl = useIntl();
|
|
151
150
|
var ref = useRef();
|
|
152
|
-
var prevIsFocused = usePrevious(props.editor.value.selection.isFocused);
|
|
151
|
+
var prevIsFocused = usePrevious((_props$editor = props.editor) === null || _props$editor === void 0 ? void 0 : _props$editor.value.selection.isFocused);
|
|
152
|
+
|
|
153
|
+
if (props.showExpandIcon) {
|
|
154
|
+
process.env.NODE_ENV !== "production" ? warning(typeof props.onClickExpand === 'function', 'Editor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
155
|
+
}
|
|
153
156
|
|
|
154
157
|
var _useState = useState(false),
|
|
155
158
|
_useState2 = _slicedToArray(_useState, 2),
|
|
@@ -161,7 +164,9 @@ var Editor = function Editor(props) {
|
|
|
161
164
|
toggleLanguage(props.language);
|
|
162
165
|
}, [toggleLanguage, props.language]);
|
|
163
166
|
var updateRenderToggleButton = useCallback(function () {
|
|
164
|
-
var
|
|
167
|
+
var _ref$current;
|
|
168
|
+
|
|
169
|
+
var doesExceedCollapsedHeightLimit = Number((_ref$current = ref.current) === null || _ref$current === void 0 ? void 0 : _ref$current.clientHeight) > COLLAPSED_HEIGHT;
|
|
165
170
|
|
|
166
171
|
if (doesExceedCollapsedHeightLimit && !renderToggleButton) {
|
|
167
172
|
setRenderToggleButton(true);
|
|
@@ -173,9 +178,9 @@ var Editor = function Editor(props) {
|
|
|
173
178
|
}, [setRenderToggleButton, renderToggleButton]);
|
|
174
179
|
useEffect(function () {
|
|
175
180
|
updateRenderToggleButton();
|
|
176
|
-
}, [props.editor.value.document, updateRenderToggleButton]); // opens the input if it regains focus and it's closed
|
|
181
|
+
}, [(_props$editor2 = props.editor) === null || _props$editor2 === void 0 ? void 0 : _props$editor2.value.document, updateRenderToggleButton]); // opens the input if it regains focus and it's closed
|
|
177
182
|
|
|
178
|
-
if (prevIsFocused !== props.editor.value.selection.isFocused && props.editor.value.selection.isFocused && !props.isOpen) {
|
|
183
|
+
if (prevIsFocused !== ((_props$editor3 = props.editor) === null || _props$editor3 === void 0 ? void 0 : _props$editor3.value.selection.isFocused) && (_props$editor4 = props.editor) !== null && _props$editor4 !== void 0 && _props$editor4.value.selection.isFocused && !props.isOpen) {
|
|
179
184
|
onToggle();
|
|
180
185
|
}
|
|
181
186
|
|
|
@@ -200,6 +205,10 @@ var Editor = function Editor(props) {
|
|
|
200
205
|
toggle = _ref2.toggle,
|
|
201
206
|
containerStyles = _ref2.containerStyles,
|
|
202
207
|
registerContentNode = _ref2.registerContentNode;
|
|
208
|
+
var refObj = {
|
|
209
|
+
containerRef: ref,
|
|
210
|
+
registerContentNode: registerContentNode
|
|
211
|
+
};
|
|
203
212
|
return jsxs(Stack, {
|
|
204
213
|
scale: "xs",
|
|
205
214
|
children: [jsxs(EditorWrapper, {
|
|
@@ -213,18 +222,14 @@ var Editor = function Editor(props) {
|
|
|
213
222
|
children: props.language.toUpperCase()
|
|
214
223
|
})
|
|
215
224
|
}), jsx(RichTextBody, {
|
|
216
|
-
ref:
|
|
217
|
-
containerRef: ref,
|
|
218
|
-
registerContentNode: registerContentNode
|
|
219
|
-
},
|
|
225
|
+
ref: refObj,
|
|
220
226
|
styles: {
|
|
221
227
|
container: _ref$1
|
|
222
228
|
},
|
|
223
|
-
isOpen: props.isOpen,
|
|
224
229
|
hasError: props.hasError,
|
|
225
230
|
isDisabled: props.isDisabled,
|
|
226
231
|
hasWarning: props.hasWarning,
|
|
227
|
-
isReadOnly: props.isReadOnly,
|
|
232
|
+
isReadOnly: Boolean(props.isReadOnly),
|
|
228
233
|
editor: props.editor,
|
|
229
234
|
containerStyles: containerStyles,
|
|
230
235
|
showExpandIcon: props.showExpandIcon,
|
|
@@ -239,7 +244,7 @@ var Editor = function Editor(props) {
|
|
|
239
244
|
// does the trick.
|
|
240
245
|
// TODO: revisit the logic and the implementation to maybe avoid having to apply this style.
|
|
241
246
|
, {
|
|
242
|
-
css: /*#__PURE__*/css("margin-top:", shouldToggleButtonTakeSpace ? 'inherit' : '0px !important', ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AAyJsB","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */"),
|
|
247
|
+
css: /*#__PURE__*/css("margin-top:", shouldToggleButtonTakeSpace ? 'inherit' : '0px !important', ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAwMsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"),
|
|
243
248
|
children: [function () {
|
|
244
249
|
if (props.error) return jsx(LeftColumn, {
|
|
245
250
|
children: jsx("div", {
|
|
@@ -254,7 +259,7 @@ var Editor = function Editor(props) {
|
|
|
254
259
|
return null;
|
|
255
260
|
}(), renderToggleButton && jsx(RightColumn, {
|
|
256
261
|
children: jsx(ToggleButtonWrapper, {
|
|
257
|
-
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/css("position:absolute;top:0;right:0;margin-top:", customProperties.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AAmL2B","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:Editor;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.js"],"names":[],"mappings":"AAiLoB","file":"editor.js","sourcesContent":["import { useRef, useState, useCallback, useEffect, cloneElement } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nconst Editor = (props) => {\n  const intl = useIntl();\n  const ref = useRef();\n  const prevIsFocused = usePrevious(props.editor.value.selection.isFocused);\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      ref.current.clientHeight > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor.value.selection.isFocused &&\n    props.editor.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={{\n                  containerRef: ref,\n                  registerContentNode,\n                }}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                isOpen={props.isOpen}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={props.isReadOnly}\n                editor={props.editor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\n// eslint-disable-next-line react/display-name\nconst renderEditor = (props, editor, next) => {\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={isFocused}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\nEditor.propTypes = {\n  children: PropTypes.node.isRequired,\n  id: PropTypes.string,\n  isOpen: PropTypes.bool.isRequired,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  hasError: PropTypes.bool,\n  editor: PropTypes.any,\n  error: PropTypes.node,\n  warning: PropTypes.node,\n  defaultExpandMultilineText: PropTypes.bool.isRequired,\n  toggleLanguage: PropTypes.func.isRequired,\n  language: PropTypes.string.isRequired,\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n  hasLanguagesControl: PropTypes.bool,\n};\n\nrenderEditor.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  disabled: PropTypes.bool,\n  readOnly: PropTypes.bool,\n  editor: PropTypes.any,\n  options: PropTypes.shape({\n    language: PropTypes.string.isRequired,\n    error: PropTypes.node,\n    warning: PropTypes.node,\n    hasWarning: PropTypes.bool,\n    hasError: PropTypes.bool,\n    defaultExpandMultilineText: PropTypes.bool.isRequired,\n    toggleLanguage: PropTypes.func.isRequired,\n    isOpen: PropTypes.bool.isRequired,\n    showExpandIcon: PropTypes.bool.isRequired,\n    onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n    hasLanguagesControl: PropTypes.bool,\n  }),\n};\n\nexport default renderEditor;\n"]} */"],
|
|
262
|
+
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/css("position:absolute;top:0;right:0;margin-top:", customProperties.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAkO2B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:Editor;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAgOoB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"],
|
|
258
263
|
children: jsx(FlatButton, {
|
|
259
264
|
onClick: toggle,
|
|
260
265
|
label: intl.formatMessage(isOpen ? messagesMultilineInput.collapse : messagesMultilineInput.expand),
|
|
@@ -270,10 +275,34 @@ var Editor = function Editor(props) {
|
|
|
270
275
|
});
|
|
271
276
|
}
|
|
272
277
|
});
|
|
273
|
-
};
|
|
278
|
+
};
|
|
274
279
|
|
|
280
|
+
Editor$1.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
281
|
+
children: _pt.node.isRequired,
|
|
282
|
+
id: _pt.string,
|
|
283
|
+
isOpen: _pt.bool.isRequired,
|
|
284
|
+
isDisabled: _pt.bool,
|
|
285
|
+
isReadOnly: _pt.bool,
|
|
286
|
+
hasWarning: _pt.bool,
|
|
287
|
+
hasError: _pt.bool,
|
|
288
|
+
editor: _pt.any,
|
|
289
|
+
error: _pt.node,
|
|
290
|
+
warning: _pt.node,
|
|
291
|
+
defaultExpandMultilineText: _pt.bool.isRequired,
|
|
292
|
+
toggleLanguage: _pt.func.isRequired,
|
|
293
|
+
language: _pt.string.isRequired,
|
|
294
|
+
showExpandIcon: _pt.bool.isRequired,
|
|
295
|
+
onClickExpand: _pt.func,
|
|
296
|
+
hasLanguagesControl: _pt.bool
|
|
297
|
+
} : {};
|
|
275
298
|
|
|
276
299
|
var renderEditor = function renderEditor(props, editor, next) {
|
|
300
|
+
var _props$editor5;
|
|
301
|
+
|
|
302
|
+
if (props.options.showExpandIcon) {
|
|
303
|
+
process.env.NODE_ENV !== "production" ? warning(typeof props.options.onClickExpand === 'function', 'renderEditor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
304
|
+
}
|
|
305
|
+
|
|
277
306
|
var internalId = "".concat(props.id, "__internal__id");
|
|
278
307
|
var children = /*#__PURE__*/cloneElement(next(), {
|
|
279
308
|
id: internalId
|
|
@@ -285,12 +314,12 @@ var renderEditor = function renderEditor(props, editor, next) {
|
|
|
285
314
|
isReadOnly: props.readOnly
|
|
286
315
|
}, pick(props.options, ['defaultExpandMultilineText', 'language', 'warning', 'error', 'hasWarning', 'hasError', 'toggleLanguage', 'isOpen', 'showExpandIcon', 'onClickExpand', 'hasLanguagesControl'])), filterDataAttributes(props));
|
|
287
316
|
|
|
288
|
-
var isFocused = props.editor.value.selection.isFocused;
|
|
289
|
-
return jsxs(Editor, _objectSpread$3(_objectSpread$3({
|
|
317
|
+
var isFocused = (_props$editor5 = props.editor) === null || _props$editor5 === void 0 ? void 0 : _props$editor5.value.selection.isFocused;
|
|
318
|
+
return jsxs(Editor$1, _objectSpread$3(_objectSpread$3({
|
|
290
319
|
editor: editor
|
|
291
320
|
}, passedProps), {}, {
|
|
292
321
|
children: [children, jsx(HiddenInput, {
|
|
293
|
-
isFocused: isFocused,
|
|
322
|
+
isFocused: Boolean(isFocused),
|
|
294
323
|
handleFocus: editor.focus,
|
|
295
324
|
disabled: props.disabled,
|
|
296
325
|
readOnly: props.readOnly,
|
|
@@ -299,58 +328,35 @@ var renderEditor = function renderEditor(props, editor, next) {
|
|
|
299
328
|
}));
|
|
300
329
|
};
|
|
301
330
|
|
|
302
|
-
Editor.displayName = 'Editor';
|
|
303
|
-
Editor.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
304
|
-
children: PropTypes.node.isRequired,
|
|
305
|
-
id: PropTypes.string,
|
|
306
|
-
isOpen: PropTypes.bool.isRequired,
|
|
307
|
-
isDisabled: PropTypes.bool,
|
|
308
|
-
isReadOnly: PropTypes.bool,
|
|
309
|
-
hasWarning: PropTypes.bool,
|
|
310
|
-
hasError: PropTypes.bool,
|
|
311
|
-
editor: PropTypes.any,
|
|
312
|
-
error: PropTypes.node,
|
|
313
|
-
warning: PropTypes.node,
|
|
314
|
-
defaultExpandMultilineText: PropTypes.bool.isRequired,
|
|
315
|
-
toggleLanguage: PropTypes.func.isRequired,
|
|
316
|
-
language: PropTypes.string.isRequired,
|
|
317
|
-
showExpandIcon: PropTypes.bool.isRequired,
|
|
318
|
-
onClickExpand: requiredIf(PropTypes.func, function (props) {
|
|
319
|
-
return props.showExpandIcon;
|
|
320
|
-
}),
|
|
321
|
-
hasLanguagesControl: PropTypes.bool
|
|
322
|
-
} : {};
|
|
323
|
-
renderEditor.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
324
|
-
id: PropTypes.string,
|
|
325
|
-
name: PropTypes.string,
|
|
326
|
-
disabled: PropTypes.bool,
|
|
327
|
-
readOnly: PropTypes.bool,
|
|
328
|
-
editor: PropTypes.any,
|
|
329
|
-
options: PropTypes.shape({
|
|
330
|
-
language: PropTypes.string.isRequired,
|
|
331
|
-
error: PropTypes.node,
|
|
332
|
-
warning: PropTypes.node,
|
|
333
|
-
hasWarning: PropTypes.bool,
|
|
334
|
-
hasError: PropTypes.bool,
|
|
335
|
-
defaultExpandMultilineText: PropTypes.bool.isRequired,
|
|
336
|
-
toggleLanguage: PropTypes.func.isRequired,
|
|
337
|
-
isOpen: PropTypes.bool.isRequired,
|
|
338
|
-
showExpandIcon: PropTypes.bool.isRequired,
|
|
339
|
-
onClickExpand: requiredIf(PropTypes.func, function (props) {
|
|
340
|
-
return props.showExpandIcon;
|
|
341
|
-
}),
|
|
342
|
-
hasLanguagesControl: PropTypes.bool
|
|
343
|
-
})
|
|
344
|
-
} : {};
|
|
331
|
+
Editor$1.displayName = 'Editor';
|
|
345
332
|
var renderEditor$1 = renderEditor;
|
|
346
333
|
|
|
334
|
+
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = _Reflect$construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
|
|
335
|
+
|
|
336
|
+
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !_Reflect$construct) return false; if (_Reflect$construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(_Reflect$construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
|
|
337
|
+
|
|
347
338
|
function ownKeys$2(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
348
339
|
|
|
349
340
|
function _objectSpread$2(target) { for (var i = 1; i < arguments.length; i++) { var _context2, _context3; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty(_context2 = ownKeys$2(Object(source), !0)).call(_context2, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(target, _Object$getOwnPropertyDescriptors(source)) : _forEachInstanceProperty(_context3 = ownKeys$2(Object(source))).call(_context3, function (key) { _Object$defineProperty(target, key, _Object$getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
350
341
|
|
|
351
|
-
|
|
342
|
+
// This is a temporary wrapper component helping to mitigate typing issues of `slate-react@0.22.10` package.
|
|
343
|
+
// TODO: remove after upgrade of `slate-react` to the latest version
|
|
344
|
+
var Editor = function Editor(props) {
|
|
345
|
+
return jsx(Editor$2, _objectSpread$2({}, props));
|
|
346
|
+
};
|
|
352
347
|
|
|
353
|
-
|
|
348
|
+
Editor.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
349
|
+
id: _pt.string,
|
|
350
|
+
name: _pt.string,
|
|
351
|
+
onFocus: _pt.func,
|
|
352
|
+
onBlur: _pt.func,
|
|
353
|
+
disabled: _pt.bool,
|
|
354
|
+
readOnly: _pt.bool,
|
|
355
|
+
value: _pt.any.isRequired,
|
|
356
|
+
onChange: _pt.func,
|
|
357
|
+
plugins: _pt.any.isRequired,
|
|
358
|
+
renderEditor: _pt.any.isRequired
|
|
359
|
+
} : {};
|
|
354
360
|
|
|
355
361
|
var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
356
362
|
_inherits(RichTextInput, _PureComponent);
|
|
@@ -386,6 +392,8 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
386
392
|
// onChange unless this value changes.
|
|
387
393
|
|
|
388
394
|
if (hasSerializedValueChanged) {
|
|
395
|
+
var _this$props$onChange, _this$props;
|
|
396
|
+
|
|
389
397
|
var fakeEvent = {
|
|
390
398
|
target: {
|
|
391
399
|
id: _this.props.id,
|
|
@@ -394,8 +402,7 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
394
402
|
value: serializedValue
|
|
395
403
|
}
|
|
396
404
|
};
|
|
397
|
-
|
|
398
|
-
_this.props.onChange(fakeEvent);
|
|
405
|
+
(_this$props$onChange = (_this$props = _this.props).onChange) === null || _this$props$onChange === void 0 ? void 0 : _this$props$onChange.call(_this$props, fakeEvent);
|
|
399
406
|
}
|
|
400
407
|
|
|
401
408
|
if (hasInternalSlateValueChanged && !hasSerializedValueChanged) {
|
|
@@ -404,7 +411,7 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
404
411
|
}
|
|
405
412
|
};
|
|
406
413
|
|
|
407
|
-
_this.onBlur = function (
|
|
414
|
+
_this.onBlur = function (_event, _editor, next) {
|
|
408
415
|
next();
|
|
409
416
|
|
|
410
417
|
if (_this.props.onBlur) {
|
|
@@ -416,12 +423,14 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
416
423
|
};
|
|
417
424
|
|
|
418
425
|
_setTimeout(function () {
|
|
419
|
-
|
|
426
|
+
var _this$props$onBlur, _this$props2;
|
|
427
|
+
|
|
428
|
+
return (_this$props$onBlur = (_this$props2 = _this.props).onBlur) === null || _this$props$onBlur === void 0 ? void 0 : _this$props$onBlur.call(_this$props2, fakeEvent);
|
|
420
429
|
}, 0);
|
|
421
430
|
}
|
|
422
431
|
};
|
|
423
432
|
|
|
424
|
-
_this.onFocus = function (
|
|
433
|
+
_this.onFocus = function (_event, _editor, next) {
|
|
425
434
|
next();
|
|
426
435
|
|
|
427
436
|
if (_this.props.onFocus) {
|
|
@@ -433,7 +442,9 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
433
442
|
};
|
|
434
443
|
|
|
435
444
|
_setTimeout(function () {
|
|
436
|
-
|
|
445
|
+
var _this$props$onFocus, _this$props3;
|
|
446
|
+
|
|
447
|
+
return (_this$props$onFocus = (_this$props3 = _this.props).onFocus) === null || _this$props$onFocus === void 0 ? void 0 : _this$props$onFocus.call(_this$props3, fakeEvent);
|
|
437
448
|
}, 0);
|
|
438
449
|
}
|
|
439
450
|
};
|
|
@@ -453,7 +464,15 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
453
464
|
}, {
|
|
454
465
|
key: "render",
|
|
455
466
|
value: function render() {
|
|
456
|
-
|
|
467
|
+
if (!this.props.isReadOnly) {
|
|
468
|
+
process.env.NODE_ENV !== "production" ? warning(typeof this.props.onChange === 'function', 'RichTextInput: `onChange` is required when field is not read only.') : void 0;
|
|
469
|
+
}
|
|
470
|
+
|
|
471
|
+
if (this.props.showExpandIcon) {
|
|
472
|
+
process.env.NODE_ENV !== "production" ? warning(typeof this.props.onClickExpand === 'function', 'RichTextInput: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
return jsx(Editor, _objectSpread$2(_objectSpread$2({}, filterDataAttributes(this.props)), {}, {
|
|
457
476
|
id: this.props.id,
|
|
458
477
|
name: this.props.name,
|
|
459
478
|
disabled: this.props.isDisabled,
|
|
@@ -464,9 +483,7 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
464
483
|
// warning in the console,
|
|
465
484
|
// so instead we pass our extra this.props through this `options` prop.
|
|
466
485
|
,
|
|
467
|
-
options:
|
|
468
|
-
hasLanguagesControl: this.props.hasLanguagesControl
|
|
469
|
-
}, pick(this.props, ['language', 'onToggle', 'toggleLanguage', 'isOpen', 'warning', 'error', 'defaultExpandMultilineText', 'hasWarning', 'hasError', 'placeholder', 'onClickExpand', 'showExpandIcon'])),
|
|
486
|
+
options: pick(this.props, ['language', 'onToggle', 'toggleLanguage', 'isOpen', 'warning', 'error', 'defaultExpandMultilineText', 'hasWarning', 'hasError', 'placeholder', 'onClickExpand', 'showExpandIcon', 'hasLanguagesControl']),
|
|
470
487
|
onChange: this.onValueChange,
|
|
471
488
|
plugins: richTextPlugins,
|
|
472
489
|
renderEditor: renderEditor$1
|
|
@@ -483,25 +500,26 @@ RichTextInput.defaultProps = {
|
|
|
483
500
|
};
|
|
484
501
|
RichTextInput.displayName = 'RichTextInput';
|
|
485
502
|
RichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
486
|
-
defaultExpandMultilineText:
|
|
487
|
-
hasError:
|
|
488
|
-
hasWarning:
|
|
489
|
-
id:
|
|
490
|
-
name:
|
|
491
|
-
placeholder:
|
|
492
|
-
isDisabled:
|
|
493
|
-
isReadOnly:
|
|
494
|
-
onChange:
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
503
|
+
defaultExpandMultilineText: _pt.bool,
|
|
504
|
+
hasError: _pt.bool,
|
|
505
|
+
hasWarning: _pt.bool,
|
|
506
|
+
id: _pt.string,
|
|
507
|
+
name: _pt.string,
|
|
508
|
+
placeholder: _pt.string,
|
|
509
|
+
isDisabled: _pt.bool,
|
|
510
|
+
isReadOnly: _pt.bool,
|
|
511
|
+
onChange: _pt.func,
|
|
512
|
+
onBlur: _pt.func,
|
|
513
|
+
onFocus: _pt.func,
|
|
514
|
+
value: _pt.string.isRequired,
|
|
515
|
+
showExpandIcon: _pt.bool.isRequired,
|
|
516
|
+
onClickExpand: _pt.func,
|
|
517
|
+
hasLanguagesControl: _pt.bool,
|
|
518
|
+
language: _pt.string.isRequired,
|
|
519
|
+
isOpen: _pt.bool.isRequired,
|
|
520
|
+
toggleLanguage: _pt.func.isRequired,
|
|
521
|
+
warning: _pt.node,
|
|
522
|
+
error: _pt.string
|
|
505
523
|
} : {};
|
|
506
524
|
var RichTextInput$1 = RichTextInput;
|
|
507
525
|
|
|
@@ -518,13 +536,15 @@ var RequiredValueErrorMessage = function RequiredValueErrorMessage() {
|
|
|
518
536
|
RequiredValueErrorMessage.displayName = 'RequiredValueErrorMessage';
|
|
519
537
|
var RequiredValueErrorMessage$1 = RequiredValueErrorMessage;
|
|
520
538
|
|
|
521
|
-
var _templateObject;
|
|
522
|
-
|
|
523
539
|
function ownKeys(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
524
540
|
|
|
525
|
-
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var
|
|
541
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var _context3, _context4; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty(_context3 = ownKeys(Object(source), !0)).call(_context3, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(target, _Object$getOwnPropertyDescriptors(source)) : _forEachInstanceProperty(_context4 = ownKeys(Object(source))).call(_context4, function (key) { _Object$defineProperty(target, key, _Object$getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
526
542
|
|
|
527
543
|
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)."; }
|
|
544
|
+
var defaultProps = {
|
|
545
|
+
horizontalConstraint: 'scale',
|
|
546
|
+
showExpandIcon: false
|
|
547
|
+
};
|
|
528
548
|
|
|
529
549
|
var expandedTranslationsReducer = function expandedTranslationsReducer(state, action) {
|
|
530
550
|
switch (action.type) {
|
|
@@ -559,13 +579,25 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
559
579
|
} : {
|
|
560
580
|
name: "9a4cy7-LocalizedRichTextInput",
|
|
561
581
|
styles: "align-self:flex-start;label:LocalizedRichTextInput;",
|
|
562
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-rich-text-input.js"],"names":[],"mappings":"AA8JoB","file":"localized-rich-text-input.js","sourcesContent":["import { useReducer, useCallback } from 'react';\nimport PropTypes from 'prop-types';\nimport requiredIf from 'react-required-if';\nimport { oneLine } from 'common-tags';\nimport { css } from '@emotion/react';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  isTouched,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport { localized } from '@commercetools-uikit/rich-text-utils';\nimport RichTextInput from './rich-text-input';\nimport RequiredValueErrorMessage from './required-value-error-message';\n\nconst expandedTranslationsReducer = (state, action) => {\n  switch (action.type) {\n    case 'toggle':\n      return {\n        ...state,\n        [action.payload]: !state[action.payload],\n      };\n\n    case 'toggleAll': {\n      const newState = Object.keys(state).reduce((translations, locale) => {\n        return {\n          [locale]: true,\n          ...translations,\n        };\n      }, {});\n      return newState;\n    }\n    default:\n      return state;\n  }\n};\n\n// This component supports expanding/collapsing rich text inputs, but it also\n// supports showing/hiding the remaining languages.\n// These two features are both about opening/closing something, and so the code\n// can get quite confusing. We try to stick to expand/collapse for the\n// multiline inputs, while we use show/hide/open/close for the remaining\n// languages.\nconst LocalizedRichTextInput = (props) => {\n  const initialExpandedTranslationsState = Object.keys(props.value).reduce(\n    (translations, locale) => {\n      return {\n        [locale]: Boolean(props.defaultExpandMultilineText),\n        ...translations,\n      };\n    },\n    {}\n  );\n\n  const [expandedTranslationsState, expandedTranslationsDispatch] = useReducer(\n    expandedTranslationsReducer,\n    initialExpandedTranslationsState\n  );\n\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages ||\n    // useToggleState's default is `true`, but we want `false`\n    false;\n\n  const [areLanguagesOpened, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const toggleLanguage = useCallback(\n    (language) => {\n      expandedTranslationsDispatch({ type: 'toggle', payload: language });\n    },\n    [expandedTranslationsDispatch]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  const hasWarningInRemainingLanguages =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {\n    if (!areLanguagesOpened) {\n      // this update within render replaces the old `getDerivedStateFromProps` functionality\n      // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesControl =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesOpened) return null;\n            const isLastLanguage = index === languages.length - 1;\n\n            const hasLanguagesControl =\n              (isFirstLanguage && !areLanguagesOpened) || isLastLanguage;\n\n            return (\n              <RichTextInput\n                key={language}\n                autoComplete={props.autoComplete}\n                id={LocalizedRichTextInput.getId(props.id, language)}\n                name={LocalizedRichTextInput.getName(props.name, language)}\n                value={props.value[language]}\n                onChange={props.onChange}\n                language={language}\n                isOpen={expandedTranslationsState[language]}\n                toggleLanguage={toggleLanguage}\n                placeholder={\n                  props.placeholder ? props.placeholder[language] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isAutofocussed={index === 0 && props.isAutofocussed}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                toggleLanguages={toggleLanguages}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[language])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[language])\n                )}\n                warning={props.warnings && props.warnings[language]}\n                error={props.errors && props.errors[language]}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n                hasLanguagesControl={hasLanguagesControl}\n                {...createLocalizedDataAttributes(props, language)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesControl && (\n          <div\n            css={css`\n              align-self: flex-start;\n            `}\n          >\n            <LocalizedInputToggle\n              isOpen={areLanguagesOpened}\n              onClick={toggleLanguages}\n              isDisabled={\n                areLanguagesOpened &&\n                Boolean(\n                  hasErrorInRemainingLanguages || hasWarningInRemainingLanguages\n                )\n              }\n              remainingLocalizations={languages.length - 1}\n            />\n          </div>\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedRichTextInput.displayName = 'LocalizedRichTextInput';\n\nLocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedRichTextInput.propTypes = {\n  id: PropTypes.string,\n  name: PropTypes.string,\n  autoComplete: PropTypes.string,\n  // then input doesn't accept a \"languages\" prop, instead all possible\n  // languages have to exist (with empty or filled slate values) on the value:\n  //   { en: slateValue, de: slateValue, es: slateValue }\n  value: PropTypes.objectOf(PropTypes.any).isRequired,\n  onChange: requiredIf(PropTypes.func, (props) => !props.isReadOnly),\n  selectedLanguage: PropTypes.string.isRequired,\n  onBlur: PropTypes.func,\n  onFocus: PropTypes.func,\n  defaultExpandMultilineText: PropTypes.bool,\n  hideLanguageExpansionControls: PropTypes.bool,\n  defaultExpandLanguages: (props, propName, componentName, ...rest) => {\n    if (\n      props.hideLanguageExpansionControls &&\n      typeof props[propName] === 'boolean'\n    ) {\n      throw new Error(\n        oneLine`\n          ${componentName}: \"${propName}\" does not have any effect when\n          \"hideLanguageExpansionControls\" is set.\n        `\n      );\n    }\n    return PropTypes.bool(props, propName, componentName, ...rest);\n  },\n  isAutofocussed: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  isReadOnly: PropTypes.bool,\n  placeholder: PropTypes.objectOf(PropTypes.string),\n  horizontalConstraint: PropTypes.oneOf([\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  hasError: PropTypes.bool,\n  hasWarning: PropTypes.bool,\n  errors: PropTypes.objectOf(PropTypes.node),\n  warnings: PropTypes.objectOf(PropTypes.node),\n  showExpandIcon: PropTypes.bool.isRequired,\n  onClickExpand: requiredIf(PropTypes.func, (props) => props.showExpandIcon),\n};\n\nLocalizedRichTextInput.getId = getId;\n\nLocalizedRichTextInput.getName = getName;\n\nLocalizedRichTextInput.defaultProps = {\n  horizontalConstraint: 'scale',\n  showExpandIcon: false,\n};\n\nLocalizedRichTextInput.createLocalizedString = localized.createLocalizedString;\n\nLocalizedRichTextInput.isEmpty = localized.isEmpty;\n\nLocalizedRichTextInput.omitEmptyTranslations = localized.omitEmptyTranslations;\n\nLocalizedRichTextInput.isTouched = isTouched;\n\nexport default LocalizedRichTextInput;\n"]} */",
|
|
582
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-rich-text-input.tsx"],"names":[],"mappings":"AAqToB","file":"localized-rich-text-input.tsx","sourcesContent":["import {\n  useReducer,\n  useCallback,\n  type ReactNode,\n  type MouseEvent,\n  type KeyboardEvent,\n} from 'react';\nimport { css } from '@emotion/react';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  isTouched,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport { localized } from '@commercetools-uikit/rich-text-utils';\nimport { warning } from '@commercetools-uikit/utils';\nimport RichTextInput from './rich-text-input';\nimport RequiredValueErrorMessage from './required-value-error-message';\n\ntype TErrors = Record<string, string>;\ntype TWarnings = Record<string, ReactNode>;\n\ntype TEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    language?: string;\n    value?: string;\n  };\n};\n\ntype TLocalizedRichTextInputProps = {\n  /**\n   * Used as prefix of HTML `id` property. Each input field id will have the language as a suffix (`${idPrefix}.${lang}`), e.g. `foo.en\n   */\n  id?: string;\n  /**\n   * Used as HTML `name` property for each input field. Each input field name will have the language as a suffix (`${namePrefix}.${lang}`), e.g. `foo.en`\n   */\n  name?: string;\n  // then input doesn't accept a \"languages\" prop, instead all possible\n  // languages have to exist (with empty or filled slate values) on the value:\n  //   { en: slateValue, de: slateValue, es: slateValue }\n  /**\n   * Values to use. Keyed by language, the values are the actual values, e.g. `{ en: '<p>Horse</p>', de: '<p>Pferd</p>' }\n   */\n  value: Record<string, string>;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: (event: TEvent) => void;\n  /**\n   * Specifies which language will be shown in case the `LocalizedRichTextInput` is collapsed.\n   */\n  selectedLanguage: string;\n  /**\n   *Called when any field is blurred. Is called with the `event` of that field.\n   */\n  onBlur?: (event: TEvent) => void;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: (event: TEvent) => void;\n  /**\n   * Expands input components holding multiline values instead of collapsing them by default.\n   */\n  defaultExpandMultilineText?: boolean;\n  /**\n   * Will hide the language expansion controls when set to `true`. All languages will be shown when set to `true`\n   */\n  hideLanguageExpansionControls?: boolean;\n  /**\n   * Controls whether one or all languages are visible by default. Pass `true` to show all languages by default.\n   */\n  defaultExpandLanguages?: boolean;\n  /**\n   * Disables all input\n   */\n  isDisabled?: boolean;\n  /**\n   * Disables all input fields and shows them in read-only mode.\n   */\n  isReadOnly?: boolean;\n  /**\n   * Placeholders for each language. Object of the same shape as\n   */\n  placeholder?: Record<string, string>;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\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   * Will apply the error state to each input without showing any error message.\n   */\n  hasError?: boolean;\n  /**\n   * Will apply the warning state to each input without showing any warning message.\n   */\n  hasWarning?: boolean;\n  /**\n   * Used to show errors underneath the inputs of specific languages. Pass an object whose key is a language and whose value is the error to show for that key.\n   */\n  errors?: TErrors;\n  /**\n   * Used to show warnings underneath the inputs of specific languages. Pass an object whose key is a language and whose value is the warning to show for that key.\n   */\n  warnings?: TWarnings;\n  /**\n   * Shows an `expand` icon in the toolbar\n   */\n  showExpandIcon: boolean;\n  /**\n   * Called when the `expand` button is clicked\n   */\n  onClickExpand?: () => boolean;\n};\n\ntype TReducerState = {\n  [id: string]: boolean;\n};\n\ntype TReducerAction =\n  | { type: 'toggle'; payload: string }\n  | { type: 'toggleAll'; payload: string };\n\nconst defaultProps: Pick<\n  TLocalizedRichTextInputProps,\n  'horizontalConstraint' | 'showExpandIcon'\n> = {\n  horizontalConstraint: 'scale',\n  showExpandIcon: false,\n};\n\nconst expandedTranslationsReducer = (\n  state: TReducerState,\n  action: TReducerAction\n) => {\n  switch (action.type) {\n    case 'toggle':\n      return {\n        ...state,\n        [action.payload]: !state[action.payload],\n      };\n\n    case 'toggleAll': {\n      const newState = Object.keys(state).reduce((translations, locale) => {\n        return {\n          [locale]: true,\n          ...translations,\n        };\n      }, {});\n      return newState;\n    }\n    default:\n      return state;\n  }\n};\n\n// This component supports expanding/collapsing rich text inputs, but it also\n// supports showing/hiding the remaining languages.\n// These two features are both about opening/closing something, and so the code\n// can get quite confusing. We try to stick to expand/collapse for the\n// multiline inputs, while we use show/hide/open/close for the remaining\n// languages.\nconst LocalizedRichTextInput = (props: TLocalizedRichTextInputProps) => {\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedRichTextInput: `onChange` is required when input is not read only.'\n    );\n  }\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'LocalizedRichTextInput: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  if (props.hideLanguageExpansionControls) {\n    warning(\n      typeof props.defaultExpandLanguages !== 'boolean',\n      'LocalizedRichTextInput: \"defaultExpandLanguages\" does not have any effect when \"hideLanguageExpansionControls\" is set.'\n    );\n  }\n\n  const initialExpandedTranslationsState = Object.keys(props.value).reduce(\n    (translations, locale) => {\n      return {\n        [locale]: Boolean(props.defaultExpandMultilineText),\n        ...translations,\n      };\n    },\n    {}\n  );\n\n  const [expandedTranslationsState, expandedTranslationsDispatch] = useReducer(\n    expandedTranslationsReducer,\n    initialExpandedTranslationsState\n  );\n\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages ||\n    // useToggleState's default is `true`, but we want `false`\n    false;\n\n  const [areLanguagesOpened, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const toggleLanguage = useCallback(\n    (language) => {\n      expandedTranslationsDispatch({ type: 'toggle', payload: language });\n    },\n    [expandedTranslationsDispatch]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  const hasWarningInRemainingLanguages =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {\n    if (!areLanguagesOpened) {\n      // this update within render replaces the old `getDerivedStateFromProps` functionality\n      // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesControl =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesOpened) return null;\n            const isLastLanguage = index === languages.length - 1;\n\n            const hasLanguagesControl =\n              (isFirstLanguage && !areLanguagesOpened) || isLastLanguage;\n\n            return (\n              <RichTextInput\n                key={language}\n                id={LocalizedRichTextInput.getId(props.id, language)}\n                name={LocalizedRichTextInput.getName(props.name, language)}\n                value={props.value[language]}\n                onChange={props.onChange}\n                language={language}\n                isOpen={expandedTranslationsState[language]}\n                toggleLanguage={toggleLanguage}\n                placeholder={\n                  props.placeholder ? props.placeholder[language] : undefined\n                }\n                onBlur={props.onBlur}\n                onFocus={props.onFocus}\n                isDisabled={props.isDisabled}\n                isReadOnly={props.isReadOnly}\n                hasError={Boolean(\n                  props.hasError || (props.errors && props.errors[language])\n                )}\n                hasWarning={Boolean(\n                  props.hasWarning ||\n                    (props.warnings && props.warnings[language])\n                )}\n                warning={props.warnings && props.warnings[language]}\n                error={props.errors && props.errors[language]}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n                hasLanguagesControl={hasLanguagesControl}\n                {...createLocalizedDataAttributes(props, language)}\n              />\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesControl && (\n          <div\n            css={css`\n              align-self: flex-start;\n            `}\n          >\n            <LocalizedInputToggle\n              isOpen={areLanguagesOpened}\n              onClick={\n                toggleLanguages as (\n                  event:\n                    | MouseEvent<HTMLButtonElement>\n                    | KeyboardEvent<HTMLButtonElement>\n                    | boolean\n                ) => void\n              }\n              isDisabled={\n                areLanguagesOpened &&\n                Boolean(\n                  hasErrorInRemainingLanguages || hasWarningInRemainingLanguages\n                )\n              }\n              remainingLocalizations={languages.length - 1}\n            />\n          </div>\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedRichTextInput.displayName = 'LocalizedRichTextInput';\n\nLocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedRichTextInput.getId = getId;\n\nLocalizedRichTextInput.getName = getName;\n\nLocalizedRichTextInput.defaultProps = defaultProps;\n\nLocalizedRichTextInput.createLocalizedString = localized.createLocalizedString;\n\nLocalizedRichTextInput.isEmpty = localized.isEmpty;\n\nLocalizedRichTextInput.omitEmptyTranslations = localized.omitEmptyTranslations;\n\nLocalizedRichTextInput.isTouched = isTouched;\n\nexport default LocalizedRichTextInput;\n"]} */",
|
|
563
583
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
564
584
|
};
|
|
565
585
|
|
|
566
586
|
var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
567
587
|
var _context2;
|
|
568
588
|
|
|
589
|
+
if (!props.isReadOnly) {
|
|
590
|
+
process.env.NODE_ENV !== "production" ? warning(typeof props.onChange === 'function', 'LocalizedRichTextInput: `onChange` is required when input is not read only.') : void 0;
|
|
591
|
+
}
|
|
592
|
+
|
|
593
|
+
if (props.showExpandIcon) {
|
|
594
|
+
process.env.NODE_ENV !== "production" ? warning(typeof props.onClickExpand === 'function', 'LocalizedRichTextInput: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
595
|
+
}
|
|
596
|
+
|
|
597
|
+
if (props.hideLanguageExpansionControls) {
|
|
598
|
+
process.env.NODE_ENV !== "production" ? warning(typeof props.defaultExpandLanguages !== 'boolean', 'LocalizedRichTextInput: "defaultExpandLanguages" does not have any effect when "hideLanguageExpansionControls" is set.') : void 0;
|
|
599
|
+
}
|
|
600
|
+
|
|
569
601
|
var initialExpandedTranslationsState = _reduceInstanceProperty(_context2 = _Object$keys(props.value)).call(_context2, function (translations, locale) {
|
|
570
602
|
return _objectSpread(_defineProperty({}, locale, Boolean(props.defaultExpandMultilineText)), translations);
|
|
571
603
|
}, {});
|
|
@@ -613,7 +645,6 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
613
645
|
var isLastLanguage = index === languages.length - 1;
|
|
614
646
|
var hasLanguagesControl = isFirstLanguage && !areLanguagesOpened || isLastLanguage;
|
|
615
647
|
return jsx(RichTextInput$1, _objectSpread({
|
|
616
|
-
autoComplete: props.autoComplete,
|
|
617
648
|
id: LocalizedRichTextInput.getId(props.id, language),
|
|
618
649
|
name: LocalizedRichTextInput.getName(props.name, language),
|
|
619
650
|
value: props.value[language],
|
|
@@ -624,10 +655,8 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
624
655
|
placeholder: props.placeholder ? props.placeholder[language] : undefined,
|
|
625
656
|
onBlur: props.onBlur,
|
|
626
657
|
onFocus: props.onFocus,
|
|
627
|
-
isAutofocussed: index === 0 && props.isAutofocussed,
|
|
628
658
|
isDisabled: props.isDisabled,
|
|
629
659
|
isReadOnly: props.isReadOnly,
|
|
630
|
-
toggleLanguages: toggleLanguages,
|
|
631
660
|
hasError: Boolean(props.hasError || props.errors && props.errors[language]),
|
|
632
661
|
hasWarning: Boolean(props.hasWarning || props.warnings && props.warnings[language]),
|
|
633
662
|
warning: props.warnings && props.warnings[language],
|
|
@@ -650,57 +679,33 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
650
679
|
});
|
|
651
680
|
};
|
|
652
681
|
|
|
653
|
-
LocalizedRichTextInput.displayName = 'LocalizedRichTextInput';
|
|
654
|
-
LocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage$1;
|
|
655
682
|
LocalizedRichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
656
|
-
id:
|
|
657
|
-
name:
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
670
|
-
|
|
671
|
-
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
}
|
|
677
|
-
|
|
678
|
-
for (var _len = arguments.length, rest = new Array(_len > 3 ? _len - 3 : 0), _key = 3; _key < _len; _key++) {
|
|
679
|
-
rest[_key - 3] = arguments[_key];
|
|
680
|
-
}
|
|
681
|
-
|
|
682
|
-
return PropTypes.bool.apply(PropTypes, _concatInstanceProperty(_context3 = [props, propName, componentName]).call(_context3, rest));
|
|
683
|
-
},
|
|
684
|
-
isAutofocussed: PropTypes.bool,
|
|
685
|
-
isDisabled: PropTypes.bool,
|
|
686
|
-
isReadOnly: PropTypes.bool,
|
|
687
|
-
placeholder: PropTypes.objectOf(PropTypes.string),
|
|
688
|
-
horizontalConstraint: PropTypes.oneOf([7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
689
|
-
hasError: PropTypes.bool,
|
|
690
|
-
hasWarning: PropTypes.bool,
|
|
691
|
-
errors: PropTypes.objectOf(PropTypes.node),
|
|
692
|
-
warnings: PropTypes.objectOf(PropTypes.node),
|
|
693
|
-
showExpandIcon: PropTypes.bool.isRequired,
|
|
694
|
-
onClickExpand: requiredIf(PropTypes.func, function (props) {
|
|
695
|
-
return props.showExpandIcon;
|
|
696
|
-
})
|
|
683
|
+
id: _pt.string,
|
|
684
|
+
name: _pt.string,
|
|
685
|
+
value: _pt.objectOf(_pt.string).isRequired,
|
|
686
|
+
onChange: _pt.func,
|
|
687
|
+
selectedLanguage: _pt.string.isRequired,
|
|
688
|
+
onBlur: _pt.func,
|
|
689
|
+
onFocus: _pt.func,
|
|
690
|
+
defaultExpandMultilineText: _pt.bool,
|
|
691
|
+
hideLanguageExpansionControls: _pt.bool,
|
|
692
|
+
defaultExpandLanguages: _pt.bool,
|
|
693
|
+
isDisabled: _pt.bool,
|
|
694
|
+
isReadOnly: _pt.bool,
|
|
695
|
+
placeholder: _pt.objectOf(_pt.string),
|
|
696
|
+
horizontalConstraint: _pt.oneOf([7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
697
|
+
hasError: _pt.bool,
|
|
698
|
+
hasWarning: _pt.bool,
|
|
699
|
+
errors: _pt.objectOf(_pt.string),
|
|
700
|
+
warnings: _pt.objectOf(_pt.node),
|
|
701
|
+
showExpandIcon: _pt.bool.isRequired,
|
|
702
|
+
onClickExpand: _pt.func
|
|
697
703
|
} : {};
|
|
704
|
+
LocalizedRichTextInput.displayName = 'LocalizedRichTextInput';
|
|
705
|
+
LocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage$1;
|
|
698
706
|
LocalizedRichTextInput.getId = getId;
|
|
699
707
|
LocalizedRichTextInput.getName = getName;
|
|
700
|
-
LocalizedRichTextInput.defaultProps =
|
|
701
|
-
horizontalConstraint: 'scale',
|
|
702
|
-
showExpandIcon: false
|
|
703
|
-
};
|
|
708
|
+
LocalizedRichTextInput.defaultProps = defaultProps;
|
|
704
709
|
LocalizedRichTextInput.createLocalizedString = localized.createLocalizedString;
|
|
705
710
|
LocalizedRichTextInput.isEmpty = localized.isEmpty;
|
|
706
711
|
LocalizedRichTextInput.omitEmptyTranslations = localized.omitEmptyTranslations;
|
|
@@ -708,6 +713,6 @@ LocalizedRichTextInput.isTouched = isTouched;
|
|
|
708
713
|
var LocalizedRichTextInput$1 = LocalizedRichTextInput;
|
|
709
714
|
|
|
710
715
|
// NOTE: This string will be replaced on build time with the package version.
|
|
711
|
-
var version = "13.0.
|
|
716
|
+
var version = "13.0.4";
|
|
712
717
|
|
|
713
718
|
export { LocalizedRichTextInput$1 as default, version };
|