import React, { useState, useCallback, useMemo } from 'react' import { jsx } from 'slate-hyperscript' import { Transforms, createEditor, Descendant } from 'slate' import { withHistory } from 'slate-history' import { css } from 'emotion' import { Slate, Editable, withReact, useSelected, useFocused, } from 'slate-react' const ELEMENT_TAGS = { A: el => ({ type: 'link', url: el.getAttribute('href') }), BLOCKQUOTE: () => ({ type: 'quote' }), H1: () => ({ type: 'heading-one' }), H2: () => ({ type: 'heading-two' }), H3: () => ({ type: 'heading-three' }), H4: () => ({ type: 'heading-four' }), H5: () => ({ type: 'heading-five' }), H6: () => ({ type: 'heading-six' }), IMG: el => ({ type: 'image', url: el.getAttribute('src') }), LI: () => ({ type: 'list-item' }), OL: () => ({ type: 'numbered-list' }), P: () => ({ type: 'paragraph' }), PRE: () => ({ type: 'code' }), UL: () => ({ type: 'bulleted-list' }), } // COMPAT: `B` is omitted here because Google Docs uses `` in weird ways. const TEXT_TAGS = { CODE: () => ({ code: true }), DEL: () => ({ strikethrough: true }), EM: () => ({ italic: true }), I: () => ({ italic: true }), S: () => ({ strikethrough: true }), STRONG: () => ({ bold: true }), U: () => ({ underline: true }), } export const deserialize = el => { if (el.nodeType === 3) { return el.textContent } else if (el.nodeType !== 1) { return null } else if (el.nodeName === 'BR') { return '\n' } const { nodeName } = el let parent = el if ( nodeName === 'PRE' && el.childNodes[0] && el.childNodes[0].nodeName === 'CODE' ) { parent = el.childNodes[0] } let children = Array.from(parent.childNodes) .map(deserialize) .flat() if (children.length === 0) { children = [{ text: '' }] } if (el.nodeName === 'BODY') { return jsx('fragment', {}, children) } if (ELEMENT_TAGS[nodeName]) { const attrs = ELEMENT_TAGS[nodeName](el) return jsx('element', attrs, children) } if (TEXT_TAGS[nodeName]) { const attrs = TEXT_TAGS[nodeName](el) return children.map(child => jsx('text', attrs, child)) } return children } const PasteHtmlExample = () => { const [value, setValue] = useState(initialValue) const renderElement = useCallback(props => , []) const renderLeaf = useCallback(props => , []) const editor = useMemo( () => withHtml(withReact(withHistory(createEditor()))), [] ) return ( setValue(value)}> ) } const withHtml = editor => { const { insertData, isInline, isVoid } = editor editor.isInline = element => { return element.type === 'link' ? true : isInline(element) } editor.isVoid = element => { return element.type === 'image' ? true : isVoid(element) } editor.insertData = data => { const html = data.getData('text/html') if (html) { const parsed = new DOMParser().parseFromString(html, 'text/html') const fragment = deserialize(parsed.body) Transforms.insertFragment(editor, fragment) return } insertData(data) } return editor } const Element = props => { const { attributes, children, element } = props switch (element.type) { default: return

{children}

case 'quote': return
{children}
case 'code': return (
          {children}
        
) case 'bulleted-list': return
    {children}
case 'heading-one': return

{children}

case 'heading-two': return

{children}

case 'heading-three': return

{children}

case 'heading-four': return

{children}

case 'heading-five': return
{children}
case 'heading-six': return
{children}
case 'list-item': return
  • {children}
  • case 'numbered-list': return
      {children}
    case 'link': return ( {children} ) case 'image': return } } const ImageElement = ({ attributes, children, element }) => { const selected = useSelected() const focused = useFocused() return (
    {children}
    ) } const Leaf = ({ attributes, children, leaf }) => { if (leaf.bold) { children = {children} } if (leaf.code) { children = {children} } if (leaf.italic) { children = {children} } if (leaf.underline) { children = {children} } if (leaf.strikethrough) { children = {children} } return {children} } const initialValue: Descendant[] = [ { type: 'paragraph', children: [ { text: "By default, pasting content into a Slate editor will use the clipboard's ", }, { text: "'text/plain'", code: true }, { text: " data. That's okay for some use cases, but sometimes you want users to be able to paste in content and have it maintain its formatting. To do this, your editor needs to handle ", }, { text: "'text/html'", code: true }, { text: ' data. ' }, ], }, { type: 'paragraph', children: [{ text: 'This is an example of doing exactly that!' }], }, { type: 'paragraph', children: [ { text: "Try it out for yourself! Copy and paste some rendered HTML rich text content (not the source code) from another site into this editor and it's formatting should be preserved.", }, ], }, ] export default PasteHtmlExample