, span , span. , . - .
drop , , span. createCollapsedRangeFromPoint, Tim Down. .
this jsfiddle. Chrome Firefox, IE11. , .
var $mainInput = $('#main-input');
var draggedText;
var dragStartContainer;
var dragEndContainer;
var dragStartOffset;
var dragEndOffset;
$mainInput.on('dragstart', function(e) {
var dragRange = getSelectionRange();
dragStartContainer = dragRange.startContainer;
dragEndContainer = dragRange.endContainer;
dragStartOffset = dragRange.startOffset;
dragEndOffset = dragRange.endOffset;
draggedText = e.originalEvent.dataTransfer.getData("text");
});
$mainInput.on('drop', function(e) {
var range = createCollapsedRangeFromPoint(e.clientX, e.clientY);
var container = range.startContainer;
if (container && (container !== dragStartContainer || container !== dragEndContainer)) {
e.preventDefault();
var startOffset = range.startOffset;
var str = container.textContent;
var strPrepend = str.substr(0, range.startOffset);
var strAppend = str.substr(range.startOffset);
container.textContent = strPrepend + draggedText + strAppend;
var spaceCount = getSpaceCountAtStart(container);
setSelectionRange(dragStartContainer, dragEndContainer, dragStartOffset, dragEndOffset);
document.execCommand("delete", false);
var spaceCountCorrection = getSpaceCountAtStart(container) - spaceCount;
var startRemoved = (dragEndContainer === container ? dragEndOffset - 1 : 0);
var startOffset = Math.max(0, startOffset + spaceCountCorrection - startRemoved);
var endOffset = startOffset + draggedText.length;
setSelectionRange(container, container, startOffset, endOffset);
}
});
function getSpaceCountAtStart(container) {
var innerHTML = container.parentNode.innerHTML;
var count = 0;
for (var i = 0; i < innerHTML.length; i++) {
if (innerHTML[i] === ' ') {
count += 1;
} else {
break;
}
}
return count;
}
function createCollapsedRangeFromPoint(x, y) {
var doc = document;
var position, range = null;
if (typeof doc.caretPositionFromPoint != "undefined") {
position = doc.caretPositionFromPoint(x, y);
range = doc.createRange();
range.setStart(position.offsetNode, position.offset);
range.collapse(true);
} else if (typeof doc.caretRangeFromPoint != "undefined") {
range = doc.caretRangeFromPoint(x, y);
} else if (typeof doc.body.createTextRange != "undefined") {
range = doc.body.createTextRange();
range.moveToPoint(x, y);
}
return range;
}
function getSelectionRange() {
var sel;
if (window.getSelection) {
sel = window.getSelection();
if (sel.rangeCount) {
return sel.getRangeAt(0);
}
} else if (document.selection) {
return document.createRange();
}
return null;
}
function setSelectionRange(startContainer, endContainer, startOffset, endOffset) {
var sel = window.getSelection();
sel.removeAllRanges();
var range = document.createRange();
range.setStart(startContainer, startOffset);
range.setEnd(endContainer, endOffset);
sel.addRange(range);
}