decap-cms-widget-markdown 3.1.6 → 3.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +6 -0
- package/dist/decap-cms-widget-markdown.js +4 -4
- package/dist/decap-cms-widget-markdown.js.map +1 -1
- package/dist/esm/MarkdownControl/VisualEditor.js +4 -3
- package/dist/esm/MarkdownControl/plugins/html/withHtml.js +140 -0
- package/dist/esm/MarkdownControl/renderers.js +18 -18
- package/dist/esm/serializers/slateRemark.js +1 -1
- package/package.json +2 -2
- package/src/MarkdownControl/VisualEditor.js +4 -1
- package/src/MarkdownControl/plugins/html/withHtml.js +98 -0
- package/src/MarkdownControl/renderers.js +2 -2
- package/src/serializers/slateRemark.js +1 -1
- package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorAfterSoftBreak.js +0 -16
- package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorInNonDefaultBlock copy.js +0 -19
|
@@ -31,6 +31,7 @@ var _serializers = require("../serializers");
|
|
|
31
31
|
var _withShortcodes = _interopRequireDefault(require("./plugins/shortcodes/withShortcodes"));
|
|
32
32
|
var _insertShortcode = _interopRequireDefault(require("./plugins/shortcodes/insertShortcode"));
|
|
33
33
|
var _defaultEmptyBlock = _interopRequireDefault(require("./plugins/blocks/defaultEmptyBlock"));
|
|
34
|
+
var _withHtml = _interopRequireDefault(require("./plugins/html/withHtml"));
|
|
34
35
|
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
|
|
35
36
|
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && Object.prototype.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
|
|
36
37
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
@@ -63,7 +64,7 @@ const InsertionPoint = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
63
64
|
} : {
|
|
64
65
|
name: "ma7xbc",
|
|
65
66
|
styles: "flex:1 1 auto;cursor:text",
|
|
66
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/MarkdownControl/VisualEditor.js"],"names":[],"mappings":"AA8CiC","file":"../../../src/MarkdownControl/VisualEditor.js","sourcesContent":["// @refresh reset\nimport React, { useCallback, useEffect, useMemo, useState } from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { ClassNames, css as coreCss } from '@emotion/react';\nimport { lengths, fonts, zIndex } from 'decap-cms-ui-default';\nimport styled from '@emotion/styled';\nimport { createEditor, Transforms, Editor as SlateEditor } from 'slate';\nimport { Editable, ReactEditor, Slate, withReact } from 'slate-react';\nimport { withHistory } from 'slate-history';\nimport { fromJS } from 'immutable';\nimport { isEqual } from 'lodash';\n\nimport { editorStyleVars, EditorControlBar } from '../styles';\nimport Toolbar from './Toolbar';\nimport { Element, Leaf } from './renderers';\nimport withLists from './plugins/lists/withLists';\nimport withBlocks from './plugins/blocks/withBlocks';\nimport withInlines from './plugins/inlines/withInlines';\nimport toggleMark from './plugins/inlines/events/toggleMark';\nimport toggleLink from './plugins/inlines/events/toggleLink';\nimport getActiveLink from './plugins/inlines/selectors/getActiveLink';\nimport isMarkActive from './plugins/inlines/locations/isMarkActive';\nimport isCursorInBlockType from './plugins/blocks/locations/isCursorInBlockType';\nimport { markdownToSlate, slateToMarkdown } from '../serializers';\nimport withShortcodes from './plugins/shortcodes/withShortcodes';\nimport insertShortcode from './plugins/shortcodes/insertShortcode';\nimport defaultEmptyBlock from './plugins/blocks/defaultEmptyBlock';\n\nfunction visualEditorStyles({ minimal }) {\n  return `\n  position: relative;\n  overflow: auto;\n  font-family: ${fonts.primary};\n  min-height: ${minimal ? 'auto' : lengths.richTextEditorMinHeight};\n  border-top-left-radius: 0;\n  border-top-right-radius: 0;\n  border-top: 0;\n  margin-top: -${editorStyleVars.stickyDistanceBottom};\n  padding: 0;\n  display: flex;\n  flex-direction: column;\n  z-index: ${zIndex.zIndex100};\n`;\n}\n\nconst InsertionPoint = styled.div`\n  flex: 1 1 auto;\n  cursor: text;\n`;\n\nexport function mergeMediaConfig(editorComponents, field) {\n  // merge editor media library config to image components\n  if (editorComponents.has('image')) {\n    const imageComponent = editorComponents.get('image');\n    const fields = imageComponent?.fields;\n\n    if (fields) {\n      imageComponent.fields = fields.update(\n        fields.findIndex(f => f.get('widget') === 'image'),\n        f => {\n          // merge `media_library` config\n          if (field.has('media_library')) {\n            f = f.set(\n              'media_library',\n              field.get('media_library').mergeDeep(f.get('media_library')),\n            );\n          }\n          // merge 'media_folder'\n          if (field.has('media_folder') && !f.has('media_folder')) {\n            f = f.set('media_folder', field.get('media_folder'));\n          }\n          // merge 'public_folder'\n          if (field.has('public_folder') && !f.has('public_folder')) {\n            f = f.set('public_folder', field.get('public_folder'));\n          }\n          return f;\n        },\n      );\n    }\n  }\n}\n\nfunction Editor(props) {\n  const {\n    onAddAsset,\n    getAsset,\n    className,\n    field,\n    isShowModeToggle,\n    t,\n    isDisabled,\n    getEditorComponents,\n    getRemarkPlugins,\n    onChange,\n  } = props;\n\n  const editor = useMemo(\n    () =>\n      withReact(withHistory(withShortcodes(withBlocks(withLists(withInlines(createEditor())))))),\n    [],\n  );\n\n  const emptyValue = [defaultEmptyBlock()];\n  let editorComponents = getEditorComponents();\n  const codeBlockComponent = fromJS(editorComponents.find(({ type }) => type === 'code-block'));\n\n  editorComponents =\n    codeBlockComponent || editorComponents.has('code-block')\n      ? editorComponents\n      : editorComponents.set('code-block', { label: 'Code Block', type: 'code-block' });\n\n  mergeMediaConfig(editorComponents, field);\n\n  const [editorValue, setEditorValue] = useState(\n    props.value\n      ? markdownToSlate(props.value, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        })\n      : emptyValue,\n  );\n\n  const renderElement = useCallback(\n    props => (\n      <Element {...props} classNameWrapper={className} codeBlockComponent={codeBlockComponent} />\n    ),\n    [],\n  );\n  const renderLeaf = useCallback(props => <Leaf {...props} />, []);\n\n  useEffect(() => {\n    if (props.pendingFocus) {\n      ReactEditor.focus(editor);\n    }\n  }, []);\n\n  function handleMarkClick(format) {\n    ReactEditor.focus(editor);\n    toggleMark(editor, format);\n  }\n\n  function handleBlockClick(format) {\n    ReactEditor.focus(editor);\n    if (format.endsWith('-list')) {\n      editor.toggleList(format);\n    } else {\n      editor.toggleBlock(format);\n    }\n  }\n\n  function handleLinkClick() {\n    toggleLink(editor, t('editor.editorWidgets.markdown.linkPrompt'));\n    ReactEditor.focus(editor);\n  }\n\n  function handleToggleMode() {\n    props.onMode('raw');\n  }\n\n  function handleInsertShortcode(pluginConfig) {\n    insertShortcode(editor, pluginConfig);\n  }\n\n  function handleKeyDown(event) {\n    for (const handler of editor.keyDownHandlers || []) {\n      if (handler(event, editor) === false) {\n        break;\n      }\n    }\n    ReactEditor.focus(editor);\n  }\n\n  function handleClickBelowDocument() {\n    ReactEditor.focus(editor);\n    Transforms.select(editor, { path: [0, 0], offset: 0 });\n    Transforms.select(editor, SlateEditor.end(editor, []));\n  }\n  const [toolbarKey, setToolbarKey] = useState(0);\n\n  function handleChange(newValue) {\n    if (!isEqual(newValue, editorValue)) {\n      setEditorValue(() => newValue);\n      onChange(\n        slateToMarkdown(newValue, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        }),\n      );\n    }\n    setToolbarKey(prev => prev + 1);\n  }\n\n  function hasMark(format) {\n    return isMarkActive(editor, format);\n  }\n\n  function hasInline(format) {\n    if (format == 'link') {\n      return !!getActiveLink(editor);\n    }\n    return false;\n  }\n\n  function hasBlock(format) {\n    return isCursorInBlockType(editor, format);\n  }\n  function hasQuote() {\n    return isCursorInBlockType(editor, 'quote');\n  }\n  function hasListItems(type) {\n    return isCursorInBlockType(editor, type);\n  }\n\n  return (\n    <div\n      css={coreCss`\n        position: relative;\n      `}\n    >\n      <Slate editor={editor} value={editorValue} onChange={handleChange}>\n        <EditorControlBar>\n          {\n            <Toolbar\n              key={toolbarKey}\n              onMarkClick={handleMarkClick}\n              onBlockClick={handleBlockClick}\n              onLinkClick={handleLinkClick}\n              onToggleMode={handleToggleMode}\n              plugins={editorComponents}\n              onSubmit={handleInsertShortcode}\n              onAddAsset={onAddAsset}\n              getAsset={getAsset}\n              buttons={field.get('buttons')}\n              editorComponents={field.get('editor_components')}\n              hasMark={hasMark}\n              hasInline={hasInline}\n              hasBlock={hasBlock}\n              hasQuote={hasQuote}\n              hasListItems={hasListItems}\n              isShowModeToggle={isShowModeToggle}\n              t={t}\n              disabled={isDisabled}\n            />\n          }\n        </EditorControlBar>\n        {\n          <ClassNames>\n            {({ css, cx }) => (\n              <div\n                className={cx(\n                  className,\n                  css`\n                    ${visualEditorStyles({ minimal: field.get('minimal') })}\n                  `,\n                )}\n              >\n                {editorValue.length !== 0 && (\n                  <Editable\n                    className={css`\n                      padding: 16px 20px 0;\n                    `}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    onKeyDown={handleKeyDown}\n                    autoFocus={false}\n                  />\n                )}\n                <InsertionPoint onClick={handleClickBelowDocument} />\n              </div>\n            )}\n          </ClassNames>\n        }\n      </Slate>\n    </div>\n  );\n}\n\nEditor.propTypes = {\n  onAddAsset: PropTypes.func.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onChange: PropTypes.func.isRequired,\n  onMode: PropTypes.func.isRequired,\n  className: PropTypes.string.isRequired,\n  value: PropTypes.string,\n  field: ImmutablePropTypes.map.isRequired,\n  getEditorComponents: PropTypes.func.isRequired,\n  getRemarkPlugins: PropTypes.func.isRequired,\n  isShowModeToggle: PropTypes.bool.isRequired,\n  t: PropTypes.func.isRequired,\n};\n\nexport default Editor;\n"]} */",
|
|
67
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/MarkdownControl/VisualEditor.js"],"names":[],"mappings":"AA+CiC","file":"../../../src/MarkdownControl/VisualEditor.js","sourcesContent":["// @refresh reset\nimport React, { useCallback, useEffect, useMemo, useState } from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { ClassNames, css as coreCss } from '@emotion/react';\nimport { lengths, fonts, zIndex } from 'decap-cms-ui-default';\nimport styled from '@emotion/styled';\nimport { createEditor, Transforms, Editor as SlateEditor } from 'slate';\nimport { Editable, ReactEditor, Slate, withReact } from 'slate-react';\nimport { withHistory } from 'slate-history';\nimport { fromJS } from 'immutable';\nimport { isEqual } from 'lodash';\n\nimport { editorStyleVars, EditorControlBar } from '../styles';\nimport Toolbar from './Toolbar';\nimport { Element, Leaf } from './renderers';\nimport withLists from './plugins/lists/withLists';\nimport withBlocks from './plugins/blocks/withBlocks';\nimport withInlines from './plugins/inlines/withInlines';\nimport toggleMark from './plugins/inlines/events/toggleMark';\nimport toggleLink from './plugins/inlines/events/toggleLink';\nimport getActiveLink from './plugins/inlines/selectors/getActiveLink';\nimport isMarkActive from './plugins/inlines/locations/isMarkActive';\nimport isCursorInBlockType from './plugins/blocks/locations/isCursorInBlockType';\nimport { markdownToSlate, slateToMarkdown } from '../serializers';\nimport withShortcodes from './plugins/shortcodes/withShortcodes';\nimport insertShortcode from './plugins/shortcodes/insertShortcode';\nimport defaultEmptyBlock from './plugins/blocks/defaultEmptyBlock';\nimport withHtml from './plugins/html/withHtml';\n\nfunction visualEditorStyles({ minimal }) {\n  return `\n  position: relative;\n  overflow: auto;\n  font-family: ${fonts.primary};\n  min-height: ${minimal ? 'auto' : lengths.richTextEditorMinHeight};\n  border-top-left-radius: 0;\n  border-top-right-radius: 0;\n  border-top: 0;\n  margin-top: -${editorStyleVars.stickyDistanceBottom};\n  padding: 0;\n  display: flex;\n  flex-direction: column;\n  z-index: ${zIndex.zIndex100};\n`;\n}\n\nconst InsertionPoint = styled.div`\n  flex: 1 1 auto;\n  cursor: text;\n`;\n\nexport function mergeMediaConfig(editorComponents, field) {\n  // merge editor media library config to image components\n  if (editorComponents.has('image')) {\n    const imageComponent = editorComponents.get('image');\n    const fields = imageComponent?.fields;\n\n    if (fields) {\n      imageComponent.fields = fields.update(\n        fields.findIndex(f => f.get('widget') === 'image'),\n        f => {\n          // merge `media_library` config\n          if (field.has('media_library')) {\n            f = f.set(\n              'media_library',\n              field.get('media_library').mergeDeep(f.get('media_library')),\n            );\n          }\n          // merge 'media_folder'\n          if (field.has('media_folder') && !f.has('media_folder')) {\n            f = f.set('media_folder', field.get('media_folder'));\n          }\n          // merge 'public_folder'\n          if (field.has('public_folder') && !f.has('public_folder')) {\n            f = f.set('public_folder', field.get('public_folder'));\n          }\n          return f;\n        },\n      );\n    }\n  }\n}\n\nfunction Editor(props) {\n  const {\n    onAddAsset,\n    getAsset,\n    className,\n    field,\n    isShowModeToggle,\n    t,\n    isDisabled,\n    getEditorComponents,\n    getRemarkPlugins,\n    onChange,\n  } = props;\n\n  const editor = useMemo(\n    () =>\n      withHtml(\n        withReact(withHistory(withShortcodes(withBlocks(withLists(withInlines(createEditor())))))),\n      ),\n    [],\n  );\n\n  const emptyValue = [defaultEmptyBlock()];\n  let editorComponents = getEditorComponents();\n  const codeBlockComponent = fromJS(editorComponents.find(({ type }) => type === 'code-block'));\n\n  editorComponents =\n    codeBlockComponent || editorComponents.has('code-block')\n      ? editorComponents\n      : editorComponents.set('code-block', { label: 'Code Block', type: 'code-block' });\n\n  mergeMediaConfig(editorComponents, field);\n\n  const [editorValue, setEditorValue] = useState(\n    props.value\n      ? markdownToSlate(props.value, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        })\n      : emptyValue,\n  );\n\n  const renderElement = useCallback(\n    props => (\n      <Element {...props} classNameWrapper={className} codeBlockComponent={codeBlockComponent} />\n    ),\n    [],\n  );\n  const renderLeaf = useCallback(props => <Leaf {...props} />, []);\n\n  useEffect(() => {\n    if (props.pendingFocus) {\n      ReactEditor.focus(editor);\n    }\n  }, []);\n\n  function handleMarkClick(format) {\n    ReactEditor.focus(editor);\n    toggleMark(editor, format);\n  }\n\n  function handleBlockClick(format) {\n    ReactEditor.focus(editor);\n    if (format.endsWith('-list')) {\n      editor.toggleList(format);\n    } else {\n      editor.toggleBlock(format);\n    }\n  }\n\n  function handleLinkClick() {\n    toggleLink(editor, t('editor.editorWidgets.markdown.linkPrompt'));\n    ReactEditor.focus(editor);\n  }\n\n  function handleToggleMode() {\n    props.onMode('raw');\n  }\n\n  function handleInsertShortcode(pluginConfig) {\n    insertShortcode(editor, pluginConfig);\n  }\n\n  function handleKeyDown(event) {\n    for (const handler of editor.keyDownHandlers || []) {\n      if (handler(event, editor) === false) {\n        break;\n      }\n    }\n    ReactEditor.focus(editor);\n  }\n\n  function handleClickBelowDocument() {\n    ReactEditor.focus(editor);\n    Transforms.select(editor, { path: [0, 0], offset: 0 });\n    Transforms.select(editor, SlateEditor.end(editor, []));\n  }\n  const [toolbarKey, setToolbarKey] = useState(0);\n\n  function handleChange(newValue) {\n    if (!isEqual(newValue, editorValue)) {\n      setEditorValue(() => newValue);\n      onChange(\n        slateToMarkdown(newValue, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        }),\n      );\n    }\n    setToolbarKey(prev => prev + 1);\n  }\n\n  function hasMark(format) {\n    return isMarkActive(editor, format);\n  }\n\n  function hasInline(format) {\n    if (format == 'link') {\n      return !!getActiveLink(editor);\n    }\n    return false;\n  }\n\n  function hasBlock(format) {\n    return isCursorInBlockType(editor, format);\n  }\n  function hasQuote() {\n    return isCursorInBlockType(editor, 'quote');\n  }\n  function hasListItems(type) {\n    return isCursorInBlockType(editor, type);\n  }\n\n  return (\n    <div\n      css={coreCss`\n        position: relative;\n      `}\n    >\n      <Slate editor={editor} value={editorValue} onChange={handleChange}>\n        <EditorControlBar>\n          {\n            <Toolbar\n              key={toolbarKey}\n              onMarkClick={handleMarkClick}\n              onBlockClick={handleBlockClick}\n              onLinkClick={handleLinkClick}\n              onToggleMode={handleToggleMode}\n              plugins={editorComponents}\n              onSubmit={handleInsertShortcode}\n              onAddAsset={onAddAsset}\n              getAsset={getAsset}\n              buttons={field.get('buttons')}\n              editorComponents={field.get('editor_components')}\n              hasMark={hasMark}\n              hasInline={hasInline}\n              hasBlock={hasBlock}\n              hasQuote={hasQuote}\n              hasListItems={hasListItems}\n              isShowModeToggle={isShowModeToggle}\n              t={t}\n              disabled={isDisabled}\n            />\n          }\n        </EditorControlBar>\n        {\n          <ClassNames>\n            {({ css, cx }) => (\n              <div\n                className={cx(\n                  className,\n                  css`\n                    ${visualEditorStyles({ minimal: field.get('minimal') })}\n                  `,\n                )}\n              >\n                {editorValue.length !== 0 && (\n                  <Editable\n                    className={css`\n                      padding: 16px 20px 0;\n                    `}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    onKeyDown={handleKeyDown}\n                    autoFocus={false}\n                  />\n                )}\n                <InsertionPoint onClick={handleClickBelowDocument} />\n              </div>\n            )}\n          </ClassNames>\n        }\n      </Slate>\n    </div>\n  );\n}\n\nEditor.propTypes = {\n  onAddAsset: PropTypes.func.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onChange: PropTypes.func.isRequired,\n  onMode: PropTypes.func.isRequired,\n  className: PropTypes.string.isRequired,\n  value: PropTypes.string,\n  field: ImmutablePropTypes.map.isRequired,\n  getEditorComponents: PropTypes.func.isRequired,\n  getRemarkPlugins: PropTypes.func.isRequired,\n  isShowModeToggle: PropTypes.bool.isRequired,\n  t: PropTypes.func.isRequired,\n};\n\nexport default Editor;\n"]} */",
|
|
67
68
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
68
69
|
});
|
|
69
70
|
function mergeMediaConfig(editorComponents, field) {
|
|
@@ -96,7 +97,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
96
97
|
} : {
|
|
97
98
|
name: "j8ayvn-Editor",
|
|
98
99
|
styles: "position:relative;label:Editor;",
|
|
99
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/MarkdownControl/VisualEditor.js"],"names":[],"mappings":"AAwNkB","file":"../../../src/MarkdownControl/VisualEditor.js","sourcesContent":["// @refresh reset\nimport React, { useCallback, useEffect, useMemo, useState } from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { ClassNames, css as coreCss } from '@emotion/react';\nimport { lengths, fonts, zIndex } from 'decap-cms-ui-default';\nimport styled from '@emotion/styled';\nimport { createEditor, Transforms, Editor as SlateEditor } from 'slate';\nimport { Editable, ReactEditor, Slate, withReact } from 'slate-react';\nimport { withHistory } from 'slate-history';\nimport { fromJS } from 'immutable';\nimport { isEqual } from 'lodash';\n\nimport { editorStyleVars, EditorControlBar } from '../styles';\nimport Toolbar from './Toolbar';\nimport { Element, Leaf } from './renderers';\nimport withLists from './plugins/lists/withLists';\nimport withBlocks from './plugins/blocks/withBlocks';\nimport withInlines from './plugins/inlines/withInlines';\nimport toggleMark from './plugins/inlines/events/toggleMark';\nimport toggleLink from './plugins/inlines/events/toggleLink';\nimport getActiveLink from './plugins/inlines/selectors/getActiveLink';\nimport isMarkActive from './plugins/inlines/locations/isMarkActive';\nimport isCursorInBlockType from './plugins/blocks/locations/isCursorInBlockType';\nimport { markdownToSlate, slateToMarkdown } from '../serializers';\nimport withShortcodes from './plugins/shortcodes/withShortcodes';\nimport insertShortcode from './plugins/shortcodes/insertShortcode';\nimport defaultEmptyBlock from './plugins/blocks/defaultEmptyBlock';\n\nfunction visualEditorStyles({ minimal }) {\n  return `\n  position: relative;\n  overflow: auto;\n  font-family: ${fonts.primary};\n  min-height: ${minimal ? 'auto' : lengths.richTextEditorMinHeight};\n  border-top-left-radius: 0;\n  border-top-right-radius: 0;\n  border-top: 0;\n  margin-top: -${editorStyleVars.stickyDistanceBottom};\n  padding: 0;\n  display: flex;\n  flex-direction: column;\n  z-index: ${zIndex.zIndex100};\n`;\n}\n\nconst InsertionPoint = styled.div`\n  flex: 1 1 auto;\n  cursor: text;\n`;\n\nexport function mergeMediaConfig(editorComponents, field) {\n  // merge editor media library config to image components\n  if (editorComponents.has('image')) {\n    const imageComponent = editorComponents.get('image');\n    const fields = imageComponent?.fields;\n\n    if (fields) {\n      imageComponent.fields = fields.update(\n        fields.findIndex(f => f.get('widget') === 'image'),\n        f => {\n          // merge `media_library` config\n          if (field.has('media_library')) {\n            f = f.set(\n              'media_library',\n              field.get('media_library').mergeDeep(f.get('media_library')),\n            );\n          }\n          // merge 'media_folder'\n          if (field.has('media_folder') && !f.has('media_folder')) {\n            f = f.set('media_folder', field.get('media_folder'));\n          }\n          // merge 'public_folder'\n          if (field.has('public_folder') && !f.has('public_folder')) {\n            f = f.set('public_folder', field.get('public_folder'));\n          }\n          return f;\n        },\n      );\n    }\n  }\n}\n\nfunction Editor(props) {\n  const {\n    onAddAsset,\n    getAsset,\n    className,\n    field,\n    isShowModeToggle,\n    t,\n    isDisabled,\n    getEditorComponents,\n    getRemarkPlugins,\n    onChange,\n  } = props;\n\n  const editor = useMemo(\n    () =>\n      withReact(withHistory(withShortcodes(withBlocks(withLists(withInlines(createEditor())))))),\n    [],\n  );\n\n  const emptyValue = [defaultEmptyBlock()];\n  let editorComponents = getEditorComponents();\n  const codeBlockComponent = fromJS(editorComponents.find(({ type }) => type === 'code-block'));\n\n  editorComponents =\n    codeBlockComponent || editorComponents.has('code-block')\n      ? editorComponents\n      : editorComponents.set('code-block', { label: 'Code Block', type: 'code-block' });\n\n  mergeMediaConfig(editorComponents, field);\n\n  const [editorValue, setEditorValue] = useState(\n    props.value\n      ? markdownToSlate(props.value, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        })\n      : emptyValue,\n  );\n\n  const renderElement = useCallback(\n    props => (\n      <Element {...props} classNameWrapper={className} codeBlockComponent={codeBlockComponent} />\n    ),\n    [],\n  );\n  const renderLeaf = useCallback(props => <Leaf {...props} />, []);\n\n  useEffect(() => {\n    if (props.pendingFocus) {\n      ReactEditor.focus(editor);\n    }\n  }, []);\n\n  function handleMarkClick(format) {\n    ReactEditor.focus(editor);\n    toggleMark(editor, format);\n  }\n\n  function handleBlockClick(format) {\n    ReactEditor.focus(editor);\n    if (format.endsWith('-list')) {\n      editor.toggleList(format);\n    } else {\n      editor.toggleBlock(format);\n    }\n  }\n\n  function handleLinkClick() {\n    toggleLink(editor, t('editor.editorWidgets.markdown.linkPrompt'));\n    ReactEditor.focus(editor);\n  }\n\n  function handleToggleMode() {\n    props.onMode('raw');\n  }\n\n  function handleInsertShortcode(pluginConfig) {\n    insertShortcode(editor, pluginConfig);\n  }\n\n  function handleKeyDown(event) {\n    for (const handler of editor.keyDownHandlers || []) {\n      if (handler(event, editor) === false) {\n        break;\n      }\n    }\n    ReactEditor.focus(editor);\n  }\n\n  function handleClickBelowDocument() {\n    ReactEditor.focus(editor);\n    Transforms.select(editor, { path: [0, 0], offset: 0 });\n    Transforms.select(editor, SlateEditor.end(editor, []));\n  }\n  const [toolbarKey, setToolbarKey] = useState(0);\n\n  function handleChange(newValue) {\n    if (!isEqual(newValue, editorValue)) {\n      setEditorValue(() => newValue);\n      onChange(\n        slateToMarkdown(newValue, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        }),\n      );\n    }\n    setToolbarKey(prev => prev + 1);\n  }\n\n  function hasMark(format) {\n    return isMarkActive(editor, format);\n  }\n\n  function hasInline(format) {\n    if (format == 'link') {\n      return !!getActiveLink(editor);\n    }\n    return false;\n  }\n\n  function hasBlock(format) {\n    return isCursorInBlockType(editor, format);\n  }\n  function hasQuote() {\n    return isCursorInBlockType(editor, 'quote');\n  }\n  function hasListItems(type) {\n    return isCursorInBlockType(editor, type);\n  }\n\n  return (\n    <div\n      css={coreCss`\n        position: relative;\n      `}\n    >\n      <Slate editor={editor} value={editorValue} onChange={handleChange}>\n        <EditorControlBar>\n          {\n            <Toolbar\n              key={toolbarKey}\n              onMarkClick={handleMarkClick}\n              onBlockClick={handleBlockClick}\n              onLinkClick={handleLinkClick}\n              onToggleMode={handleToggleMode}\n              plugins={editorComponents}\n              onSubmit={handleInsertShortcode}\n              onAddAsset={onAddAsset}\n              getAsset={getAsset}\n              buttons={field.get('buttons')}\n              editorComponents={field.get('editor_components')}\n              hasMark={hasMark}\n              hasInline={hasInline}\n              hasBlock={hasBlock}\n              hasQuote={hasQuote}\n              hasListItems={hasListItems}\n              isShowModeToggle={isShowModeToggle}\n              t={t}\n              disabled={isDisabled}\n            />\n          }\n        </EditorControlBar>\n        {\n          <ClassNames>\n            {({ css, cx }) => (\n              <div\n                className={cx(\n                  className,\n                  css`\n                    ${visualEditorStyles({ minimal: field.get('minimal') })}\n                  `,\n                )}\n              >\n                {editorValue.length !== 0 && (\n                  <Editable\n                    className={css`\n                      padding: 16px 20px 0;\n                    `}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    onKeyDown={handleKeyDown}\n                    autoFocus={false}\n                  />\n                )}\n                <InsertionPoint onClick={handleClickBelowDocument} />\n              </div>\n            )}\n          </ClassNames>\n        }\n      </Slate>\n    </div>\n  );\n}\n\nEditor.propTypes = {\n  onAddAsset: PropTypes.func.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onChange: PropTypes.func.isRequired,\n  onMode: PropTypes.func.isRequired,\n  className: PropTypes.string.isRequired,\n  value: PropTypes.string,\n  field: ImmutablePropTypes.map.isRequired,\n  getEditorComponents: PropTypes.func.isRequired,\n  getRemarkPlugins: PropTypes.func.isRequired,\n  isShowModeToggle: PropTypes.bool.isRequired,\n  t: PropTypes.func.isRequired,\n};\n\nexport default Editor;\n"]} */",
|
|
100
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/MarkdownControl/VisualEditor.js"],"names":[],"mappings":"AA2NkB","file":"../../../src/MarkdownControl/VisualEditor.js","sourcesContent":["// @refresh reset\nimport React, { useCallback, useEffect, useMemo, useState } from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { ClassNames, css as coreCss } from '@emotion/react';\nimport { lengths, fonts, zIndex } from 'decap-cms-ui-default';\nimport styled from '@emotion/styled';\nimport { createEditor, Transforms, Editor as SlateEditor } from 'slate';\nimport { Editable, ReactEditor, Slate, withReact } from 'slate-react';\nimport { withHistory } from 'slate-history';\nimport { fromJS } from 'immutable';\nimport { isEqual } from 'lodash';\n\nimport { editorStyleVars, EditorControlBar } from '../styles';\nimport Toolbar from './Toolbar';\nimport { Element, Leaf } from './renderers';\nimport withLists from './plugins/lists/withLists';\nimport withBlocks from './plugins/blocks/withBlocks';\nimport withInlines from './plugins/inlines/withInlines';\nimport toggleMark from './plugins/inlines/events/toggleMark';\nimport toggleLink from './plugins/inlines/events/toggleLink';\nimport getActiveLink from './plugins/inlines/selectors/getActiveLink';\nimport isMarkActive from './plugins/inlines/locations/isMarkActive';\nimport isCursorInBlockType from './plugins/blocks/locations/isCursorInBlockType';\nimport { markdownToSlate, slateToMarkdown } from '../serializers';\nimport withShortcodes from './plugins/shortcodes/withShortcodes';\nimport insertShortcode from './plugins/shortcodes/insertShortcode';\nimport defaultEmptyBlock from './plugins/blocks/defaultEmptyBlock';\nimport withHtml from './plugins/html/withHtml';\n\nfunction visualEditorStyles({ minimal }) {\n  return `\n  position: relative;\n  overflow: auto;\n  font-family: ${fonts.primary};\n  min-height: ${minimal ? 'auto' : lengths.richTextEditorMinHeight};\n  border-top-left-radius: 0;\n  border-top-right-radius: 0;\n  border-top: 0;\n  margin-top: -${editorStyleVars.stickyDistanceBottom};\n  padding: 0;\n  display: flex;\n  flex-direction: column;\n  z-index: ${zIndex.zIndex100};\n`;\n}\n\nconst InsertionPoint = styled.div`\n  flex: 1 1 auto;\n  cursor: text;\n`;\n\nexport function mergeMediaConfig(editorComponents, field) {\n  // merge editor media library config to image components\n  if (editorComponents.has('image')) {\n    const imageComponent = editorComponents.get('image');\n    const fields = imageComponent?.fields;\n\n    if (fields) {\n      imageComponent.fields = fields.update(\n        fields.findIndex(f => f.get('widget') === 'image'),\n        f => {\n          // merge `media_library` config\n          if (field.has('media_library')) {\n            f = f.set(\n              'media_library',\n              field.get('media_library').mergeDeep(f.get('media_library')),\n            );\n          }\n          // merge 'media_folder'\n          if (field.has('media_folder') && !f.has('media_folder')) {\n            f = f.set('media_folder', field.get('media_folder'));\n          }\n          // merge 'public_folder'\n          if (field.has('public_folder') && !f.has('public_folder')) {\n            f = f.set('public_folder', field.get('public_folder'));\n          }\n          return f;\n        },\n      );\n    }\n  }\n}\n\nfunction Editor(props) {\n  const {\n    onAddAsset,\n    getAsset,\n    className,\n    field,\n    isShowModeToggle,\n    t,\n    isDisabled,\n    getEditorComponents,\n    getRemarkPlugins,\n    onChange,\n  } = props;\n\n  const editor = useMemo(\n    () =>\n      withHtml(\n        withReact(withHistory(withShortcodes(withBlocks(withLists(withInlines(createEditor())))))),\n      ),\n    [],\n  );\n\n  const emptyValue = [defaultEmptyBlock()];\n  let editorComponents = getEditorComponents();\n  const codeBlockComponent = fromJS(editorComponents.find(({ type }) => type === 'code-block'));\n\n  editorComponents =\n    codeBlockComponent || editorComponents.has('code-block')\n      ? editorComponents\n      : editorComponents.set('code-block', { label: 'Code Block', type: 'code-block' });\n\n  mergeMediaConfig(editorComponents, field);\n\n  const [editorValue, setEditorValue] = useState(\n    props.value\n      ? markdownToSlate(props.value, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        })\n      : emptyValue,\n  );\n\n  const renderElement = useCallback(\n    props => (\n      <Element {...props} classNameWrapper={className} codeBlockComponent={codeBlockComponent} />\n    ),\n    [],\n  );\n  const renderLeaf = useCallback(props => <Leaf {...props} />, []);\n\n  useEffect(() => {\n    if (props.pendingFocus) {\n      ReactEditor.focus(editor);\n    }\n  }, []);\n\n  function handleMarkClick(format) {\n    ReactEditor.focus(editor);\n    toggleMark(editor, format);\n  }\n\n  function handleBlockClick(format) {\n    ReactEditor.focus(editor);\n    if (format.endsWith('-list')) {\n      editor.toggleList(format);\n    } else {\n      editor.toggleBlock(format);\n    }\n  }\n\n  function handleLinkClick() {\n    toggleLink(editor, t('editor.editorWidgets.markdown.linkPrompt'));\n    ReactEditor.focus(editor);\n  }\n\n  function handleToggleMode() {\n    props.onMode('raw');\n  }\n\n  function handleInsertShortcode(pluginConfig) {\n    insertShortcode(editor, pluginConfig);\n  }\n\n  function handleKeyDown(event) {\n    for (const handler of editor.keyDownHandlers || []) {\n      if (handler(event, editor) === false) {\n        break;\n      }\n    }\n    ReactEditor.focus(editor);\n  }\n\n  function handleClickBelowDocument() {\n    ReactEditor.focus(editor);\n    Transforms.select(editor, { path: [0, 0], offset: 0 });\n    Transforms.select(editor, SlateEditor.end(editor, []));\n  }\n  const [toolbarKey, setToolbarKey] = useState(0);\n\n  function handleChange(newValue) {\n    if (!isEqual(newValue, editorValue)) {\n      setEditorValue(() => newValue);\n      onChange(\n        slateToMarkdown(newValue, {\n          voidCodeBlock: !!codeBlockComponent,\n          remarkPlugins: getRemarkPlugins(),\n        }),\n      );\n    }\n    setToolbarKey(prev => prev + 1);\n  }\n\n  function hasMark(format) {\n    return isMarkActive(editor, format);\n  }\n\n  function hasInline(format) {\n    if (format == 'link') {\n      return !!getActiveLink(editor);\n    }\n    return false;\n  }\n\n  function hasBlock(format) {\n    return isCursorInBlockType(editor, format);\n  }\n  function hasQuote() {\n    return isCursorInBlockType(editor, 'quote');\n  }\n  function hasListItems(type) {\n    return isCursorInBlockType(editor, type);\n  }\n\n  return (\n    <div\n      css={coreCss`\n        position: relative;\n      `}\n    >\n      <Slate editor={editor} value={editorValue} onChange={handleChange}>\n        <EditorControlBar>\n          {\n            <Toolbar\n              key={toolbarKey}\n              onMarkClick={handleMarkClick}\n              onBlockClick={handleBlockClick}\n              onLinkClick={handleLinkClick}\n              onToggleMode={handleToggleMode}\n              plugins={editorComponents}\n              onSubmit={handleInsertShortcode}\n              onAddAsset={onAddAsset}\n              getAsset={getAsset}\n              buttons={field.get('buttons')}\n              editorComponents={field.get('editor_components')}\n              hasMark={hasMark}\n              hasInline={hasInline}\n              hasBlock={hasBlock}\n              hasQuote={hasQuote}\n              hasListItems={hasListItems}\n              isShowModeToggle={isShowModeToggle}\n              t={t}\n              disabled={isDisabled}\n            />\n          }\n        </EditorControlBar>\n        {\n          <ClassNames>\n            {({ css, cx }) => (\n              <div\n                className={cx(\n                  className,\n                  css`\n                    ${visualEditorStyles({ minimal: field.get('minimal') })}\n                  `,\n                )}\n              >\n                {editorValue.length !== 0 && (\n                  <Editable\n                    className={css`\n                      padding: 16px 20px 0;\n                    `}\n                    renderElement={renderElement}\n                    renderLeaf={renderLeaf}\n                    onKeyDown={handleKeyDown}\n                    autoFocus={false}\n                  />\n                )}\n                <InsertionPoint onClick={handleClickBelowDocument} />\n              </div>\n            )}\n          </ClassNames>\n        }\n      </Slate>\n    </div>\n  );\n}\n\nEditor.propTypes = {\n  onAddAsset: PropTypes.func.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onChange: PropTypes.func.isRequired,\n  onMode: PropTypes.func.isRequired,\n  className: PropTypes.string.isRequired,\n  value: PropTypes.string,\n  field: ImmutablePropTypes.map.isRequired,\n  getEditorComponents: PropTypes.func.isRequired,\n  getRemarkPlugins: PropTypes.func.isRequired,\n  isShowModeToggle: PropTypes.bool.isRequired,\n  t: PropTypes.func.isRequired,\n};\n\nexport default Editor;\n"]} */",
|
|
100
101
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
101
102
|
};
|
|
102
103
|
function Editor(props) {
|
|
@@ -112,7 +113,7 @@ function Editor(props) {
|
|
|
112
113
|
getRemarkPlugins,
|
|
113
114
|
onChange
|
|
114
115
|
} = props;
|
|
115
|
-
const editor = (0, _react.useMemo)(() => (0, _slateReact.withReact)((0, _slateHistory.withHistory)((0, _withShortcodes.default)((0, _withBlocks.default)((0, _withLists.default)((0, _withInlines.default)((0, _slate.createEditor)())))))), []);
|
|
116
|
+
const editor = (0, _react.useMemo)(() => (0, _withHtml.default)((0, _slateReact.withReact)((0, _slateHistory.withHistory)((0, _withShortcodes.default)((0, _withBlocks.default)((0, _withLists.default)((0, _withInlines.default)((0, _slate.createEditor)()))))))), []);
|
|
116
117
|
const emptyValue = [(0, _defaultEmptyBlock.default)()];
|
|
117
118
|
let editorComponents = getEditorComponents();
|
|
118
119
|
const codeBlockComponent = (0, _immutable.fromJS)(editorComponents.find(({
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
var _slateHyperscript = require("slate-hyperscript");
|
|
8
|
+
var _slate = require("slate");
|
|
9
|
+
// source: https://github.com/ianstormtaylor/slate/blob/main/site/examples/ts/paste-html.tsx
|
|
10
|
+
|
|
11
|
+
const ELEMENT_TAGS = {
|
|
12
|
+
A: el => ({
|
|
13
|
+
type: 'link',
|
|
14
|
+
url: el.getAttribute('href')
|
|
15
|
+
}),
|
|
16
|
+
BLOCKQUOTE: () => ({
|
|
17
|
+
type: 'quote'
|
|
18
|
+
}),
|
|
19
|
+
H1: () => ({
|
|
20
|
+
type: 'heading-one'
|
|
21
|
+
}),
|
|
22
|
+
H2: () => ({
|
|
23
|
+
type: 'heading-two'
|
|
24
|
+
}),
|
|
25
|
+
H3: () => ({
|
|
26
|
+
type: 'heading-three'
|
|
27
|
+
}),
|
|
28
|
+
H4: () => ({
|
|
29
|
+
type: 'heading-four'
|
|
30
|
+
}),
|
|
31
|
+
H5: () => ({
|
|
32
|
+
type: 'heading-five'
|
|
33
|
+
}),
|
|
34
|
+
H6: () => ({
|
|
35
|
+
type: 'heading-six'
|
|
36
|
+
}),
|
|
37
|
+
IMG: el => ({
|
|
38
|
+
type: 'image',
|
|
39
|
+
url: el.getAttribute('src')
|
|
40
|
+
}),
|
|
41
|
+
LI: () => ({
|
|
42
|
+
type: 'list-item'
|
|
43
|
+
}),
|
|
44
|
+
OL: () => ({
|
|
45
|
+
type: 'numbered-list'
|
|
46
|
+
}),
|
|
47
|
+
P: () => ({
|
|
48
|
+
type: 'paragraph'
|
|
49
|
+
}),
|
|
50
|
+
PRE: () => ({
|
|
51
|
+
type: 'code'
|
|
52
|
+
}),
|
|
53
|
+
UL: () => ({
|
|
54
|
+
type: 'bulleted-list'
|
|
55
|
+
})
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
// COMPAT: `B` is omitted here because Google Docs uses `<b>` in weird ways.
|
|
59
|
+
const TEXT_TAGS = {
|
|
60
|
+
CODE: () => ({
|
|
61
|
+
code: true
|
|
62
|
+
}),
|
|
63
|
+
DEL: () => ({
|
|
64
|
+
strikethrough: true
|
|
65
|
+
}),
|
|
66
|
+
EM: () => ({
|
|
67
|
+
italic: true
|
|
68
|
+
}),
|
|
69
|
+
I: () => ({
|
|
70
|
+
italic: true
|
|
71
|
+
}),
|
|
72
|
+
S: () => ({
|
|
73
|
+
strikethrough: true
|
|
74
|
+
}),
|
|
75
|
+
STRONG: () => ({
|
|
76
|
+
bold: true
|
|
77
|
+
}),
|
|
78
|
+
U: () => ({
|
|
79
|
+
underline: true
|
|
80
|
+
})
|
|
81
|
+
};
|
|
82
|
+
function deserialize(el) {
|
|
83
|
+
if (el.nodeType === 3) {
|
|
84
|
+
return el.textContent;
|
|
85
|
+
} else if (el.nodeType !== 1) {
|
|
86
|
+
return null;
|
|
87
|
+
} else if (el.nodeName === 'BR') {
|
|
88
|
+
return '\n';
|
|
89
|
+
}
|
|
90
|
+
const {
|
|
91
|
+
nodeName
|
|
92
|
+
} = el;
|
|
93
|
+
let parent = el;
|
|
94
|
+
if (nodeName === 'PRE' && el.childNodes[0] && el.childNodes[0].nodeName === 'CODE') {
|
|
95
|
+
parent = el.childNodes[0];
|
|
96
|
+
}
|
|
97
|
+
let children = Array.from(parent.childNodes).map(deserialize).flat();
|
|
98
|
+
if (children.length === 0) {
|
|
99
|
+
children = [{
|
|
100
|
+
text: ''
|
|
101
|
+
}];
|
|
102
|
+
}
|
|
103
|
+
if (el.nodeName === 'BODY') {
|
|
104
|
+
return (0, _slateHyperscript.jsx)('fragment', {}, children);
|
|
105
|
+
}
|
|
106
|
+
if (ELEMENT_TAGS[nodeName]) {
|
|
107
|
+
const attrs = ELEMENT_TAGS[nodeName](el);
|
|
108
|
+
return (0, _slateHyperscript.jsx)('element', attrs, children);
|
|
109
|
+
}
|
|
110
|
+
if (TEXT_TAGS[nodeName]) {
|
|
111
|
+
const attrs = TEXT_TAGS[nodeName](el);
|
|
112
|
+
return children.map(child => (0, _slateHyperscript.jsx)('text', attrs, child));
|
|
113
|
+
}
|
|
114
|
+
return children;
|
|
115
|
+
}
|
|
116
|
+
function withHtml(editor) {
|
|
117
|
+
const {
|
|
118
|
+
insertData,
|
|
119
|
+
isInline,
|
|
120
|
+
isVoid
|
|
121
|
+
} = editor;
|
|
122
|
+
editor.isInline = element => {
|
|
123
|
+
return element.type === 'link' ? true : isInline(element);
|
|
124
|
+
};
|
|
125
|
+
editor.isVoid = element => {
|
|
126
|
+
return element.type === 'image' ? true : isVoid(element);
|
|
127
|
+
};
|
|
128
|
+
editor.insertData = data => {
|
|
129
|
+
const html = data.getData('text/html');
|
|
130
|
+
if (html) {
|
|
131
|
+
const parsed = new DOMParser().parseFromString(html, 'text/html');
|
|
132
|
+
const fragment = deserialize(parsed.body);
|
|
133
|
+
_slate.Transforms.insertFragment(editor, fragment);
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
insertData(data);
|
|
137
|
+
};
|
|
138
|
+
return editor;
|
|
139
|
+
}
|
|
140
|
+
var _default = exports.default = withHtml;
|