4

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...?

Bambax
  • 2,920
  • 6
  • 34
  • 43

2 Answers2

7

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
      }
});
Binary9
  • 378
  • 1
  • 7
  • Thank you very much! The event is fired for every node inserted, though; it'd be nice to just be notified "a drop has been performed", but for now this works fine! ;-) – Bambax Sep 02 '11 at 09:46
  • 1
    This is a reasonable workaround for now. Note that IE < 9 does not support DOM mutation events, so this won't work in those browsers. Also, there's a good chance DOM mutation events will be removed and replaced with something else in future browsers (browser makers don't like them and consider them a mistake). – Tim Down Sep 02 '11 at 10:22
  • Indeed the spec says so: "this specification describes mutation events for reference and completeness of legacy behavior, but deprecates the use of both the MutationEvent interface and the MutationNameEvent interface" (http://www.w3.org/TR/DOM-Level-3-Events/#events-mutationevents) – Bambax Sep 02 '11 at 10:25
  • 1
    What would be a good approach for, say, IE 7-8? (Forgetting about IE6 -- although contentEditable was introduced in IE5, I think). – Bambax Sep 02 '11 at 10:26
  • Would the MutationObserver object be the right replacement for this event? https://developer.mozilla.org/en-US/docs/Web/API/MutationObserver – Alexis Wilke Jan 12 '14 at 05:05
0

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>
Magnus
  • 1,584
  • 19
  • 14