@commercetools-uikit/localized-rich-text-input 14.0.2 → 15.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.dev.js +187 -237
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.prod.js +178 -166
- package/dist/commercetools-uikit-localized-rich-text-input.esm.js +189 -239
- package/dist/declarations/src/editor.d.ts +13 -32
- package/dist/declarations/src/localized-rich-text-input.d.ts +16 -24
- package/dist/declarations/src/rich-text-input.d.ts +18 -56
- package/package.json +19 -19
- package/dist/declarations/src/editor.types.d.ts +0 -103
|
@@ -11,7 +11,6 @@ var _Object$defineProperties = require('@babel/runtime-corejs3/core-js-stable/ob
|
|
|
11
11
|
var _Object$defineProperty = require('@babel/runtime-corejs3/core-js-stable/object/define-property');
|
|
12
12
|
var _slicedToArray = require('@babel/runtime-corejs3/helpers/slicedToArray');
|
|
13
13
|
var _defineProperty = require('@babel/runtime-corejs3/helpers/defineProperty');
|
|
14
|
-
var _pt = require('prop-types');
|
|
15
14
|
var _reduceInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/reduce');
|
|
16
15
|
var _Object$keys = require('@babel/runtime-corejs3/core-js-stable/object/keys');
|
|
17
16
|
var _mapInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/map');
|
|
@@ -30,10 +29,8 @@ var _createClass = require('@babel/runtime-corejs3/helpers/createClass');
|
|
|
30
29
|
var _inherits = require('@babel/runtime-corejs3/helpers/inherits');
|
|
31
30
|
var _possibleConstructorReturn = require('@babel/runtime-corejs3/helpers/possibleConstructorReturn');
|
|
32
31
|
var _getPrototypeOf = require('@babel/runtime-corejs3/helpers/getPrototypeOf');
|
|
32
|
+
var _pt = require('prop-types');
|
|
33
33
|
var _concatInstanceProperty = require('@babel/runtime-corejs3/core-js-stable/instance/concat');
|
|
34
|
-
var _setTimeout = require('@babel/runtime-corejs3/core-js-stable/set-timeout');
|
|
35
|
-
var pick = require('lodash/pick');
|
|
36
|
-
var slateReact = require('slate-react');
|
|
37
34
|
var _styled = require('@emotion/styled/base');
|
|
38
35
|
var reactIntl = require('react-intl');
|
|
39
36
|
var designSystem = require('@commercetools-uikit/design-system');
|
|
@@ -41,6 +38,11 @@ var CollapsibleMotion = require('@commercetools-uikit/collapsible-motion');
|
|
|
41
38
|
var icons = require('@commercetools-uikit/icons');
|
|
42
39
|
var Text = require('@commercetools-uikit/text');
|
|
43
40
|
var FlatButton = require('@commercetools-uikit/flat-button');
|
|
41
|
+
var slateReact = require('slate-react');
|
|
42
|
+
var slate = require('slate');
|
|
43
|
+
var slateHistory = require('slate-history');
|
|
44
|
+
var isHotkey = require('is-hotkey');
|
|
45
|
+
var pipe = require('lodash/fp/pipe');
|
|
44
46
|
var jsxRuntime = require('@emotion/react/jsx-runtime');
|
|
45
47
|
var messages = require('@commercetools-uikit/messages');
|
|
46
48
|
|
|
@@ -53,20 +55,20 @@ var _forEachInstanceProperty__default = /*#__PURE__*/_interopDefault(_forEachIns
|
|
|
53
55
|
var _Object$getOwnPropertyDescriptors__default = /*#__PURE__*/_interopDefault(_Object$getOwnPropertyDescriptors);
|
|
54
56
|
var _Object$defineProperties__default = /*#__PURE__*/_interopDefault(_Object$defineProperties);
|
|
55
57
|
var _Object$defineProperty__default = /*#__PURE__*/_interopDefault(_Object$defineProperty);
|
|
56
|
-
var _pt__default = /*#__PURE__*/_interopDefault(_pt);
|
|
57
58
|
var _reduceInstanceProperty__default = /*#__PURE__*/_interopDefault(_reduceInstanceProperty);
|
|
58
59
|
var _Object$keys__default = /*#__PURE__*/_interopDefault(_Object$keys);
|
|
59
60
|
var _mapInstanceProperty__default = /*#__PURE__*/_interopDefault(_mapInstanceProperty);
|
|
60
61
|
var Stack__default = /*#__PURE__*/_interopDefault(Stack);
|
|
61
62
|
var Constraints__default = /*#__PURE__*/_interopDefault(Constraints);
|
|
62
63
|
var _Reflect$construct__default = /*#__PURE__*/_interopDefault(_Reflect$construct);
|
|
64
|
+
var _pt__default = /*#__PURE__*/_interopDefault(_pt);
|
|
63
65
|
var _concatInstanceProperty__default = /*#__PURE__*/_interopDefault(_concatInstanceProperty);
|
|
64
|
-
var _setTimeout__default = /*#__PURE__*/_interopDefault(_setTimeout);
|
|
65
|
-
var pick__default = /*#__PURE__*/_interopDefault(pick);
|
|
66
66
|
var _styled__default = /*#__PURE__*/_interopDefault(_styled);
|
|
67
67
|
var CollapsibleMotion__default = /*#__PURE__*/_interopDefault(CollapsibleMotion);
|
|
68
68
|
var Text__default = /*#__PURE__*/_interopDefault(Text);
|
|
69
69
|
var FlatButton__default = /*#__PURE__*/_interopDefault(FlatButton);
|
|
70
|
+
var isHotkey__default = /*#__PURE__*/_interopDefault(isHotkey);
|
|
71
|
+
var pipe__default = /*#__PURE__*/_interopDefault(pipe);
|
|
70
72
|
|
|
71
73
|
function ownKeys$4(object, enumerableOnly) { var keys = _Object$keys__default["default"](object); if (_Object$getOwnPropertySymbols__default["default"]) { var symbols = _Object$getOwnPropertySymbols__default["default"](object); enumerableOnly && (symbols = _filterInstanceProperty__default["default"](symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor__default["default"](object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
72
74
|
|
|
@@ -114,6 +116,12 @@ function ownKeys$3(object, enumerableOnly) { var keys = _Object$keys__default["d
|
|
|
114
116
|
function _objectSpread$3(target) { for (var i = 1; i < arguments.length; i++) { var _context, _context2; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty__default["default"](_context = ownKeys$3(Object(source), !0)).call(_context, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](target, _Object$getOwnPropertyDescriptors__default["default"](source)) : _forEachInstanceProperty__default["default"](_context2 = ownKeys$3(Object(source))).call(_context2, function (key) { _Object$defineProperty__default["default"](target, key, _Object$getOwnPropertyDescriptor__default["default"](source, key)); }); } return target; }
|
|
115
117
|
|
|
116
118
|
function _EMOTION_STRINGIFIED_CSS_ERROR__$1() { 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)."; }
|
|
119
|
+
var HOTKEYS = {
|
|
120
|
+
'mod+b': 'bold',
|
|
121
|
+
'mod+i': 'italic',
|
|
122
|
+
'mod+u': 'underline',
|
|
123
|
+
'mod+`': 'code'
|
|
124
|
+
};
|
|
117
125
|
var COLLAPSED_HEIGHT = 32;
|
|
118
126
|
|
|
119
127
|
var LeftColumn = _styled__default["default"]("div", process.env.NODE_ENV === "production" ? {
|
|
@@ -127,7 +135,7 @@ var LeftColumn = _styled__default["default"]("div", process.env.NODE_ENV === "pr
|
|
|
127
135
|
} : {
|
|
128
136
|
name: "147rp59",
|
|
129
137
|
styles: "flex:1;display:flex;align-items:flex-start",
|
|
130
|
-
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"]} */",
|
|
138
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA6D6B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */",
|
|
131
139
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
132
140
|
});
|
|
133
141
|
|
|
@@ -142,7 +150,7 @@ var RightColumn = _styled__default["default"]("div", process.env.NODE_ENV === "p
|
|
|
142
150
|
} : {
|
|
143
151
|
name: "1m04uhl",
|
|
144
152
|
styles: "position:relative;flex:0;display:flex;align-items:flex-start",
|
|
145
|
-
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"]} */",
|
|
153
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAmE8B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */",
|
|
146
154
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
147
155
|
});
|
|
148
156
|
|
|
@@ -157,26 +165,36 @@ var Row = _styled__default["default"]("div", process.env.NODE_ENV === "productio
|
|
|
157
165
|
} : {
|
|
158
166
|
name: "skgbeu",
|
|
159
167
|
styles: "display:flex;justify-content:flex-end",
|
|
160
|
-
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"]} */",
|
|
168
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA0EsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */",
|
|
161
169
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
162
170
|
});
|
|
163
171
|
|
|
172
|
+
var renderElement = function renderElement(props) {
|
|
173
|
+
return jsxRuntime.jsx(richTextUtils.Element, _objectSpread$3({}, props));
|
|
174
|
+
};
|
|
175
|
+
|
|
176
|
+
var renderLeaf = function renderLeaf(props) {
|
|
177
|
+
return jsxRuntime.jsx(richTextUtils.Leaf, _objectSpread$3({}, props));
|
|
178
|
+
};
|
|
179
|
+
|
|
164
180
|
var _ref$1 = process.env.NODE_ENV === "production" ? {
|
|
165
181
|
name: "bcltzc",
|
|
166
182
|
styles: "flex:auto;width:0;border-top-left-radius:0;border-bottom-left-radius:0"
|
|
167
183
|
} : {
|
|
168
184
|
name: "16ohn6x-container",
|
|
169
185
|
styles: "flex:auto;width:0;border-top-left-radius:0;border-bottom-left-radius:0;label:container;",
|
|
170
|
-
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"]} */",
|
|
186
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAiOkC","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */",
|
|
171
187
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
172
188
|
};
|
|
173
189
|
|
|
174
|
-
var Editor
|
|
175
|
-
var _props$editor, _props$editor2, _props$editor3, _props$editor4;
|
|
176
|
-
|
|
190
|
+
var Editor = /*#__PURE__*/react$1.forwardRef(function (props, forwardedRef) {
|
|
177
191
|
var intl = reactIntl.useIntl();
|
|
178
192
|
var ref = react$1.useRef();
|
|
179
|
-
var
|
|
193
|
+
var createEditorWithPlugins = pipe__default["default"](slateReact.withReact, slateHistory.withHistory); // eslint-disable-next-line react-hooks/exhaustive-deps
|
|
194
|
+
|
|
195
|
+
var editor = react$1.useMemo(function () {
|
|
196
|
+
return createEditorWithPlugins(slate.createEditor());
|
|
197
|
+
}, []);
|
|
180
198
|
|
|
181
199
|
if (props.showExpandIcon) {
|
|
182
200
|
process.env.NODE_ENV !== "production" ? utils.warning(typeof props.onClickExpand === 'function', 'Editor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
@@ -206,20 +224,31 @@ var Editor$1 = function Editor(props) {
|
|
|
206
224
|
}, [setRenderToggleButton, renderToggleButton]);
|
|
207
225
|
react$1.useEffect(function () {
|
|
208
226
|
updateRenderToggleButton();
|
|
209
|
-
}, [
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
}
|
|
227
|
+
}, [editor, updateRenderToggleButton]); // resetting
|
|
228
|
+
|
|
229
|
+
var resetValue = react$1.useCallback(function (newValue) {
|
|
230
|
+
richTextUtils.resetEditor(editor, newValue);
|
|
231
|
+
}, [editor]);
|
|
232
|
+
/*
|
|
233
|
+
Resetting the editor requires access to `editor` object returned from `useSlate` hook.
|
|
234
|
+
Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle`
|
|
235
|
+
to be called from the parent component.
|
|
236
|
+
e.g. <button onMouseDown={() => ref.current?.resetValue("<p><strong>Value after reset</strong></p>")}>Reset</button>
|
|
237
|
+
*/
|
|
214
238
|
|
|
239
|
+
react$1.useImperativeHandle(forwardedRef, function () {
|
|
240
|
+
return {
|
|
241
|
+
resetValue: resetValue
|
|
242
|
+
};
|
|
243
|
+
});
|
|
215
244
|
var shouldToggleButtonTakeSpace =
|
|
216
|
-
/*
|
|
245
|
+
/*
|
|
217
246
|
- if hasLanguagesControl and there are no errors/warnings to display
|
|
218
247
|
- then the toggleButton is absolutely positioned
|
|
219
248
|
This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.
|
|
220
249
|
If there is a error or warning showing,
|
|
221
250
|
then it can be placed statically because it will then be a sibling to the error/warning message
|
|
222
|
-
and LocalizedInputToggle is placed below the errors/warnings.
|
|
251
|
+
and LocalizedInputToggle is placed below the errors/warnings.
|
|
223
252
|
*/
|
|
224
253
|
renderToggleButton && !props.hasLanguagesControl || props.error || props.warning;
|
|
225
254
|
var theme = react.useTheme();
|
|
@@ -239,31 +268,73 @@ var Editor$1 = function Editor(props) {
|
|
|
239
268
|
};
|
|
240
269
|
return jsxRuntime.jsxs(Stack__default["default"], {
|
|
241
270
|
scale: "xs",
|
|
242
|
-
children: [jsxRuntime.
|
|
271
|
+
children: [jsxRuntime.jsx(EditorWrapper, {
|
|
243
272
|
isDisabled: props.isDisabled,
|
|
244
273
|
isReadOnly: props.isReadOnly,
|
|
245
|
-
children:
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
},
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
274
|
+
children: jsxRuntime.jsxs(slateReact.Slate, {
|
|
275
|
+
editor: editor,
|
|
276
|
+
value: props.value,
|
|
277
|
+
onChange: props.onChange,
|
|
278
|
+
children: [jsxRuntime.jsx(EditorLanguageLabel, {
|
|
279
|
+
htmlFor: props.id,
|
|
280
|
+
theme: theme,
|
|
281
|
+
children: jsxRuntime.jsx(Text__default["default"].Detail, {
|
|
282
|
+
tone: "secondary",
|
|
283
|
+
children: props.language.toUpperCase()
|
|
284
|
+
})
|
|
285
|
+
}), jsxRuntime.jsxs(richTextUtils.RichTextBody // @ts-ignore
|
|
286
|
+
, {
|
|
287
|
+
ref: refObj,
|
|
288
|
+
styles: {
|
|
289
|
+
container: _ref$1
|
|
290
|
+
},
|
|
291
|
+
hasError: props.hasError,
|
|
292
|
+
isDisabled: props.isDisabled,
|
|
293
|
+
hasWarning: props.hasWarning,
|
|
294
|
+
isReadOnly: Boolean(props.isReadOnly),
|
|
295
|
+
showExpandIcon: Boolean(props.showExpandIcon),
|
|
296
|
+
onClickExpand: props.onClickExpand,
|
|
297
|
+
containerStyles: containerStyles,
|
|
298
|
+
children: [jsxRuntime.jsx(slateReact.Editable, _objectSpread$3(_objectSpread$3({}, utils.filterDataAttributes(props)), {}, {
|
|
299
|
+
name: props.name,
|
|
300
|
+
renderElement: renderElement,
|
|
301
|
+
renderLeaf: renderLeaf,
|
|
302
|
+
placeholder: props.placeholder,
|
|
303
|
+
autoFocus: props.isAutofocused,
|
|
304
|
+
onBlur: props.onBlur,
|
|
305
|
+
onFocus: function onFocus(event) {
|
|
306
|
+
var _props$onFocus;
|
|
307
|
+
|
|
308
|
+
(_props$onFocus = props.onFocus) === null || _props$onFocus === void 0 ? void 0 : _props$onFocus.call(props, event); // opens the input if it regains focus and it's closed
|
|
309
|
+
|
|
310
|
+
if (!isOpen) {
|
|
311
|
+
toggle();
|
|
312
|
+
richTextUtils.focusEditor(editor);
|
|
313
|
+
}
|
|
314
|
+
},
|
|
315
|
+
readOnly: props.isReadOnly,
|
|
316
|
+
disabled: props.isDisabled,
|
|
317
|
+
onKeyDown: function onKeyDown(event) {
|
|
318
|
+
for (var hotkey in HOTKEYS) {
|
|
319
|
+
if (isHotkey__default["default"](hotkey, event)) {
|
|
320
|
+
event.preventDefault();
|
|
321
|
+
var mark = HOTKEYS[hotkey];
|
|
322
|
+
richTextUtils.toggleMark(editor, mark);
|
|
323
|
+
break;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
})), props.children, jsxRuntime.jsx(richTextUtils.HiddenInput, {
|
|
328
|
+
isFocused: slateReact.ReactEditor.isFocused(editor),
|
|
329
|
+
handleFocus: function handleFocus() {
|
|
330
|
+
richTextUtils.focusEditor(editor);
|
|
331
|
+
},
|
|
332
|
+
id: props.id,
|
|
333
|
+
disabled: props.isDisabled,
|
|
334
|
+
readOnly: props.isReadOnly
|
|
335
|
+
})]
|
|
336
|
+
})]
|
|
337
|
+
})
|
|
267
338
|
}, props.language), jsxRuntime.jsxs(Row // NOTE: applying this style withing the `styled` component results in the production
|
|
268
339
|
// bundle to apply the style in the wrong order.
|
|
269
340
|
// For instance, we need to override the marging of the spacing component, which also
|
|
@@ -272,7 +343,7 @@ var Editor$1 = function Editor(props) {
|
|
|
272
343
|
// does the trick.
|
|
273
344
|
// TODO: revisit the logic and the implementation to maybe avoid having to apply this style.
|
|
274
345
|
, {
|
|
275
|
-
css: /*#__PURE__*/react.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"]} */"),
|
|
346
|
+
css: /*#__PURE__*/react.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":"AAkSsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */"),
|
|
276
347
|
children: [function () {
|
|
277
348
|
if (props.error) return jsxRuntime.jsx(LeftColumn, {
|
|
278
349
|
children: jsxRuntime.jsx("div", {
|
|
@@ -287,7 +358,7 @@ var Editor$1 = function Editor(props) {
|
|
|
287
358
|
return null;
|
|
288
359
|
}(), renderToggleButton && jsxRuntime.jsx(RightColumn, {
|
|
289
360
|
children: jsxRuntime.jsx(ToggleButtonWrapper, {
|
|
290
|
-
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/react.css("position:absolute;top:0;right:0;margin-top:", designSystem.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"]} */"],
|
|
361
|
+
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/react.css("position:absolute;top:0;right:0;margin-top:", designSystem.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":"AA4T2B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\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":"AA0ToB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  useMemo,\n  useImperativeHandle,\n  forwardRef,\n  type ReactNode,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n  type FocusEventHandler,\n} from 'react';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { warning, filterDataAttributes } from '@commercetools-uikit/utils';\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  Element,\n  Leaf,\n  toggleMark,\n  resetEditor,\n  focusEditor,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  Editable,\n  withReact,\n  Slate,\n  ReactEditor,\n  type RenderElementProps,\n  type RenderLeafProps,\n} from 'slate-react';\nimport { createEditor, type Descendant } from 'slate';\nimport { withHistory } from 'slate-history';\nimport isHotkey from 'is-hotkey';\nimport pipe from 'lodash/fp/pipe';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\n\nconst HOTKEYS = {\n  'mod+b': 'bold',\n  'mod+i': 'italic',\n  'mod+u': 'underline',\n  'mod+`': 'code',\n};\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  name?: string;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\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  value: Descendant[];\n  onChange: (state: Descendant[]) => void;\n  onFocus?: FocusEventHandler<HTMLDivElement>;\n  onBlur?: FocusEventHandler<HTMLDivElement>;\n  isAutofocused?: boolean;\n  placeholder?: string;\n  ref?: Ref<unknown>;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst renderElement = (props: RenderElementProps) => <Element {...props} />;\nconst renderLeaf = (props: RenderLeafProps) => <Leaf {...props} />;\n\nconst Editor = forwardRef((props: TEditorProps, forwardedRef) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n\n  const createEditorWithPlugins = pipe(withReact, withHistory);\n  // eslint-disable-next-line react-hooks/exhaustive-deps\n  const editor = useMemo(() => createEditorWithPlugins(createEditor()), []);\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  }, [editor, updateRenderToggleButton]);\n\n  // resetting\n  const resetValue = useCallback(\n    (newValue: string) => {\n      resetEditor(editor, newValue);\n    },\n    [editor]\n  );\n  /* \n  Resetting the editor requires access to `editor` object returned from `useSlate` hook.\n  Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle` \n  to be called from the parent component. \n  e.g. <button onMouseDown={() => ref.current?.resetValue(\"<p><strong>Value after reset</strong></p>\")}>Reset</button>\n  */\n  useImperativeHandle(forwardedRef, () => {\n    return {\n      resetValue,\n    };\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\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              <Slate\n                editor={editor}\n                value={props.value}\n                onChange={props.onChange}\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                <RichTextBody\n                  // @ts-ignore\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                  showExpandIcon={Boolean(props.showExpandIcon)}\n                  onClickExpand={props.onClickExpand}\n                  containerStyles={containerStyles}\n                >\n                  <Editable\n                    {...filterDataAttributes(props)}\n                    name={props.name}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    placeholder={props.placeholder}\n                    autoFocus={props.isAutofocused}\n                    onBlur={props.onBlur}\n                    onFocus={(event) => {\n                      props.onFocus?.(event);\n                      // opens the input if it regains focus and it's closed\n                      if (!isOpen) {\n                        toggle();\n                        focusEditor(editor);\n                      }\n                    }}\n                    readOnly={props.isReadOnly}\n                    disabled={props.isDisabled}\n                    onKeyDown={(event) => {\n                      for (const hotkey in HOTKEYS) {\n                        if (isHotkey(hotkey, event)) {\n                          event.preventDefault();\n                          const mark = HOTKEYS[hotkey as keyof typeof HOTKEYS];\n                          toggleMark(editor, mark);\n                          break;\n                        }\n                      }\n                    }}\n                  />\n                  {props.children}\n                  <HiddenInput\n                    isFocused={ReactEditor.isFocused(editor)}\n                    handleFocus={() => {\n                      focusEditor(editor);\n                    }}\n                    id={props.id}\n                    disabled={props.isDisabled}\n                    readOnly={props.isReadOnly}\n                  />\n                </RichTextBody>\n              </Slate>\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});\nEditor.displayName = 'Editor';\n\nexport default Editor;\n"]} */"],
|
|
291
362
|
children: jsxRuntime.jsx(FlatButton__default["default"], {
|
|
292
363
|
onClick: toggle,
|
|
293
364
|
label: intl.formatMessage(isOpen ? inputUtils.messagesMultilineInput.collapse : inputUtils.messagesMultilineInput.expand),
|
|
@@ -303,88 +374,17 @@ var Editor$1 = function Editor(props) {
|
|
|
303
374
|
});
|
|
304
375
|
}
|
|
305
376
|
});
|
|
306
|
-
};
|
|
307
|
-
|
|
308
|
-
Editor$1
|
|
309
|
-
children: _pt__default["default"].node.isRequired,
|
|
310
|
-
id: _pt__default["default"].string,
|
|
311
|
-
isOpen: _pt__default["default"].bool.isRequired,
|
|
312
|
-
isDisabled: _pt__default["default"].bool,
|
|
313
|
-
isReadOnly: _pt__default["default"].bool,
|
|
314
|
-
hasWarning: _pt__default["default"].bool,
|
|
315
|
-
hasError: _pt__default["default"].bool,
|
|
316
|
-
editor: _pt__default["default"].any,
|
|
317
|
-
error: _pt__default["default"].node,
|
|
318
|
-
warning: _pt__default["default"].node,
|
|
319
|
-
defaultExpandMultilineText: _pt__default["default"].bool.isRequired,
|
|
320
|
-
toggleLanguage: _pt__default["default"].func.isRequired,
|
|
321
|
-
language: _pt__default["default"].string.isRequired,
|
|
322
|
-
showExpandIcon: _pt__default["default"].bool.isRequired,
|
|
323
|
-
onClickExpand: _pt__default["default"].func,
|
|
324
|
-
hasLanguagesControl: _pt__default["default"].bool
|
|
325
|
-
} : {};
|
|
326
|
-
|
|
327
|
-
var renderEditor = function renderEditor(props, editor, next) {
|
|
328
|
-
var _props$editor5;
|
|
329
|
-
|
|
330
|
-
if (props.options.showExpandIcon) {
|
|
331
|
-
process.env.NODE_ENV !== "production" ? utils.warning(typeof props.options.onClickExpand === 'function', 'renderEditor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
var internalId = "".concat(props.id, "__internal__id");
|
|
335
|
-
var children = /*#__PURE__*/react$1.cloneElement(next(), {
|
|
336
|
-
id: internalId
|
|
337
|
-
});
|
|
338
|
-
|
|
339
|
-
var passedProps = _objectSpread$3(_objectSpread$3({
|
|
340
|
-
id: props.id,
|
|
341
|
-
isDisabled: props.disabled,
|
|
342
|
-
isReadOnly: props.readOnly
|
|
343
|
-
}, pick__default["default"](props.options, ['defaultExpandMultilineText', 'language', 'warning', 'error', 'hasWarning', 'hasError', 'toggleLanguage', 'isOpen', 'showExpandIcon', 'onClickExpand', 'hasLanguagesControl'])), utils.filterDataAttributes(props));
|
|
344
|
-
|
|
345
|
-
var isFocused = (_props$editor5 = props.editor) === null || _props$editor5 === void 0 ? void 0 : _props$editor5.value.selection.isFocused;
|
|
346
|
-
return jsxRuntime.jsxs(Editor$1, _objectSpread$3(_objectSpread$3({
|
|
347
|
-
editor: editor
|
|
348
|
-
}, passedProps), {}, {
|
|
349
|
-
children: [children, jsxRuntime.jsx(richTextUtils.HiddenInput, {
|
|
350
|
-
isFocused: Boolean(isFocused),
|
|
351
|
-
handleFocus: editor.focus,
|
|
352
|
-
disabled: props.disabled,
|
|
353
|
-
readOnly: props.readOnly,
|
|
354
|
-
id: props.id
|
|
355
|
-
})]
|
|
356
|
-
}));
|
|
357
|
-
};
|
|
358
|
-
|
|
359
|
-
Editor$1.displayName = 'Editor';
|
|
360
|
-
var renderEditor$1 = renderEditor;
|
|
361
|
-
|
|
362
|
-
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = _Reflect$construct__default["default"](Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
|
|
363
|
-
|
|
364
|
-
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !_Reflect$construct__default["default"]) return false; if (_Reflect$construct__default["default"].sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(_Reflect$construct__default["default"](Boolean, [], function () {})); return true; } catch (e) { return false; } }
|
|
377
|
+
});
|
|
378
|
+
Editor.displayName = 'Editor';
|
|
379
|
+
var Editor$1 = Editor;
|
|
365
380
|
|
|
366
381
|
function ownKeys$2(object, enumerableOnly) { var keys = _Object$keys__default["default"](object); if (_Object$getOwnPropertySymbols__default["default"]) { var symbols = _Object$getOwnPropertySymbols__default["default"](object); enumerableOnly && (symbols = _filterInstanceProperty__default["default"](symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor__default["default"](object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
367
382
|
|
|
368
383
|
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__default["default"](_context2 = ownKeys$2(Object(source), !0)).call(_context2, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors__default["default"] ? _Object$defineProperties__default["default"](target, _Object$getOwnPropertyDescriptors__default["default"](source)) : _forEachInstanceProperty__default["default"](_context3 = ownKeys$2(Object(source))).call(_context3, function (key) { _Object$defineProperty__default["default"](target, key, _Object$getOwnPropertyDescriptor__default["default"](source, key)); }); } return target; }
|
|
369
384
|
|
|
370
|
-
|
|
371
|
-
// TODO: remove after upgrade of `slate-react` to the latest version
|
|
372
|
-
var Editor = function Editor(props) {
|
|
373
|
-
return jsxRuntime.jsx(slateReact.Editor, _objectSpread$2({}, props));
|
|
374
|
-
};
|
|
385
|
+
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = _Reflect$construct__default["default"](Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
|
|
375
386
|
|
|
376
|
-
|
|
377
|
-
id: _pt__default["default"].string,
|
|
378
|
-
name: _pt__default["default"].string,
|
|
379
|
-
onFocus: _pt__default["default"].func,
|
|
380
|
-
onBlur: _pt__default["default"].func,
|
|
381
|
-
disabled: _pt__default["default"].bool,
|
|
382
|
-
readOnly: _pt__default["default"].bool,
|
|
383
|
-
value: _pt__default["default"].any.isRequired,
|
|
384
|
-
onChange: _pt__default["default"].func,
|
|
385
|
-
plugins: _pt__default["default"].any.isRequired,
|
|
386
|
-
renderEditor: _pt__default["default"].any.isRequired
|
|
387
|
-
} : {};
|
|
387
|
+
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !_Reflect$construct__default["default"]) return false; if (_Reflect$construct__default["default"].sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(_Reflect$construct__default["default"](Boolean, [], function () {})); return true; } catch (e) { return false; } }
|
|
388
388
|
|
|
389
389
|
var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
390
390
|
_inherits(RichTextInput, _PureComponent);
|
|
@@ -404,33 +404,25 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
404
404
|
|
|
405
405
|
_this = _super.call.apply(_super, _concatInstanceProperty__default["default"](_context = [this]).call(_context, args));
|
|
406
406
|
_this.serializedValue = _this.props.value || '';
|
|
407
|
-
_this.internalSlateValue = richTextUtils.html.deserialize(_this.props.value || '');
|
|
407
|
+
_this.internalSlateValue = richTextUtils.validSlateStateAdapter(richTextUtils.html.deserialize(_this.props.value || ''));
|
|
408
408
|
|
|
409
|
-
_this.onValueChange = function (
|
|
410
|
-
var serializedValue = richTextUtils.html.serialize(
|
|
409
|
+
_this.onValueChange = function (state) {
|
|
410
|
+
var serializedValue = richTextUtils.html.serialize(state); // because we are not using setState, we need to make sure that
|
|
411
411
|
// we perform an update when the slate value changes
|
|
412
412
|
// as this can contain things like cursor location
|
|
413
413
|
// in this case, the internalSlateValue would change
|
|
414
414
|
// but the serializedValue would NOT change.
|
|
415
415
|
|
|
416
|
-
var hasInternalSlateValueChanged = _this.internalSlateValue !==
|
|
416
|
+
var hasInternalSlateValueChanged = _this.internalSlateValue !== state;
|
|
417
417
|
var hasSerializedValueChanged = serializedValue !== _this.serializedValue;
|
|
418
|
-
_this.internalSlateValue =
|
|
418
|
+
_this.internalSlateValue = richTextUtils.validSlateStateAdapter(state);
|
|
419
419
|
_this.serializedValue = serializedValue; // the consumer only cares about the serializedValue, so it doesn't make sense to call
|
|
420
420
|
// onChange unless this value changes.
|
|
421
421
|
|
|
422
422
|
if (hasSerializedValueChanged) {
|
|
423
423
|
var _this$props$onChange, _this$props;
|
|
424
424
|
|
|
425
|
-
|
|
426
|
-
target: {
|
|
427
|
-
id: _this.props.id,
|
|
428
|
-
name: _this.props.name,
|
|
429
|
-
language: _this.props.language,
|
|
430
|
-
value: serializedValue
|
|
431
|
-
}
|
|
432
|
-
};
|
|
433
|
-
(_this$props$onChange = (_this$props = _this.props).onChange) === null || _this$props$onChange === void 0 ? void 0 : _this$props$onChange.call(_this$props, fakeEvent);
|
|
425
|
+
(_this$props$onChange = (_this$props = _this.props).onChange) === null || _this$props$onChange === void 0 ? void 0 : _this$props$onChange.call(_this$props, richTextUtils.html.serialize(state));
|
|
434
426
|
}
|
|
435
427
|
|
|
436
428
|
if (hasInternalSlateValueChanged && !hasSerializedValueChanged) {
|
|
@@ -439,44 +431,6 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
439
431
|
}
|
|
440
432
|
};
|
|
441
433
|
|
|
442
|
-
_this.onBlur = function (_event, _editor, next) {
|
|
443
|
-
next();
|
|
444
|
-
|
|
445
|
-
if (_this.props.onBlur) {
|
|
446
|
-
var fakeEvent = {
|
|
447
|
-
target: {
|
|
448
|
-
id: _this.props.id,
|
|
449
|
-
name: _this.props.name
|
|
450
|
-
}
|
|
451
|
-
};
|
|
452
|
-
|
|
453
|
-
_setTimeout__default["default"](function () {
|
|
454
|
-
var _this$props$onBlur, _this$props2;
|
|
455
|
-
|
|
456
|
-
return (_this$props$onBlur = (_this$props2 = _this.props).onBlur) === null || _this$props$onBlur === void 0 ? void 0 : _this$props$onBlur.call(_this$props2, fakeEvent);
|
|
457
|
-
}, 0);
|
|
458
|
-
}
|
|
459
|
-
};
|
|
460
|
-
|
|
461
|
-
_this.onFocus = function (_event, _editor, next) {
|
|
462
|
-
next();
|
|
463
|
-
|
|
464
|
-
if (_this.props.onFocus) {
|
|
465
|
-
var fakeEvent = {
|
|
466
|
-
target: {
|
|
467
|
-
id: _this.props.id,
|
|
468
|
-
name: _this.props.name
|
|
469
|
-
}
|
|
470
|
-
};
|
|
471
|
-
|
|
472
|
-
_setTimeout__default["default"](function () {
|
|
473
|
-
var _this$props$onFocus, _this$props3;
|
|
474
|
-
|
|
475
|
-
return (_this$props$onFocus = (_this$props3 = _this.props).onFocus) === null || _this$props$onFocus === void 0 ? void 0 : _this$props$onFocus.call(_this$props3, fakeEvent);
|
|
476
|
-
}, 0);
|
|
477
|
-
}
|
|
478
|
-
};
|
|
479
|
-
|
|
480
434
|
return _this;
|
|
481
435
|
}
|
|
482
436
|
|
|
@@ -484,7 +438,7 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
484
438
|
key: "componentDidUpdate",
|
|
485
439
|
value: function componentDidUpdate() {
|
|
486
440
|
if (this.props.value !== this.serializedValue) {
|
|
487
|
-
this.internalSlateValue = richTextUtils.html.deserialize(this.props.value);
|
|
441
|
+
this.internalSlateValue = richTextUtils.validSlateStateAdapter(richTextUtils.html.deserialize(this.props.value));
|
|
488
442
|
this.serializedValue = this.props.value;
|
|
489
443
|
this.forceUpdate();
|
|
490
444
|
}
|
|
@@ -500,21 +454,28 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
500
454
|
process.env.NODE_ENV !== "production" ? utils.warning(typeof this.props.onClickExpand === 'function', 'RichTextInput: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
501
455
|
}
|
|
502
456
|
|
|
503
|
-
return jsxRuntime.jsx(Editor, _objectSpread$2(_objectSpread$2({}, utils.filterDataAttributes(this.props)), {}, {
|
|
504
|
-
id: this.props.id,
|
|
457
|
+
return jsxRuntime.jsx(Editor$1, _objectSpread$2(_objectSpread$2({}, utils.filterDataAttributes(this.props)), {}, {
|
|
505
458
|
name: this.props.name,
|
|
506
|
-
|
|
507
|
-
readOnly: this.props.isReadOnly || this.props.isDisabled,
|
|
508
|
-
value: this.internalSlateValue,
|
|
509
|
-
onFocus: this.onFocus,
|
|
510
|
-
onBlur: this.onBlur // we can only pass this.props to the Editor that Slate understands without getting
|
|
511
|
-
// warning in the console,
|
|
512
|
-
// so instead we pass our extra this.props through this `options` prop.
|
|
513
|
-
,
|
|
514
|
-
options: pick__default["default"](this.props, ['language', 'onToggle', 'toggleLanguage', 'isOpen', 'warning', 'error', 'defaultExpandMultilineText', 'hasWarning', 'hasError', 'placeholder', 'onClickExpand', 'showExpandIcon', 'hasLanguagesControl']),
|
|
459
|
+
isReadOnly: this.props.isReadOnly || this.props.isDisabled,
|
|
515
460
|
onChange: this.onValueChange,
|
|
516
|
-
|
|
517
|
-
|
|
461
|
+
id: this.props.id,
|
|
462
|
+
value: this.internalSlateValue,
|
|
463
|
+
onFocus: this.props.onFocus,
|
|
464
|
+
onBlur: this.props.onBlur,
|
|
465
|
+
isDisabled: this.props.isDisabled,
|
|
466
|
+
defaultExpandMultilineText: this.props.defaultExpandMultilineText,
|
|
467
|
+
hasWarning: this.props.hasWarning,
|
|
468
|
+
hasError: this.props.hasError,
|
|
469
|
+
placeholder: this.props.placeholder,
|
|
470
|
+
showExpandIcon: this.props.showExpandIcon,
|
|
471
|
+
onClickExpand: this.props.onClickExpand,
|
|
472
|
+
language: this.props.language,
|
|
473
|
+
toggleLanguage: this.props.toggleLanguage,
|
|
474
|
+
isOpen: this.props.isOpen,
|
|
475
|
+
warning: this.props.warning,
|
|
476
|
+
error: this.props.error,
|
|
477
|
+
hasLanguagesControl: this.props.hasLanguagesControl,
|
|
478
|
+
ref: this.props.parentRef
|
|
518
479
|
}));
|
|
519
480
|
}
|
|
520
481
|
}]);
|
|
@@ -528,28 +489,23 @@ RichTextInput.defaultProps = {
|
|
|
528
489
|
};
|
|
529
490
|
RichTextInput.displayName = 'RichTextInput';
|
|
530
491
|
RichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
531
|
-
defaultExpandMultilineText: _pt__default["default"].bool,
|
|
532
|
-
hasError: _pt__default["default"].bool,
|
|
533
|
-
hasWarning: _pt__default["default"].bool,
|
|
534
|
-
id: _pt__default["default"].string,
|
|
535
492
|
name: _pt__default["default"].string,
|
|
536
|
-
placeholder: _pt__default["default"].string,
|
|
537
|
-
isDisabled: _pt__default["default"].bool,
|
|
538
|
-
isReadOnly: _pt__default["default"].bool,
|
|
539
493
|
onChange: _pt__default["default"].func,
|
|
540
|
-
onBlur: _pt__default["default"].func,
|
|
541
|
-
onFocus: _pt__default["default"].func,
|
|
542
494
|
value: _pt__default["default"].string.isRequired,
|
|
543
|
-
|
|
544
|
-
onClickExpand: _pt__default["default"].func,
|
|
545
|
-
hasLanguagesControl: _pt__default["default"].bool,
|
|
495
|
+
parentRef: _pt__default["default"].any,
|
|
546
496
|
language: _pt__default["default"].string.isRequired,
|
|
547
497
|
isOpen: _pt__default["default"].bool.isRequired,
|
|
548
498
|
toggleLanguage: _pt__default["default"].func.isRequired,
|
|
549
499
|
warning: _pt__default["default"].node,
|
|
550
500
|
error: _pt__default["default"].string
|
|
551
501
|
} : {};
|
|
552
|
-
var
|
|
502
|
+
var RichTextInputWithRef = /*#__PURE__*/react$1.forwardRef(function (props, ref) {
|
|
503
|
+
return jsxRuntime.jsx(RichTextInput, _objectSpread$2({
|
|
504
|
+
parentRef: ref
|
|
505
|
+
}, props));
|
|
506
|
+
});
|
|
507
|
+
RichTextInputWithRef.displayName = 'RichTextInputWithRef';
|
|
508
|
+
var RichTextInput$1 = RichTextInputWithRef;
|
|
553
509
|
|
|
554
510
|
function ownKeys$1(object, enumerableOnly) { var keys = _Object$keys__default["default"](object); if (_Object$getOwnPropertySymbols__default["default"]) { var symbols = _Object$getOwnPropertySymbols__default["default"](object); enumerableOnly && (symbols = _filterInstanceProperty__default["default"](symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor__default["default"](object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
555
511
|
|
|
@@ -607,11 +563,11 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
607
563
|
} : {
|
|
608
564
|
name: "9a4cy7-LocalizedRichTextInput",
|
|
609
565
|
styles: "align-self:flex-start;label:LocalizedRichTextInput;",
|
|
610
|
-
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 TCustomEvent = {\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: TCustomEvent) => 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: TCustomEvent) => void;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: (event: TCustomEvent) => 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"]} */",
|
|
566
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-rich-text-input.tsx"],"names":[],"mappings":"AAsVsB","file":"localized-rich-text-input.tsx","sourcesContent":["import {\n  useReducer,\n  useCallback,\n  forwardRef,\n  type ReactNode,\n  type MouseEvent,\n  type KeyboardEvent,\n  type ForwardRefExoticComponent,\n  type RefAttributes,\n  type FocusEventHandler,\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, filterDataAttributes } 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>;\ntype TCustomEvent = {\n  target: {\n    id?: string;\n    name?: string;\n    language?: string;\n    value?: string;\n  };\n};\n\nexport type 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: TCustomEvent) => 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?: FocusEventHandler<HTMLDivElement>;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: FocusEventHandler<HTMLDivElement>;\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\n// When component is using `forwardRef` only `defaultProps` and `displayName` are recognized by default as static props\ntype StaticProps = {\n  RequiredValueErrorMessage: typeof RequiredValueErrorMessage;\n  getId: typeof getId;\n  getName: typeof getName;\n  createLocalizedString: typeof localized.createLocalizedString;\n  isEmpty: typeof localized.isEmpty;\n  omitEmptyTranslations: typeof localized.omitEmptyTranslations;\n  isTouched: typeof isTouched;\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: ForwardRefExoticComponent<\n  TLocalizedRichTextInputProps & RefAttributes<unknown>\n> &\n  Partial<StaticProps> = forwardRef(\n  (props: TLocalizedRichTextInputProps, ref) => {\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] =\n      useReducer(expandedTranslationsReducer, initialExpandedTranslationsState);\n\n    const defaultExpansionState = Boolean(\n      props.hideLanguageExpansionControls || props.defaultExpandLanguages\n    );\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 createChangeHandler = useCallback(\n      (language: string) => (state: string) =>\n        props.onChange?.({\n          target: {\n            id: props.id,\n            name: props.name,\n            language,\n            value: state,\n          },\n        }),\n      // eslint-disable-next-line react-hooks/exhaustive-deps\n      [props.id, props.name, props.onChange]\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                  {...filterDataAttributes(props)}\n                  key={language}\n                  id={getId(props.id, language)}\n                  name={getName(props.name, language)}\n                  value={props.value[language]}\n                  onChange={createChangeHandler(language)}\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                  defaultExpandMultilineText={Boolean(\n                    props.defaultExpandMultilineText\n                  )}\n                  ref={ref}\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 ||\n                      hasWarningInRemainingLanguages\n                  )\n                }\n                remainingLocalizations={languages.length - 1}\n              />\n            </div>\n          )}\n        </Stack>\n      </Constraints.Horizontal>\n    );\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"]} */",
|
|
611
567
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
612
568
|
};
|
|
613
569
|
|
|
614
|
-
var LocalizedRichTextInput = function
|
|
570
|
+
var LocalizedRichTextInput = /*#__PURE__*/react$1.forwardRef(function (props, ref) {
|
|
615
571
|
var _context2;
|
|
616
572
|
|
|
617
573
|
if (!props.isReadOnly) {
|
|
@@ -635,8 +591,7 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
635
591
|
expandedTranslationsState = _useReducer2[0],
|
|
636
592
|
expandedTranslationsDispatch = _useReducer2[1];
|
|
637
593
|
|
|
638
|
-
var defaultExpansionState = props.hideLanguageExpansionControls || props.defaultExpandLanguages
|
|
639
|
-
false;
|
|
594
|
+
var defaultExpansionState = Boolean(props.hideLanguageExpansionControls || props.defaultExpandLanguages);
|
|
640
595
|
|
|
641
596
|
var _useToggleState = hooks.useToggleState(defaultExpansionState),
|
|
642
597
|
_useToggleState2 = _slicedToArray(_useToggleState, 2),
|
|
@@ -649,6 +604,21 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
649
604
|
payload: language
|
|
650
605
|
});
|
|
651
606
|
}, [expandedTranslationsDispatch]);
|
|
607
|
+
var createChangeHandler = react$1.useCallback(function (language) {
|
|
608
|
+
return function (state) {
|
|
609
|
+
var _props$onChange;
|
|
610
|
+
|
|
611
|
+
return (_props$onChange = props.onChange) === null || _props$onChange === void 0 ? void 0 : _props$onChange.call(props, {
|
|
612
|
+
target: {
|
|
613
|
+
id: props.id,
|
|
614
|
+
name: props.name,
|
|
615
|
+
language: language,
|
|
616
|
+
value: state
|
|
617
|
+
}
|
|
618
|
+
});
|
|
619
|
+
};
|
|
620
|
+
}, // eslint-disable-next-line react-hooks/exhaustive-deps
|
|
621
|
+
[props.id, props.name, props.onChange]);
|
|
652
622
|
var languages = localizedUtils.sortLanguages(props.selectedLanguage, _Object$keys__default["default"](props.value));
|
|
653
623
|
var hasErrorInRemainingLanguages = props.hasError || localizedUtils.getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);
|
|
654
624
|
var hasWarningInRemainingLanguages = props.hasWarning || localizedUtils.getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);
|
|
@@ -672,11 +642,12 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
672
642
|
if (!isFirstLanguage && !areLanguagesOpened) return null;
|
|
673
643
|
var isLastLanguage = index === languages.length - 1;
|
|
674
644
|
var hasLanguagesControl = isFirstLanguage && !areLanguagesOpened || isLastLanguage;
|
|
675
|
-
return
|
|
676
|
-
|
|
677
|
-
|
|
645
|
+
return react.createElement(RichTextInput$1, _objectSpread(_objectSpread({}, utils.filterDataAttributes(props)), {}, {
|
|
646
|
+
key: language,
|
|
647
|
+
id: localizedUtils.getId(props.id, language),
|
|
648
|
+
name: localizedUtils.getName(props.name, language),
|
|
678
649
|
value: props.value[language],
|
|
679
|
-
onChange:
|
|
650
|
+
onChange: createChangeHandler(language),
|
|
680
651
|
language: language,
|
|
681
652
|
isOpen: expandedTranslationsState[language],
|
|
682
653
|
toggleLanguage: toggleLanguage,
|
|
@@ -691,8 +662,10 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
691
662
|
error: props.errors && props.errors[language],
|
|
692
663
|
showExpandIcon: props.showExpandIcon,
|
|
693
664
|
onClickExpand: props.onClickExpand,
|
|
694
|
-
hasLanguagesControl: hasLanguagesControl
|
|
695
|
-
|
|
665
|
+
hasLanguagesControl: hasLanguagesControl,
|
|
666
|
+
defaultExpandMultilineText: Boolean(props.defaultExpandMultilineText),
|
|
667
|
+
ref: ref
|
|
668
|
+
}, localizedUtils.createLocalizedDataAttributes(props, language)));
|
|
696
669
|
})
|
|
697
670
|
}), shouldRenderLanguagesControl && jsxRuntime.jsx("div", {
|
|
698
671
|
css: _ref,
|
|
@@ -705,30 +678,7 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
705
678
|
})]
|
|
706
679
|
})
|
|
707
680
|
});
|
|
708
|
-
};
|
|
709
|
-
|
|
710
|
-
LocalizedRichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
711
|
-
id: _pt__default["default"].string,
|
|
712
|
-
name: _pt__default["default"].string,
|
|
713
|
-
value: _pt__default["default"].objectOf(_pt__default["default"].string).isRequired,
|
|
714
|
-
onChange: _pt__default["default"].func,
|
|
715
|
-
selectedLanguage: _pt__default["default"].string.isRequired,
|
|
716
|
-
onBlur: _pt__default["default"].func,
|
|
717
|
-
onFocus: _pt__default["default"].func,
|
|
718
|
-
defaultExpandMultilineText: _pt__default["default"].bool,
|
|
719
|
-
hideLanguageExpansionControls: _pt__default["default"].bool,
|
|
720
|
-
defaultExpandLanguages: _pt__default["default"].bool,
|
|
721
|
-
isDisabled: _pt__default["default"].bool,
|
|
722
|
-
isReadOnly: _pt__default["default"].bool,
|
|
723
|
-
placeholder: _pt__default["default"].objectOf(_pt__default["default"].string),
|
|
724
|
-
horizontalConstraint: _pt__default["default"].oneOf([7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
725
|
-
hasError: _pt__default["default"].bool,
|
|
726
|
-
hasWarning: _pt__default["default"].bool,
|
|
727
|
-
errors: _pt__default["default"].objectOf(_pt__default["default"].string),
|
|
728
|
-
warnings: _pt__default["default"].objectOf(_pt__default["default"].node),
|
|
729
|
-
showExpandIcon: _pt__default["default"].bool.isRequired,
|
|
730
|
-
onClickExpand: _pt__default["default"].func
|
|
731
|
-
} : {};
|
|
681
|
+
});
|
|
732
682
|
LocalizedRichTextInput.displayName = 'LocalizedRichTextInput';
|
|
733
683
|
LocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage$1;
|
|
734
684
|
LocalizedRichTextInput.getId = localizedUtils.getId;
|
|
@@ -741,7 +691,7 @@ LocalizedRichTextInput.isTouched = localizedUtils.isTouched;
|
|
|
741
691
|
var LocalizedRichTextInput$1 = LocalizedRichTextInput;
|
|
742
692
|
|
|
743
693
|
// NOTE: This string will be replaced on build time with the package version.
|
|
744
|
-
var version = "
|
|
694
|
+
var version = "15.0.0";
|
|
745
695
|
|
|
746
696
|
exports["default"] = LocalizedRichTextInput$1;
|
|
747
697
|
exports.version = version;
|