1
0
mirror of https://github.com/ianstormtaylor/slate.git synced 2025-02-24 09:13:24 +01:00
slate/docs/walkthroughs/04-applying-custom-formatting.md
Matt Mazzola d457bc52b9 Docs fixes for 0.50.0 (#3133)
* docs: typos

* docs: markdown inline code formatting

https://meta.stackexchange.com/questions/82718/how-do-i-escape-a-backtick-within-in-line-code-in-markdown

* docs: remove unused value, move comment, add deps to callback

* docs: add deps to renderMark callback
2019-11-28 10:37:35 -05:00

5.1 KiB

Applying Custom Formatting

In the previous guide we learned how to create custom block types that render chunks of text inside different containers. But Slate allows for more than just "blocks".

In this guide, we'll show you how to add custom formatting options, like bold, italic, code or strikethrough.

So we start with our app from earlier:

const App = () => {
  const editor = useMemo(() => withReact(createEditor()), [])
  const renderElement = useCallback(props => {
    switch (props.element.type) {
      case 'code':
        return <CodeElement {...props} />
      default:
        return <DefaultElement {...props} />
    }
  }, [])

  return (
    <Slate editor={editor} defaultValue={defaultValue}>
      <Editable
        renderElement={renderElement}
        onKeyDown={event => {
          if (event.key === '`' && event.ctrlKey) {
            event.preventDefault()
            const { selection } = editor
            const isCode = selection
              ? Editor.match(editor, selection, { type: 'code' })
              : false

            Editor.setNodes(
              editor,
              { type: isCode ? 'paragraph' : 'code' },
              { match: 'block' }
            )
          }
        }}
      />
    </Slate>
  )
}

And now, we'll edit the onKeyDown handler to make it so that when you press control-B, it will add a "bold" mark to the currently selected text:

const App = () => {
  const editor = useMemo(() => withReact(createEditor()), [])
  const renderElement = useCallback(props => {
    switch (prop.element.type) {
      case 'code':
        return <CodeElement {...props} />
      default:
        return <DefaultElement {...props} />
    }
  }, [])

  return (
    <Slate editor={editor} defaultValue={defaultValue}>
      <Editable
        renderElement={renderElement}
        onKeyDown={event => {
          if (!event.ctrlKey) {
            return
          }

          switch (event.key) {
            // When "`" is pressed, keep our existing code block logic.
            case '`': {
              event.preventDefault()
              const { selection } = editor
              const isCode = selection
                ? Editor.match(editor, selection, { type: 'code' })
                : false

              Editor.setNodes(
                editor,
                { type: isCode ? null : 'code' },
                { match: 'block' }
              )
              break
            }

            // When "B" is pressed, add a bold mark to the text.
            case 'b': {
              event.preventDefault()
              Editor.addMarks(editor, [{ type: 'bold' }])
              break
            }
          }
        }}
      />
    </Slate>
  )
}

Okay, so we've got the hotkey handler setup... but! If you happen to now try selecting text and hitting Ctrl-B, you won't notice any change. That's because we haven't told Slate how to render a "bold" mark.

For every mark type you want to add to your schema, you need to give Slate a "renderer" for that mark, just like elements. So let's define our bold mark:

// Define a React component to render bold text with.
const BoldMark = props => {
  return <strong {...props.attributes}>{props.children}</strong>
}

Pretty familiar, right?

And now, let's tell Slate about that mark. To do that, we'll pass in the renderMark prop to our editor. Also, let's allow our mark to be toggled by changing addMark to toggleMark.

const App = () => {
  const editor = useMemo(() => withReact(createEditor()), [])
  const renderElement = useCallback(props => {
    switch (props.element.type) {
      case 'code':
        return <CodeElement {...props} />
      default:
        return <DefaultElement {...props} />
    }
  }, [])

  // Define a mark rendering function that is memoized with `useCallback`.
  const renderMark = useCallback(props => {
    switch (props.mark.type) {
      case 'bold': {
        return <BoldMark {...props} />
      }
    }
  }, [])

  return (
    <Slate editor={editor} defaultValue={defaultValue}>
      <Editable
        renderElement={renderElement}
        // Pass in the `renderMark` function.
        renderMark={renderMark}
        onKeyDown={event => {
          if (!event.ctrlKey) {
            return
          }

          switch (event.key) {
            case '`': {
              event.preventDefault()
              const { selection } = editor
              const isCode = selection
                ? Editor.match(editor, selection, { type: 'code' })
                : false

              Editor.setNodes(
                editor,
                { type: isCode ? null : 'code' },
                { match: 'block' }
              )
              break
            }

            case 'b': {
              event.preventDefault()
              Editor.addMarks(editor, [{ type: 'bold' }])
              break
            }
          }
        }}
      />
    </Slate>
  )
}

const BoldMark = props => {
  return <strong {...props.attributes}>{props.children}</strong>
}

Now, if you try selecting a piece of text and hitting Ctrl-B you should see it turn bold! Magic!