How to insert an image using draft.js?
Asked Answered
A

4

10

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

Aeromarine answered 12/5, 2018 at 10:22 Comment(0)
C
10

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",

Conduce answered 12/5, 2018 at 11:13 Comment(3)
Thanks, it really worked! But could you tell me why it didn't work without this plugin? – Aeromarine
Draft-js has limited blocks by default. However you can add custom block support to your draft-js by passing blockRendererFn to draft-js (you can see more information here draftjs.org/docs/advanced-topics-block-components.html), but it's easier to do with draft-js-plugins-editor, where this function is already defined. – Conduce
"you can add custom block support to your draft-js by passing blockRendererFn": Doc link changed, see draftjs.org/docs/advanced-topics-block-components draftjs.org/docs/advanced-topics-custom-block-render-map – Dale
D
1

You can add images without plugins, see official demo called "Media"

It uses blockRendererFn attribute of draft-js Editor component.

const Image = (props) => {
  return <img src={props.src} style={styles.media} />;
};

const Media = (props) => {
  const entity = props.contentState.getEntity(
    props.block.getEntityAt(0)
  );
  const {src} = entity.getData();
  const type = entity.getType();

  let media;
  if (type === 'image') {
    media = <Image src={src} />;
  }

  return media;
};

function mediaBlockRenderer(block) {
  if (block.getType() === 'atomic') {
    return {
      component: Media,
      editable: false,
    };
  }

  return null;
}

// . . . . .
<Editor
   blockRendererFn={mediaBlockRenderer}
Dale answered 3/9, 2021 at 8:3 Comment(1)
i was trying this but when i paste some image all i get is this "πŸ“·", the block type is unstyled – Miosis
B
1

As of 2021

draftjs v 0.11.2 integrated with react-draft-wysiwyg 1.14.7.

I was able to add an image doing the following.

const addImage = () => {

    // CREATE <img /> block
    const entityKey = editorState // from STATE
      .getCurrentContent()
      .createEntity('IMAGE', 'MUTABLE', {
        src:'some_img_url',
        height: '100px',
        width: '100px',
    }).getLastCreatedEntityKey();
     
    // NEW EDITOR STATE
    const newEditorState = AtomicBlockUtils.insertAtomicBlock(
      editorState,
      entityKey,
      ' '
    );

    // SETSTATE
    setEditorState(newEditorState);
}
Beltane answered 31/12, 2021 at 2:51 Comment(1)
Only thing that has worked for me so far! Anyone know of a way for the editorState.getCurrentContent().getPlainText() to track the url for the backend to see the image url? I can track via another useState as a workaround, but wondering if there is a way to see it from the editorState. Right now it's just empty lines. – Lenlena
T
0

You can use addImage function in the imagePlugin

// import `Editor` from `'@draft-js-plugins/editor` and not `draftjs`
import Editor from '@draft-js-plugins/editor';
// call create plugin
const imagePlugin = createImagePlugin();
// you can find `addImage` function inside the created `imagePlugin`
const { addImage } = imagePlugin;
const image = "https://.....";
// create editorState
const [editorState, setEditorState] = useState(() => EditorState.createEmpty());
// inside your insert image function
const insertImage(){
  const newState = addImage(editorState, image);
  // setEditorState(newState);  does not seems to work all time, use settimeout
  setTimeout(setEditorState, 0, newState);
}

// add imagePlugin to the list of plugins
<Editor
   editorState={editorState}
   onChange={setEditorState}
   plugins={[imagePlugin, /* Other plugins here*/]}/>
Tog answered 24/3, 2023 at 15:14 Comment(0)

© 2022 - 2024 β€” McMap. All rights reserved.