@commercetools-uikit/localized-rich-text-input 14.0.3 → 15.0.1
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 +212 -242
- package/dist/commercetools-uikit-localized-rich-text-input.cjs.prod.js +203 -171
- package/dist/commercetools-uikit-localized-rich-text-input.esm.js +211 -242
- 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
|
@@ -1,24 +1,25 @@
|
|
|
1
1
|
import _Object$getOwnPropertySymbols from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-symbols';
|
|
2
2
|
import _filterInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/filter';
|
|
3
3
|
import _Object$getOwnPropertyDescriptor from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptor';
|
|
4
|
-
import _forEachInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/for-each';
|
|
5
4
|
import _Object$getOwnPropertyDescriptors from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptors';
|
|
6
5
|
import _Object$defineProperties from '@babel/runtime-corejs3/core-js-stable/object/define-properties';
|
|
7
6
|
import _Object$defineProperty from '@babel/runtime-corejs3/core-js-stable/object/define-property';
|
|
8
7
|
import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
|
|
9
8
|
import _defineProperty from '@babel/runtime-corejs3/helpers/esm/defineProperty';
|
|
10
|
-
import _pt from 'prop-types';
|
|
11
9
|
import _reduceInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/reduce';
|
|
12
10
|
import _Object$keys from '@babel/runtime-corejs3/core-js-stable/object/keys';
|
|
11
|
+
import _concatInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/concat';
|
|
12
|
+
import _Map from '@babel/runtime-corejs3/core-js-stable/map';
|
|
13
|
+
import _forEachInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/for-each';
|
|
13
14
|
import _mapInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/map';
|
|
14
|
-
import {
|
|
15
|
-
import { css, useTheme } from '@emotion/react';
|
|
15
|
+
import { forwardRef, useRef, useMemo, useState, useCallback, useEffect, useImperativeHandle, PureComponent, useReducer } from 'react';
|
|
16
|
+
import { css, useTheme, createElement } from '@emotion/react';
|
|
16
17
|
import Stack from '@commercetools-uikit/spacings-stack';
|
|
17
18
|
import Constraints from '@commercetools-uikit/constraints';
|
|
18
|
-
import {
|
|
19
|
+
import { useToggleState } from '@commercetools-uikit/hooks';
|
|
19
20
|
import { sortLanguages, getHasErrorOnRemainingLanguages, getHasWarningOnRemainingLanguages, createLocalizedDataAttributes, getId, getName, isTouched } from '@commercetools-uikit/localized-utils';
|
|
20
21
|
import { messagesMultilineInput, messagesLocalizedInput, LocalizedInputToggle } from '@commercetools-uikit/input-utils';
|
|
21
|
-
import {
|
|
22
|
+
import { resetEditor, RichTextBody, focusEditor, toggleMark, HiddenInput, Element, Leaf, validSlateStateAdapter, html, localized } from '@commercetools-uikit/rich-text-utils';
|
|
22
23
|
import { warning, filterDataAttributes } from '@commercetools-uikit/utils';
|
|
23
24
|
import _Reflect$construct from '@babel/runtime-corejs3/core-js-stable/reflect/construct';
|
|
24
25
|
import _classCallCheck from '@babel/runtime-corejs3/helpers/esm/classCallCheck';
|
|
@@ -26,10 +27,7 @@ import _createClass from '@babel/runtime-corejs3/helpers/esm/createClass';
|
|
|
26
27
|
import _inherits from '@babel/runtime-corejs3/helpers/esm/inherits';
|
|
27
28
|
import _possibleConstructorReturn from '@babel/runtime-corejs3/helpers/esm/possibleConstructorReturn';
|
|
28
29
|
import _getPrototypeOf from '@babel/runtime-corejs3/helpers/esm/getPrototypeOf';
|
|
29
|
-
import
|
|
30
|
-
import _setTimeout from '@babel/runtime-corejs3/core-js-stable/set-timeout';
|
|
31
|
-
import pick from 'lodash/pick';
|
|
32
|
-
import { Editor as Editor$2 } from 'slate-react';
|
|
30
|
+
import _pt from 'prop-types';
|
|
33
31
|
import _styled from '@emotion/styled/base';
|
|
34
32
|
import { useIntl, FormattedMessage } from 'react-intl';
|
|
35
33
|
import { customProperties, designTokens } from '@commercetools-uikit/design-system';
|
|
@@ -37,7 +35,12 @@ import CollapsibleMotion from '@commercetools-uikit/collapsible-motion';
|
|
|
37
35
|
import { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';
|
|
38
36
|
import Text from '@commercetools-uikit/text';
|
|
39
37
|
import FlatButton from '@commercetools-uikit/flat-button';
|
|
40
|
-
import {
|
|
38
|
+
import { withReact, Slate, Editable, ReactEditor } from 'slate-react';
|
|
39
|
+
import { createEditor } from 'slate';
|
|
40
|
+
import { withHistory } from 'slate-history';
|
|
41
|
+
import isHotkey from 'is-hotkey';
|
|
42
|
+
import pipe from 'lodash/fp/pipe';
|
|
43
|
+
import { jsx, jsxs } from '@emotion/react/jsx-runtime';
|
|
41
44
|
import { ErrorMessage } from '@commercetools-uikit/messages';
|
|
42
45
|
|
|
43
46
|
function ownKeys$4(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -86,6 +89,12 @@ function ownKeys$3(object, enumerableOnly) { var keys = _Object$keys(object); if
|
|
|
86
89
|
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(_context = ownKeys$3(Object(source), !0)).call(_context, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(target, _Object$getOwnPropertyDescriptors(source)) : _forEachInstanceProperty(_context2 = ownKeys$3(Object(source))).call(_context2, function (key) { _Object$defineProperty(target, key, _Object$getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
87
90
|
|
|
88
91
|
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)."; }
|
|
92
|
+
var HOTKEYS = {
|
|
93
|
+
'mod+b': 'bold',
|
|
94
|
+
'mod+i': 'italic',
|
|
95
|
+
'mod+u': 'underline',
|
|
96
|
+
'mod+`': 'code'
|
|
97
|
+
};
|
|
89
98
|
var COLLAPSED_HEIGHT = 32;
|
|
90
99
|
|
|
91
100
|
var LeftColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
@@ -99,7 +108,7 @@ var LeftColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
99
108
|
} : {
|
|
100
109
|
name: "147rp59",
|
|
101
110
|
styles: "flex:1;display:flex;align-items:flex-start",
|
|
102
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAsC6B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
111
|
+
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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */",
|
|
103
112
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
104
113
|
});
|
|
105
114
|
|
|
@@ -114,7 +123,7 @@ var RightColumn = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
114
123
|
} : {
|
|
115
124
|
name: "1m04uhl",
|
|
116
125
|
styles: "position:relative;flex:0;display:flex;align-items:flex-start",
|
|
117
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA4C8B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
126
|
+
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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */",
|
|
118
127
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
119
128
|
});
|
|
120
129
|
|
|
@@ -129,26 +138,36 @@ var Row = _styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
129
138
|
} : {
|
|
130
139
|
name: "skgbeu",
|
|
131
140
|
styles: "display:flex;justify-content:flex-end",
|
|
132
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAmDsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
141
|
+
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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */",
|
|
133
142
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
134
143
|
});
|
|
135
144
|
|
|
145
|
+
var renderElement = function renderElement(props) {
|
|
146
|
+
return jsx(Element, _objectSpread$3({}, props));
|
|
147
|
+
};
|
|
148
|
+
|
|
149
|
+
var renderLeaf = function renderLeaf(props) {
|
|
150
|
+
return jsx(Leaf, _objectSpread$3({}, props));
|
|
151
|
+
};
|
|
152
|
+
|
|
136
153
|
var _ref$1 = process.env.NODE_ENV === "production" ? {
|
|
137
154
|
name: "bcltzc",
|
|
138
155
|
styles: "flex:auto;width:0;border-top-left-radius:0;border-bottom-left-radius:0"
|
|
139
156
|
} : {
|
|
140
157
|
name: "16ohn6x-container",
|
|
141
158
|
styles: "flex:auto;width:0;border-top-left-radius:0;border-bottom-left-radius:0;label:container;",
|
|
142
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AA6KgC","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */",
|
|
159
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAsOkC","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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */",
|
|
143
160
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
|
|
144
161
|
};
|
|
145
162
|
|
|
146
|
-
var Editor
|
|
147
|
-
var _props$editor, _props$editor2, _props$editor3, _props$editor4;
|
|
148
|
-
|
|
163
|
+
var Editor = /*#__PURE__*/forwardRef(function (props, forwardedRef) {
|
|
149
164
|
var intl = useIntl();
|
|
150
165
|
var ref = useRef();
|
|
151
|
-
var
|
|
166
|
+
var createEditorWithPlugins = pipe(withReact, withHistory); // eslint-disable-next-line react-hooks/exhaustive-deps
|
|
167
|
+
|
|
168
|
+
var editor = useMemo(function () {
|
|
169
|
+
return createEditorWithPlugins(createEditor());
|
|
170
|
+
}, []);
|
|
152
171
|
|
|
153
172
|
if (props.showExpandIcon) {
|
|
154
173
|
process.env.NODE_ENV !== "production" ? warning(typeof props.onClickExpand === 'function', 'Editor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
@@ -178,20 +197,34 @@ var Editor$1 = function Editor(props) {
|
|
|
178
197
|
}, [setRenderToggleButton, renderToggleButton]);
|
|
179
198
|
useEffect(function () {
|
|
180
199
|
updateRenderToggleButton();
|
|
181
|
-
}, [
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
200
|
+
}, [editor, updateRenderToggleButton]); // resetting
|
|
201
|
+
|
|
202
|
+
var resetValue = useCallback(function (newValue) {
|
|
203
|
+
var _newValue$props$langu;
|
|
204
|
+
|
|
205
|
+
var newStringValue = typeof newValue === 'string' ? newValue : (_newValue$props$langu = newValue === null || newValue === void 0 ? void 0 : newValue[props.language]) !== null && _newValue$props$langu !== void 0 ? _newValue$props$langu : '';
|
|
206
|
+
resetEditor(editor, newStringValue);
|
|
207
|
+
}, [editor, props.language]);
|
|
208
|
+
/*
|
|
209
|
+
Resetting the editor requires access to `editor` object returned from `useSlate` hook.
|
|
210
|
+
Therefore, `reset` function is attached to the passed `ref` object via `useImperativeHandle`
|
|
211
|
+
to be called from the parent component.
|
|
212
|
+
e.g. <button onMouseDown={() => ref.current?.resetValue("<p><strong>Value after reset</strong></p>")}>Reset</button>
|
|
213
|
+
*/
|
|
186
214
|
|
|
215
|
+
useImperativeHandle(forwardedRef, function () {
|
|
216
|
+
return {
|
|
217
|
+
resetValue: resetValue
|
|
218
|
+
};
|
|
219
|
+
});
|
|
187
220
|
var shouldToggleButtonTakeSpace =
|
|
188
|
-
/*
|
|
221
|
+
/*
|
|
189
222
|
- if hasLanguagesControl and there are no errors/warnings to display
|
|
190
223
|
- then the toggleButton is absolutely positioned
|
|
191
224
|
This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.
|
|
192
225
|
If there is a error or warning showing,
|
|
193
226
|
then it can be placed statically because it will then be a sibling to the error/warning message
|
|
194
|
-
and LocalizedInputToggle is placed below the errors/warnings.
|
|
227
|
+
and LocalizedInputToggle is placed below the errors/warnings.
|
|
195
228
|
*/
|
|
196
229
|
renderToggleButton && !props.hasLanguagesControl || props.error || props.warning;
|
|
197
230
|
var theme = useTheme();
|
|
@@ -211,31 +244,73 @@ var Editor$1 = function Editor(props) {
|
|
|
211
244
|
};
|
|
212
245
|
return jsxs(Stack, {
|
|
213
246
|
scale: "xs",
|
|
214
|
-
children: [
|
|
247
|
+
children: [jsx(EditorWrapper, {
|
|
215
248
|
isDisabled: props.isDisabled,
|
|
216
249
|
isReadOnly: props.isReadOnly,
|
|
217
|
-
children:
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
},
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
250
|
+
children: jsxs(Slate, {
|
|
251
|
+
editor: editor,
|
|
252
|
+
value: props.value,
|
|
253
|
+
onChange: props.onChange,
|
|
254
|
+
children: [jsx(EditorLanguageLabel, {
|
|
255
|
+
htmlFor: props.id,
|
|
256
|
+
theme: theme,
|
|
257
|
+
children: jsx(Text.Detail, {
|
|
258
|
+
tone: "secondary",
|
|
259
|
+
children: props.language.toUpperCase()
|
|
260
|
+
})
|
|
261
|
+
}), jsxs(RichTextBody // @ts-ignore
|
|
262
|
+
, {
|
|
263
|
+
ref: refObj,
|
|
264
|
+
styles: {
|
|
265
|
+
container: _ref$1
|
|
266
|
+
},
|
|
267
|
+
hasError: props.hasError,
|
|
268
|
+
isDisabled: props.isDisabled,
|
|
269
|
+
hasWarning: props.hasWarning,
|
|
270
|
+
isReadOnly: Boolean(props.isReadOnly),
|
|
271
|
+
showExpandIcon: Boolean(props.showExpandIcon),
|
|
272
|
+
onClickExpand: props.onClickExpand,
|
|
273
|
+
containerStyles: containerStyles,
|
|
274
|
+
children: [jsx(Editable, _objectSpread$3(_objectSpread$3({}, filterDataAttributes(props)), {}, {
|
|
275
|
+
name: props.name,
|
|
276
|
+
renderElement: renderElement,
|
|
277
|
+
renderLeaf: renderLeaf,
|
|
278
|
+
placeholder: props.placeholder,
|
|
279
|
+
autoFocus: props.isAutofocused,
|
|
280
|
+
onBlur: props.onBlur,
|
|
281
|
+
onFocus: function onFocus(event) {
|
|
282
|
+
var _props$onFocus;
|
|
283
|
+
|
|
284
|
+
(_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
|
|
285
|
+
|
|
286
|
+
if (!isOpen) {
|
|
287
|
+
toggle();
|
|
288
|
+
focusEditor(editor);
|
|
289
|
+
}
|
|
290
|
+
},
|
|
291
|
+
readOnly: props.isReadOnly,
|
|
292
|
+
disabled: props.isDisabled,
|
|
293
|
+
onKeyDown: function onKeyDown(event) {
|
|
294
|
+
for (var hotkey in HOTKEYS) {
|
|
295
|
+
if (isHotkey(hotkey, event)) {
|
|
296
|
+
event.preventDefault();
|
|
297
|
+
var mark = HOTKEYS[hotkey];
|
|
298
|
+
toggleMark(editor, mark);
|
|
299
|
+
break;
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
})), props.children, jsx(HiddenInput, {
|
|
304
|
+
isFocused: ReactEditor.isFocused(editor),
|
|
305
|
+
handleFocus: function handleFocus() {
|
|
306
|
+
focusEditor(editor);
|
|
307
|
+
},
|
|
308
|
+
id: props.id,
|
|
309
|
+
disabled: props.isDisabled,
|
|
310
|
+
readOnly: props.isReadOnly
|
|
311
|
+
})]
|
|
312
|
+
})]
|
|
313
|
+
})
|
|
239
314
|
}, props.language), jsxs(Row // NOTE: applying this style withing the `styled` component results in the production
|
|
240
315
|
// bundle to apply the style in the wrong order.
|
|
241
316
|
// For instance, we need to override the marging of the spacing component, which also
|
|
@@ -244,7 +319,7 @@ var Editor$1 = function Editor(props) {
|
|
|
244
319
|
// does the trick.
|
|
245
320
|
// TODO: revisit the logic and the implementation to maybe avoid having to apply this style.
|
|
246
321
|
, {
|
|
247
|
-
css: /*#__PURE__*/css("margin-top:", shouldToggleButtonTakeSpace ? 'inherit' : '0px !important', ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAwMsB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"),
|
|
322
|
+
css: /*#__PURE__*/css("margin-top:", shouldToggleButtonTakeSpace ? 'inherit' : '0px !important', ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAuSsB","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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */"),
|
|
248
323
|
children: [function () {
|
|
249
324
|
if (props.error) return jsx(LeftColumn, {
|
|
250
325
|
children: jsx("div", {
|
|
@@ -259,7 +334,7 @@ var Editor$1 = function Editor(props) {
|
|
|
259
334
|
return null;
|
|
260
335
|
}(), renderToggleButton && jsx(RightColumn, {
|
|
261
336
|
children: jsx(ToggleButtonWrapper, {
|
|
262
|
-
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/css("position:absolute;top:0;right:0;margin-top:", customProperties.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAkO2B","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"), process.env.NODE_ENV === "production" ? "" : ";label:Editor;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAgOoB","file":"editor.tsx","sourcesContent":["import {\n  useRef,\n  useState,\n  useCallback,\n  useEffect,\n  cloneElement,\n  type ReactNode,\n  type ReactElement,\n  type LegacyRef,\n  type RefObject,\n  type Ref,\n} from 'react';\nimport pick from 'lodash/pick';\nimport { css, useTheme } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { useIntl } from 'react-intl';\nimport { customProperties } from '@commercetools-uikit/design-system';\nimport { filterDataAttributes, warning } from '@commercetools-uikit/utils';\nimport { usePrevious } from '@commercetools-uikit/hooks';\nimport CollapsibleMotion from '@commercetools-uikit/collapsible-motion';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport { AngleUpIcon, AngleDownIcon } from '@commercetools-uikit/icons';\nimport Text from '@commercetools-uikit/text';\nimport FlatButton from '@commercetools-uikit/flat-button';\nimport { messagesMultilineInput } from '@commercetools-uikit/input-utils';\nimport {\n  RichTextBody,\n  HiddenInput,\n} from '@commercetools-uikit/rich-text-utils';\nimport {\n  EditorWrapper,\n  EditorLanguageLabel,\n  ToggleButtonWrapper,\n} from './editor.styles';\nimport type { TEditor } from './editor.types';\n\nconst COLLAPSED_HEIGHT = 32;\n\nconst LeftColumn = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst RightColumn = styled.div`\n  position: relative;\n  flex: 0;\n  display: flex;\n  align-items: flex-start;\n`;\n\nconst Row = styled.div`\n  display: flex;\n  justify-content: flex-end;\n`;\n\nexport type TEditorProps = {\n  children: ReactNode;\n  id?: string;\n  isOpen: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  editor?: TEditor;\n  error?: ReactNode;\n  warning?: ReactNode;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  language: string;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\ntype TNodeRefObject = {\n  clientHeight: number;\n} & LegacyRef<HTMLDivElement>;\n\ntype TRichTextEditorBodyRef = {\n  registerContentNode: TNodeRefObject;\n  containerRef: RefObject<HTMLDivElement>;\n};\n\nconst Editor = (props: TEditorProps) => {\n  const intl = useIntl();\n  const ref = useRef<HTMLDivElement>();\n  const prevIsFocused = usePrevious(props.editor?.value.selection.isFocused);\n\n  if (props.showExpandIcon) {\n    warning(\n      typeof props.onClickExpand === 'function',\n      'Editor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const [renderToggleButton, setRenderToggleButton] = useState(false);\n\n  const { toggleLanguage } = props;\n  const onToggle = useCallback(() => {\n    toggleLanguage(props.language);\n  }, [toggleLanguage, props.language]);\n\n  const updateRenderToggleButton = useCallback(() => {\n    const doesExceedCollapsedHeightLimit =\n      Number(ref.current?.clientHeight) > COLLAPSED_HEIGHT;\n\n    if (doesExceedCollapsedHeightLimit && !renderToggleButton) {\n      setRenderToggleButton(true);\n    }\n    if (!doesExceedCollapsedHeightLimit && renderToggleButton) {\n      setRenderToggleButton(false);\n    }\n  }, [setRenderToggleButton, renderToggleButton]);\n\n  useEffect(() => {\n    updateRenderToggleButton();\n  }, [props.editor?.value.document, updateRenderToggleButton]);\n\n  // opens the input if it regains focus and it's closed\n  if (\n    prevIsFocused !== props.editor?.value.selection.isFocused &&\n    props.editor?.value.selection.isFocused &&\n    !props.isOpen\n  ) {\n    onToggle();\n  }\n\n  const shouldToggleButtonTakeSpace =\n    /*\n      - if hasLanguagesControl and there are no errors/warnings to display\n      - then the toggleButton is absolutely positioned\n      This is because the toggle button is placed next to the LocalizedInputToggle without being siblings in the DOM.\n      If there is a error or warning showing,\n      then it can be placed statically because it will then be a sibling to the error/warning message\n      and LocalizedInputToggle is placed below the errors/warnings.\n    */\n    (renderToggleButton && !props.hasLanguagesControl) ||\n    props.error ||\n    props.warning;\n\n  const theme = useTheme();\n\n  return (\n    <CollapsibleMotion\n      minHeight={COLLAPSED_HEIGHT}\n      isClosed={!props.isOpen}\n      onToggle={onToggle}\n      isDefaultClosed={!props.defaultExpandMultilineText}\n    >\n      {({ isOpen, toggle, containerStyles, registerContentNode }) => {\n        const refObj = {\n          containerRef: ref,\n          registerContentNode,\n        };\n\n        return (\n          <Stack scale=\"xs\">\n            <EditorWrapper\n              key={props.language}\n              isDisabled={props.isDisabled}\n              isReadOnly={props.isReadOnly}\n            >\n              <EditorLanguageLabel htmlFor={props.id} theme={theme}>\n                {/* FIXME: add proper tone for disabled when tones are refactored */}\n                <Text.Detail tone=\"secondary\">\n                  {props.language.toUpperCase()}\n                </Text.Detail>\n              </EditorLanguageLabel>\n\n              <RichTextBody\n                ref={refObj as unknown as Ref<TRichTextEditorBodyRef>}\n                styles={{\n                  container: css`\n                    flex: auto;\n                    width: 0;\n                    border-top-left-radius: 0;\n                    border-bottom-left-radius: 0;\n                  `,\n                }}\n                hasError={props.hasError}\n                isDisabled={props.isDisabled}\n                hasWarning={props.hasWarning}\n                isReadOnly={Boolean(props.isReadOnly)}\n                editor={props.editor as TEditor}\n                containerStyles={containerStyles}\n                showExpandIcon={props.showExpandIcon}\n                onClickExpand={props.onClickExpand}\n              >\n                {props.children}\n              </RichTextBody>\n            </EditorWrapper>\n            <Row\n              // NOTE: applying this style withing the `styled` component results in the production\n              // bundle to apply the style in the wrong order.\n              // For instance, we need to override the marging of the spacing component, which also\n              // uses `!important`.\n              // Anyway, apparently by passing the style as a `css` prop to the `styled` component\n              // does the trick.\n              // TODO: revisit the logic and the implementation to maybe avoid having to apply this style.\n              css={css`\n                margin-top: ${shouldToggleButtonTakeSpace\n                  ? 'inherit'\n                  : '0px !important'};\n              `}\n            >\n              {(() => {\n                if (props.error)\n                  return (\n                    <LeftColumn>\n                      <div>{props.error}</div>\n                    </LeftColumn>\n                  );\n                if (props.warning)\n                  return (\n                    <LeftColumn>\n                      <div>{props.warning}</div>\n                    </LeftColumn>\n                  );\n                return null;\n              })()}\n              {renderToggleButton && (\n                <RightColumn>\n                  <ToggleButtonWrapper\n                    css={[\n                      !shouldToggleButtonTakeSpace &&\n                        css`\n                          position: absolute;\n                          top: 0;\n                          right: 0;\n                          margin-top: ${customProperties.spacingXs};\n                        `,\n                    ]}\n                  >\n                    <FlatButton\n                      onClick={toggle}\n                      label={intl.formatMessage(\n                        isOpen\n                          ? messagesMultilineInput.collapse\n                          : messagesMultilineInput.expand\n                      )}\n                      icon={\n                        isOpen ? (\n                          <AngleUpIcon size=\"small\" />\n                        ) : (\n                          <AngleDownIcon size=\"small\" />\n                        )\n                      }\n                    />\n                  </ToggleButtonWrapper>\n                </RightColumn>\n              )}\n            </Row>\n          </Stack>\n        );\n      }}\n    </CollapsibleMotion>\n  );\n};\n\ntype TOptions = {\n  language: string;\n  error?: ReactNode;\n  warning?: ReactNode;\n  hasWarning?: boolean;\n  hasError?: boolean;\n  defaultExpandMultilineText: boolean;\n  toggleLanguage: (language: string) => void;\n  isOpen: boolean;\n  showExpandIcon: boolean;\n  onClickExpand?: () => boolean;\n  hasLanguagesControl?: boolean;\n};\n\nexport type TRenderEditorProps = {\n  id?: string;\n  name?: string;\n  disabled?: boolean;\n  readOnly?: boolean;\n  editor?: TEditor;\n  options: TOptions;\n};\nexport type TRenderEditor = (\n  props: TRenderEditorProps,\n  editor: TEditor,\n  next: () => ReactElement\n) => ReturnType<typeof Editor>;\n\nconst renderEditor: TRenderEditor = (props, editor, next) => {\n  if (props.options.showExpandIcon) {\n    warning(\n      typeof props.options.onClickExpand === 'function',\n      'renderEditor: \"onClickExpand\" is required when showExpandIcon is true'\n    );\n  }\n\n  const internalId = `${props.id}__internal__id`;\n\n  const children = cloneElement(next(), {\n    id: internalId,\n  });\n\n  const passedProps = {\n    id: props.id,\n    isDisabled: props.disabled,\n    isReadOnly: props.readOnly,\n    ...pick(props.options, [\n      'defaultExpandMultilineText',\n      'language',\n      'warning',\n      'error',\n      'hasWarning',\n      'hasError',\n      'toggleLanguage',\n      'isOpen',\n      'showExpandIcon',\n      'onClickExpand',\n      'hasLanguagesControl',\n    ]),\n    ...filterDataAttributes(props),\n  };\n\n  const isFocused = props.editor?.value.selection.isFocused;\n\n  return (\n    <Editor editor={editor} {...passedProps}>\n      {children}\n      <HiddenInput\n        isFocused={Boolean(isFocused)}\n        handleFocus={editor.focus}\n        disabled={props.disabled}\n        readOnly={props.readOnly}\n        id={props.id}\n      />\n    </Editor>\n  );\n};\n\nEditor.displayName = 'Editor';\n\nexport default renderEditor;\n"]} */"],
|
|
337
|
+
css: [!shouldToggleButtonTakeSpace && /*#__PURE__*/css("position:absolute;top:0;right:0;margin-top:", customProperties.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Editor;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["editor.tsx"],"names":[],"mappings":"AAiU2B","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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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":"AA+ToB","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 | Record<string, string>) => {\n      const newStringValue =\n        typeof newValue === 'string'\n          ? newValue\n          : newValue?.[props.language] ?? '';\n\n      resetEditor(editor, newStringValue);\n    },\n    [editor, props.language]\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"]} */"],
|
|
263
338
|
children: jsx(FlatButton, {
|
|
264
339
|
onClick: toggle,
|
|
265
340
|
label: intl.formatMessage(isOpen ? messagesMultilineInput.collapse : messagesMultilineInput.expand),
|
|
@@ -275,88 +350,17 @@ var Editor$1 = function Editor(props) {
|
|
|
275
350
|
});
|
|
276
351
|
}
|
|
277
352
|
});
|
|
278
|
-
};
|
|
279
|
-
|
|
280
|
-
Editor$1
|
|
281
|
-
children: _pt.node.isRequired,
|
|
282
|
-
id: _pt.string,
|
|
283
|
-
isOpen: _pt.bool.isRequired,
|
|
284
|
-
isDisabled: _pt.bool,
|
|
285
|
-
isReadOnly: _pt.bool,
|
|
286
|
-
hasWarning: _pt.bool,
|
|
287
|
-
hasError: _pt.bool,
|
|
288
|
-
editor: _pt.any,
|
|
289
|
-
error: _pt.node,
|
|
290
|
-
warning: _pt.node,
|
|
291
|
-
defaultExpandMultilineText: _pt.bool.isRequired,
|
|
292
|
-
toggleLanguage: _pt.func.isRequired,
|
|
293
|
-
language: _pt.string.isRequired,
|
|
294
|
-
showExpandIcon: _pt.bool.isRequired,
|
|
295
|
-
onClickExpand: _pt.func,
|
|
296
|
-
hasLanguagesControl: _pt.bool
|
|
297
|
-
} : {};
|
|
298
|
-
|
|
299
|
-
var renderEditor = function renderEditor(props, editor, next) {
|
|
300
|
-
var _props$editor5;
|
|
301
|
-
|
|
302
|
-
if (props.options.showExpandIcon) {
|
|
303
|
-
process.env.NODE_ENV !== "production" ? warning(typeof props.options.onClickExpand === 'function', 'renderEditor: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
var internalId = "".concat(props.id, "__internal__id");
|
|
307
|
-
var children = /*#__PURE__*/cloneElement(next(), {
|
|
308
|
-
id: internalId
|
|
309
|
-
});
|
|
310
|
-
|
|
311
|
-
var passedProps = _objectSpread$3(_objectSpread$3({
|
|
312
|
-
id: props.id,
|
|
313
|
-
isDisabled: props.disabled,
|
|
314
|
-
isReadOnly: props.readOnly
|
|
315
|
-
}, pick(props.options, ['defaultExpandMultilineText', 'language', 'warning', 'error', 'hasWarning', 'hasError', 'toggleLanguage', 'isOpen', 'showExpandIcon', 'onClickExpand', 'hasLanguagesControl'])), filterDataAttributes(props));
|
|
316
|
-
|
|
317
|
-
var isFocused = (_props$editor5 = props.editor) === null || _props$editor5 === void 0 ? void 0 : _props$editor5.value.selection.isFocused;
|
|
318
|
-
return jsxs(Editor$1, _objectSpread$3(_objectSpread$3({
|
|
319
|
-
editor: editor
|
|
320
|
-
}, passedProps), {}, {
|
|
321
|
-
children: [children, jsx(HiddenInput, {
|
|
322
|
-
isFocused: Boolean(isFocused),
|
|
323
|
-
handleFocus: editor.focus,
|
|
324
|
-
disabled: props.disabled,
|
|
325
|
-
readOnly: props.readOnly,
|
|
326
|
-
id: props.id
|
|
327
|
-
})]
|
|
328
|
-
}));
|
|
329
|
-
};
|
|
330
|
-
|
|
331
|
-
Editor$1.displayName = 'Editor';
|
|
332
|
-
var renderEditor$1 = renderEditor;
|
|
333
|
-
|
|
334
|
-
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = _Reflect$construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
|
|
335
|
-
|
|
336
|
-
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !_Reflect$construct) return false; if (_Reflect$construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(_Reflect$construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
|
|
353
|
+
});
|
|
354
|
+
Editor.displayName = 'Editor';
|
|
355
|
+
var Editor$1 = Editor;
|
|
337
356
|
|
|
338
357
|
function ownKeys$2(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
339
358
|
|
|
340
359
|
function _objectSpread$2(target) { for (var i = 1; i < arguments.length; i++) { var _context2, _context3; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty(_context2 = ownKeys$2(Object(source), !0)).call(_context2, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(target, _Object$getOwnPropertyDescriptors(source)) : _forEachInstanceProperty(_context3 = ownKeys$2(Object(source))).call(_context3, function (key) { _Object$defineProperty(target, key, _Object$getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
341
360
|
|
|
342
|
-
|
|
343
|
-
// TODO: remove after upgrade of `slate-react` to the latest version
|
|
344
|
-
var Editor = function Editor(props) {
|
|
345
|
-
return jsx(Editor$2, _objectSpread$2({}, props));
|
|
346
|
-
};
|
|
361
|
+
function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = _Reflect$construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }
|
|
347
362
|
|
|
348
|
-
|
|
349
|
-
id: _pt.string,
|
|
350
|
-
name: _pt.string,
|
|
351
|
-
onFocus: _pt.func,
|
|
352
|
-
onBlur: _pt.func,
|
|
353
|
-
disabled: _pt.bool,
|
|
354
|
-
readOnly: _pt.bool,
|
|
355
|
-
value: _pt.any.isRequired,
|
|
356
|
-
onChange: _pt.func,
|
|
357
|
-
plugins: _pt.any.isRequired,
|
|
358
|
-
renderEditor: _pt.any.isRequired
|
|
359
|
-
} : {};
|
|
363
|
+
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !_Reflect$construct) return false; if (_Reflect$construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(_Reflect$construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
|
|
360
364
|
|
|
361
365
|
var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
362
366
|
_inherits(RichTextInput, _PureComponent);
|
|
@@ -376,33 +380,25 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
376
380
|
|
|
377
381
|
_this = _super.call.apply(_super, _concatInstanceProperty(_context = [this]).call(_context, args));
|
|
378
382
|
_this.serializedValue = _this.props.value || '';
|
|
379
|
-
_this.internalSlateValue = html.deserialize(_this.props.value || '');
|
|
383
|
+
_this.internalSlateValue = validSlateStateAdapter(html.deserialize(_this.props.value || ''));
|
|
380
384
|
|
|
381
|
-
_this.onValueChange = function (
|
|
382
|
-
var serializedValue = html.serialize(
|
|
385
|
+
_this.onValueChange = function (state) {
|
|
386
|
+
var serializedValue = html.serialize(state); // because we are not using setState, we need to make sure that
|
|
383
387
|
// we perform an update when the slate value changes
|
|
384
388
|
// as this can contain things like cursor location
|
|
385
389
|
// in this case, the internalSlateValue would change
|
|
386
390
|
// but the serializedValue would NOT change.
|
|
387
391
|
|
|
388
|
-
var hasInternalSlateValueChanged = _this.internalSlateValue !==
|
|
392
|
+
var hasInternalSlateValueChanged = _this.internalSlateValue !== state;
|
|
389
393
|
var hasSerializedValueChanged = serializedValue !== _this.serializedValue;
|
|
390
|
-
_this.internalSlateValue =
|
|
394
|
+
_this.internalSlateValue = validSlateStateAdapter(state);
|
|
391
395
|
_this.serializedValue = serializedValue; // the consumer only cares about the serializedValue, so it doesn't make sense to call
|
|
392
396
|
// onChange unless this value changes.
|
|
393
397
|
|
|
394
398
|
if (hasSerializedValueChanged) {
|
|
395
399
|
var _this$props$onChange, _this$props;
|
|
396
400
|
|
|
397
|
-
|
|
398
|
-
target: {
|
|
399
|
-
id: _this.props.id,
|
|
400
|
-
name: _this.props.name,
|
|
401
|
-
language: _this.props.language,
|
|
402
|
-
value: serializedValue
|
|
403
|
-
}
|
|
404
|
-
};
|
|
405
|
-
(_this$props$onChange = (_this$props = _this.props).onChange) === null || _this$props$onChange === void 0 ? void 0 : _this$props$onChange.call(_this$props, fakeEvent);
|
|
401
|
+
(_this$props$onChange = (_this$props = _this.props).onChange) === null || _this$props$onChange === void 0 ? void 0 : _this$props$onChange.call(_this$props, html.serialize(state));
|
|
406
402
|
}
|
|
407
403
|
|
|
408
404
|
if (hasInternalSlateValueChanged && !hasSerializedValueChanged) {
|
|
@@ -411,44 +407,6 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
411
407
|
}
|
|
412
408
|
};
|
|
413
409
|
|
|
414
|
-
_this.onBlur = function (_event, _editor, next) {
|
|
415
|
-
next();
|
|
416
|
-
|
|
417
|
-
if (_this.props.onBlur) {
|
|
418
|
-
var fakeEvent = {
|
|
419
|
-
target: {
|
|
420
|
-
id: _this.props.id,
|
|
421
|
-
name: _this.props.name
|
|
422
|
-
}
|
|
423
|
-
};
|
|
424
|
-
|
|
425
|
-
_setTimeout(function () {
|
|
426
|
-
var _this$props$onBlur, _this$props2;
|
|
427
|
-
|
|
428
|
-
return (_this$props$onBlur = (_this$props2 = _this.props).onBlur) === null || _this$props$onBlur === void 0 ? void 0 : _this$props$onBlur.call(_this$props2, fakeEvent);
|
|
429
|
-
}, 0);
|
|
430
|
-
}
|
|
431
|
-
};
|
|
432
|
-
|
|
433
|
-
_this.onFocus = function (_event, _editor, next) {
|
|
434
|
-
next();
|
|
435
|
-
|
|
436
|
-
if (_this.props.onFocus) {
|
|
437
|
-
var fakeEvent = {
|
|
438
|
-
target: {
|
|
439
|
-
id: _this.props.id,
|
|
440
|
-
name: _this.props.name
|
|
441
|
-
}
|
|
442
|
-
};
|
|
443
|
-
|
|
444
|
-
_setTimeout(function () {
|
|
445
|
-
var _this$props$onFocus, _this$props3;
|
|
446
|
-
|
|
447
|
-
return (_this$props$onFocus = (_this$props3 = _this.props).onFocus) === null || _this$props$onFocus === void 0 ? void 0 : _this$props$onFocus.call(_this$props3, fakeEvent);
|
|
448
|
-
}, 0);
|
|
449
|
-
}
|
|
450
|
-
};
|
|
451
|
-
|
|
452
410
|
return _this;
|
|
453
411
|
}
|
|
454
412
|
|
|
@@ -456,7 +414,7 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
456
414
|
key: "componentDidUpdate",
|
|
457
415
|
value: function componentDidUpdate() {
|
|
458
416
|
if (this.props.value !== this.serializedValue) {
|
|
459
|
-
this.internalSlateValue = html.deserialize(this.props.value);
|
|
417
|
+
this.internalSlateValue = validSlateStateAdapter(html.deserialize(this.props.value));
|
|
460
418
|
this.serializedValue = this.props.value;
|
|
461
419
|
this.forceUpdate();
|
|
462
420
|
}
|
|
@@ -472,21 +430,28 @@ var RichTextInput = /*#__PURE__*/function (_PureComponent) {
|
|
|
472
430
|
process.env.NODE_ENV !== "production" ? warning(typeof this.props.onClickExpand === 'function', 'RichTextInput: "onClickExpand" is required when showExpandIcon is true') : void 0;
|
|
473
431
|
}
|
|
474
432
|
|
|
475
|
-
return jsx(Editor, _objectSpread$2(_objectSpread$2({}, filterDataAttributes(this.props)), {}, {
|
|
476
|
-
id: this.props.id,
|
|
433
|
+
return jsx(Editor$1, _objectSpread$2(_objectSpread$2({}, filterDataAttributes(this.props)), {}, {
|
|
477
434
|
name: this.props.name,
|
|
478
|
-
|
|
479
|
-
readOnly: this.props.isReadOnly || this.props.isDisabled,
|
|
480
|
-
value: this.internalSlateValue,
|
|
481
|
-
onFocus: this.onFocus,
|
|
482
|
-
onBlur: this.onBlur // we can only pass this.props to the Editor that Slate understands without getting
|
|
483
|
-
// warning in the console,
|
|
484
|
-
// so instead we pass our extra this.props through this `options` prop.
|
|
485
|
-
,
|
|
486
|
-
options: pick(this.props, ['language', 'onToggle', 'toggleLanguage', 'isOpen', 'warning', 'error', 'defaultExpandMultilineText', 'hasWarning', 'hasError', 'placeholder', 'onClickExpand', 'showExpandIcon', 'hasLanguagesControl']),
|
|
435
|
+
isReadOnly: this.props.isReadOnly || this.props.isDisabled,
|
|
487
436
|
onChange: this.onValueChange,
|
|
488
|
-
|
|
489
|
-
|
|
437
|
+
id: this.props.id,
|
|
438
|
+
value: this.internalSlateValue,
|
|
439
|
+
onFocus: this.props.onFocus,
|
|
440
|
+
onBlur: this.props.onBlur,
|
|
441
|
+
isDisabled: this.props.isDisabled,
|
|
442
|
+
defaultExpandMultilineText: this.props.defaultExpandMultilineText,
|
|
443
|
+
hasWarning: this.props.hasWarning,
|
|
444
|
+
hasError: this.props.hasError,
|
|
445
|
+
placeholder: this.props.placeholder,
|
|
446
|
+
showExpandIcon: this.props.showExpandIcon,
|
|
447
|
+
onClickExpand: this.props.onClickExpand,
|
|
448
|
+
language: this.props.language,
|
|
449
|
+
toggleLanguage: this.props.toggleLanguage,
|
|
450
|
+
isOpen: this.props.isOpen,
|
|
451
|
+
warning: this.props.warning,
|
|
452
|
+
error: this.props.error,
|
|
453
|
+
hasLanguagesControl: this.props.hasLanguagesControl,
|
|
454
|
+
ref: this.props.parentRef
|
|
490
455
|
}));
|
|
491
456
|
}
|
|
492
457
|
}]);
|
|
@@ -500,28 +465,23 @@ RichTextInput.defaultProps = {
|
|
|
500
465
|
};
|
|
501
466
|
RichTextInput.displayName = 'RichTextInput';
|
|
502
467
|
RichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
503
|
-
defaultExpandMultilineText: _pt.bool,
|
|
504
|
-
hasError: _pt.bool,
|
|
505
|
-
hasWarning: _pt.bool,
|
|
506
|
-
id: _pt.string,
|
|
507
468
|
name: _pt.string,
|
|
508
|
-
placeholder: _pt.string,
|
|
509
|
-
isDisabled: _pt.bool,
|
|
510
|
-
isReadOnly: _pt.bool,
|
|
511
469
|
onChange: _pt.func,
|
|
512
|
-
onBlur: _pt.func,
|
|
513
|
-
onFocus: _pt.func,
|
|
514
470
|
value: _pt.string.isRequired,
|
|
515
|
-
|
|
516
|
-
onClickExpand: _pt.func,
|
|
517
|
-
hasLanguagesControl: _pt.bool,
|
|
471
|
+
parentRef: _pt.any,
|
|
518
472
|
language: _pt.string.isRequired,
|
|
519
473
|
isOpen: _pt.bool.isRequired,
|
|
520
474
|
toggleLanguage: _pt.func.isRequired,
|
|
521
475
|
warning: _pt.node,
|
|
522
476
|
error: _pt.string
|
|
523
477
|
} : {};
|
|
524
|
-
var
|
|
478
|
+
var RichTextInputWithRef = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
479
|
+
return jsx(RichTextInput, _objectSpread$2({
|
|
480
|
+
parentRef: ref
|
|
481
|
+
}, props));
|
|
482
|
+
});
|
|
483
|
+
RichTextInputWithRef.displayName = 'RichTextInputWithRef';
|
|
484
|
+
var RichTextInput$1 = RichTextInputWithRef;
|
|
525
485
|
|
|
526
486
|
function ownKeys$1(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
527
487
|
|
|
@@ -538,7 +498,7 @@ var RequiredValueErrorMessage$1 = RequiredValueErrorMessage;
|
|
|
538
498
|
|
|
539
499
|
function ownKeys(object, enumerableOnly) { var keys = _Object$keys(object); if (_Object$getOwnPropertySymbols) { var symbols = _Object$getOwnPropertySymbols(object); enumerableOnly && (symbols = _filterInstanceProperty(symbols).call(symbols, function (sym) { return _Object$getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
540
500
|
|
|
541
|
-
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var
|
|
501
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var _context6, _context7; var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? _forEachInstanceProperty(_context6 = ownKeys(Object(source), !0)).call(_context6, function (key) { _defineProperty(target, key, source[key]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(target, _Object$getOwnPropertyDescriptors(source)) : _forEachInstanceProperty(_context7 = ownKeys(Object(source))).call(_context7, function (key) { _Object$defineProperty(target, key, _Object$getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
542
502
|
|
|
543
503
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
544
504
|
var defaultProps = {
|
|
@@ -579,11 +539,11 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
579
539
|
} : {
|
|
580
540
|
name: "9a4cy7-LocalizedRichTextInput",
|
|
581
541
|
styles: "align-self:flex-start;label:LocalizedRichTextInput;",
|
|
582
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-rich-text-input.tsx"],"names":[],"mappings":"AAqToB","file":"localized-rich-text-input.tsx","sourcesContent":["import {\n  useReducer,\n  useCallback,\n  type ReactNode,\n  type MouseEvent,\n  type KeyboardEvent,\n} from 'react';\nimport { css } from '@emotion/react';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport { useToggleState } from '@commercetools-uikit/hooks';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  isTouched,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { LocalizedInputToggle } from '@commercetools-uikit/input-utils';\nimport { localized } from '@commercetools-uikit/rich-text-utils';\nimport { warning } from '@commercetools-uikit/utils';\nimport RichTextInput from './rich-text-input';\nimport RequiredValueErrorMessage from './required-value-error-message';\n\ntype TErrors = Record<string, string>;\ntype TWarnings = Record<string, ReactNode>;\n\ntype 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"]} */",
|
|
542
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-rich-text-input.tsx"],"names":[],"mappings":"AAmXsB","file":"localized-rich-text-input.tsx","sourcesContent":["import {\n  useReducer,\n  useCallback,\n  forwardRef,\n  useRef,\n  useImperativeHandle,\n  type ReactNode,\n  type MouseEvent,\n  type KeyboardEvent,\n  type ForwardRefExoticComponent,\n  type RefAttributes,\n  type FocusEventHandler,\n  type MutableRefObject,\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\ntype RefWithImperativeResetHandler = MutableRefObject<unknown> & {\n  resetValue: (newValue: string | Record<string, string>) => void;\n};\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 ? `${props.id}.${language}` : '',\n            name: props?.name ? `${props.name}.${language}` : '',\n            language,\n            value: state,\n          },\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 langRefs = useRef<Map<string, RefWithImperativeResetHandler>>(\n      new Map()\n    );\n\n    const resetValue = useCallback(\n      (newValue: string | Record<string, string>) => {\n        langRefs.current.forEach((langRef) => {\n          langRef.resetValue(newValue);\n        });\n      },\n      []\n    );\n\n    useImperativeHandle(ref, () => {\n      return {\n        resetValue,\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={(el: RefWithImperativeResetHandler) =>\n                    langRefs.current.set(language, el)\n                  }\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"]} */",
|
|
583
543
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
584
544
|
};
|
|
585
545
|
|
|
586
|
-
var LocalizedRichTextInput = function
|
|
546
|
+
var LocalizedRichTextInput = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
587
547
|
var _context2;
|
|
588
548
|
|
|
589
549
|
if (!props.isReadOnly) {
|
|
@@ -607,8 +567,7 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
607
567
|
expandedTranslationsState = _useReducer2[0],
|
|
608
568
|
expandedTranslationsDispatch = _useReducer2[1];
|
|
609
569
|
|
|
610
|
-
var defaultExpansionState = props.hideLanguageExpansionControls || props.defaultExpandLanguages
|
|
611
|
-
false;
|
|
570
|
+
var defaultExpansionState = Boolean(props.hideLanguageExpansionControls || props.defaultExpandLanguages);
|
|
612
571
|
|
|
613
572
|
var _useToggleState = useToggleState(defaultExpansionState),
|
|
614
573
|
_useToggleState2 = _slicedToArray(_useToggleState, 2),
|
|
@@ -621,6 +580,21 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
621
580
|
payload: language
|
|
622
581
|
});
|
|
623
582
|
}, [expandedTranslationsDispatch]);
|
|
583
|
+
var createChangeHandler = useCallback(function (language) {
|
|
584
|
+
return function (state) {
|
|
585
|
+
var _props$onChange, _context3, _context4;
|
|
586
|
+
|
|
587
|
+
(_props$onChange = props.onChange) === null || _props$onChange === void 0 ? void 0 : _props$onChange.call(props, {
|
|
588
|
+
target: {
|
|
589
|
+
id: props !== null && props !== void 0 && props.id ? _concatInstanceProperty(_context3 = "".concat(props.id, ".")).call(_context3, language) : '',
|
|
590
|
+
name: props !== null && props !== void 0 && props.name ? _concatInstanceProperty(_context4 = "".concat(props.name, ".")).call(_context4, language) : '',
|
|
591
|
+
language: language,
|
|
592
|
+
value: state
|
|
593
|
+
}
|
|
594
|
+
});
|
|
595
|
+
};
|
|
596
|
+
}, // eslint-disable-next-line react-hooks/exhaustive-deps
|
|
597
|
+
[props.id, props.name, props.onChange]);
|
|
624
598
|
var languages = sortLanguages(props.selectedLanguage, _Object$keys(props.value));
|
|
625
599
|
var hasErrorInRemainingLanguages = props.hasError || getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);
|
|
626
600
|
var hasWarningInRemainingLanguages = props.hasWarning || getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);
|
|
@@ -633,6 +607,19 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
633
607
|
}
|
|
634
608
|
}
|
|
635
609
|
|
|
610
|
+
var langRefs = useRef(new _Map());
|
|
611
|
+
var resetValue = useCallback(function (newValue) {
|
|
612
|
+
var _context5;
|
|
613
|
+
|
|
614
|
+
_forEachInstanceProperty(_context5 = langRefs.current).call(_context5, function (langRef) {
|
|
615
|
+
langRef.resetValue(newValue);
|
|
616
|
+
});
|
|
617
|
+
}, []);
|
|
618
|
+
useImperativeHandle(ref, function () {
|
|
619
|
+
return {
|
|
620
|
+
resetValue: resetValue
|
|
621
|
+
};
|
|
622
|
+
});
|
|
636
623
|
var shouldRenderLanguagesControl = languages.length > 1 && !props.hideLanguageExpansionControls;
|
|
637
624
|
return jsx(Constraints.Horizontal, {
|
|
638
625
|
max: props.horizontalConstraint,
|
|
@@ -644,11 +631,12 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
644
631
|
if (!isFirstLanguage && !areLanguagesOpened) return null;
|
|
645
632
|
var isLastLanguage = index === languages.length - 1;
|
|
646
633
|
var hasLanguagesControl = isFirstLanguage && !areLanguagesOpened || isLastLanguage;
|
|
647
|
-
return
|
|
648
|
-
|
|
649
|
-
|
|
634
|
+
return createElement(RichTextInput$1, _objectSpread(_objectSpread({}, filterDataAttributes(props)), {}, {
|
|
635
|
+
key: language,
|
|
636
|
+
id: getId(props.id, language),
|
|
637
|
+
name: getName(props.name, language),
|
|
650
638
|
value: props.value[language],
|
|
651
|
-
onChange:
|
|
639
|
+
onChange: createChangeHandler(language),
|
|
652
640
|
language: language,
|
|
653
641
|
isOpen: expandedTranslationsState[language],
|
|
654
642
|
toggleLanguage: toggleLanguage,
|
|
@@ -663,8 +651,12 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
663
651
|
error: props.errors && props.errors[language],
|
|
664
652
|
showExpandIcon: props.showExpandIcon,
|
|
665
653
|
onClickExpand: props.onClickExpand,
|
|
666
|
-
hasLanguagesControl: hasLanguagesControl
|
|
667
|
-
|
|
654
|
+
hasLanguagesControl: hasLanguagesControl,
|
|
655
|
+
defaultExpandMultilineText: Boolean(props.defaultExpandMultilineText),
|
|
656
|
+
ref: function ref(el) {
|
|
657
|
+
return langRefs.current.set(language, el);
|
|
658
|
+
}
|
|
659
|
+
}, createLocalizedDataAttributes(props, language)));
|
|
668
660
|
})
|
|
669
661
|
}), shouldRenderLanguagesControl && jsx("div", {
|
|
670
662
|
css: _ref,
|
|
@@ -677,30 +669,7 @@ var LocalizedRichTextInput = function LocalizedRichTextInput(props) {
|
|
|
677
669
|
})]
|
|
678
670
|
})
|
|
679
671
|
});
|
|
680
|
-
};
|
|
681
|
-
|
|
682
|
-
LocalizedRichTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
683
|
-
id: _pt.string,
|
|
684
|
-
name: _pt.string,
|
|
685
|
-
value: _pt.objectOf(_pt.string).isRequired,
|
|
686
|
-
onChange: _pt.func,
|
|
687
|
-
selectedLanguage: _pt.string.isRequired,
|
|
688
|
-
onBlur: _pt.func,
|
|
689
|
-
onFocus: _pt.func,
|
|
690
|
-
defaultExpandMultilineText: _pt.bool,
|
|
691
|
-
hideLanguageExpansionControls: _pt.bool,
|
|
692
|
-
defaultExpandLanguages: _pt.bool,
|
|
693
|
-
isDisabled: _pt.bool,
|
|
694
|
-
isReadOnly: _pt.bool,
|
|
695
|
-
placeholder: _pt.objectOf(_pt.string),
|
|
696
|
-
horizontalConstraint: _pt.oneOf([7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
697
|
-
hasError: _pt.bool,
|
|
698
|
-
hasWarning: _pt.bool,
|
|
699
|
-
errors: _pt.objectOf(_pt.string),
|
|
700
|
-
warnings: _pt.objectOf(_pt.node),
|
|
701
|
-
showExpandIcon: _pt.bool.isRequired,
|
|
702
|
-
onClickExpand: _pt.func
|
|
703
|
-
} : {};
|
|
672
|
+
});
|
|
704
673
|
LocalizedRichTextInput.displayName = 'LocalizedRichTextInput';
|
|
705
674
|
LocalizedRichTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage$1;
|
|
706
675
|
LocalizedRichTextInput.getId = getId;
|
|
@@ -713,6 +682,6 @@ LocalizedRichTextInput.isTouched = isTouched;
|
|
|
713
682
|
var LocalizedRichTextInput$1 = LocalizedRichTextInput;
|
|
714
683
|
|
|
715
684
|
// NOTE: This string will be replaced on build time with the package version.
|
|
716
|
-
var version = "
|
|
685
|
+
var version = "15.0.1";
|
|
717
686
|
|
|
718
687
|
export { LocalizedRichTextInput$1 as default, version };
|