简体   繁体   中英

Drag and drop event in a contentEditable element

What event is fired when one drops something in a contentEditable element (after dragging)?

I'm talking about plain old drag and drop, NOT HTML5 drag and drop (where any element can be made draggable); the use case is simply:

  • there's a contentEditable div on the page, used as an editor
  • the user drags and drops some HTML from the current page or from another page, or from another browser's window (so there really isn't any concept of "source" object: the source can come from outside the browser)
  • I need to be notified that the content has been dropped into the contentEditable div so that I can act on it (clean it)

I could poll the div to see if anything's there that's not clean, but it's expensive and "ugly"; surely there's an event that fires when a drop occurs...?

I faced the same problem while writing a tinyMCE plugin. What I found the best way to track drag and drop of elements in a contentEditable zone is to listen to the 'DOMNodeInserted' event on the contentEditable element.

Note that this element is fired by the contentEditable element when the drop is performed so that its target property is set to this element. You can retrieve the moved element by checking the event.originalEvent.target property.

Be aware that this event is fired once the dropped is finished and that the drop element has been inserted.

$('#editor').bind('DOMNodeInserted', function(event){
      if(event.originalEvent && event.originalEvent.target){
        var target = $(event.originalEvent.target);
        //now you can check what has been moved
      }
});

Here is a workaround solution, just using the ondrop event. The ondrop event works, but the problem is that it fires before the content has changed. So the workaround is to use setTimeout with a little delay.

Here is the solution snippet you can test. When (after) you drop an image on the top div, the content of that div will be copied to the bottom div. Read the comment in the code, it gives more info on how you can handle the delay.

 function doSomething() { document.getElementById('result').innerHTML=document.getElementById('edt').innerHTML; } function takeCareOfDrop() { setTimeout(function () { doSomething(); }, 100); // <-- optional setTimeout(function () { doSomething(); }, 1000); /* I call doSomething two times above. Once after 100 ms and then again after 1000 ms. If it is not acceptable to call it twice, then only use 1000 ms, or some delay appropriate for you. If it IS acceptable to call it twice, then the advantage of doing it twice is this: In my case I tried 1 ms, but that was not enough. So I suspected that in some cases 100 ms might not be enough either. But 1000 ms is a bit long. So I made it call doSomething two times, once after 100 ms and one more time after 1000 ms, just to be sure. In this way, it mostly happens instantly (100 ms), but in some cases with a bit of a delay (1000 ms). */ }
 div {height:65px;border:1px solid red;}
 Try to drop a small image on the top div. <img src="" alt=""> <div id="edt" contenteditable="true" ondrop="takeCareOfDrop()"><p>Lorem ipsum</p></div> <div id="result"</div>

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM