Search code examples
javascriptreactjsdraftjs

How to insert an image using draft.js?


Here is my code trying to insert an image in the draft.js editor. But I failed. When I clicked on the button, only several empty lines were inserted, what's the problem?

MyEditor.js

import React from 'react';
import {Editor, EditorState, AtomicBlockUtils} from 'draft-js';

export class MyEditor extends React.Component {

  constructor(props) {
    super(props);
    this.state = {editorState: EditorState.createEmpty()};
    this.onChange = (editorState) => this.setState({editorState});
  }

  render() {
    return (
      <div>
        <Editor editorState={this.state.editorState} onChange={this.onChange}/>
        <button onClick={this.handleClick}>Insert an image</button>
      </div>
    );
  }

  handleClick = () => {
    const base64 = "";
    const newEditorState = this.insertImage(this.state.editorState, base64);
    this.onChange(newEditorState);
  };

  insertImage = (editorState, base64) => {
    const contentState = editorState.getCurrentContent();
    const contentStateWithEntity = contentState.createEntity(
      'image',
      'IMMUTABLE',
      { src: base64 },
    );
    const entityKey = contentStateWithEntity.getLastCreatedEntityKey();
    const newEditorState = EditorState.set(
      editorState,
      { currentContent: contentStateWithEntity },
    );
    return AtomicBlockUtils.insertAtomicBlock(newEditorState, entityKey, ' ');
  };

}

App.js

class App extends Component {
  render() {
    return (
      <MyEditor/>
    );
  }
}

export default App;

Result

enter image description here


Solution

  • For image support you should use a draft-js image plugin such as draft-js-image-plugin: https://www.draft-js-plugins.com/plugin/image

    By default draft-js doesn't support plugins, so it's a good idea to use draft-js-plugins-editor, which extend draft-js functionality.

    So your code would be:

    import React from "react";
    import { convertToRaw, EditorState, AtomicBlockUtils } from "draft-js";
    
    import Editor from "draft-js-plugins-editor";
    import createImagePlugin from "draft-js-image-plugin";
    
    const imagePlugin = createImagePlugin();
    const plugins = [imagePlugin];
    
    export default class MyEditor extends React.Component {
      constructor(props) {
        super(props);
        this.state = { editorState: EditorState.createEmpty() };
        this.onChange = editorState => this.setState({ editorState });
      }
    
      render() {
        return (
          <div>
            <Editor
              editorState={this.state.editorState}
              onChange={this.onChange}
              plugins={plugins}
              ref={element => {
                this.editor = element;
              }}
            />
            <button onClick={this.handleClick}>Insert an image</button>
            <pre>
              {JSON.stringify(
                convertToRaw(this.state.editorState.getCurrentContent()),
                null,
                "  "
              )}
            </pre>
          </div>
        );
      }
    
      handleClick = () => {
        const base64 =
          "";
        const newEditorState = this.insertImage(this.state.editorState, base64);
        this.onChange(newEditorState);
      };
    
      insertImage = (editorState, base64) => {
        const contentState = editorState.getCurrentContent();
        const contentStateWithEntity = contentState.createEntity(
          "image",
          "IMMUTABLE",
          { src: base64 }
        );
        const entityKey = contentStateWithEntity.getLastCreatedEntityKey();
        const newEditorState = EditorState.set(editorState, {
          currentContent: contentStateWithEntity
        });
        return AtomicBlockUtils.insertAtomicBlock(newEditorState, entityKey, " ");
      };
    }

    The latest versions of draft-js-image-plugin has an error, so you can use 2.0.1 version.

    I'm using the latest draft-js version "draft-js": "0.10.5",, "draft-js-image-plugin": "2.0.1", and "draft-js-plugins-editor": "2.0.3",