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.
@@ -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;