I want to insert some html in a contenteditable div.
When editing the content, the user clicks on an icon, a dialog pops and he enters the url & anchor text in the dialog. This causes that the editable div loses its focus and the link is inserted at the beginning of the div, not when the caret was. I tried many things but I'm stuck.
"rte" id of my editable content div
"link_add" id of button in dialog
$('#link_add').click(function (e)
{
$('#rte').focus();
document.execCommand('insertHTML', false, 'html content test');
close_modal ();
e.preventDefault();
});
I also tried the solution from set execcommand just for a div, using:
function execCommandOnElement(el, commandName, value)
But this empties the div and just paste the new content.
You need to save and restore the selection. IE for one loses it if the selection is collapsed (i.e. just a caret) when the editable element loses focus. Note that IE < 9 has completely different selection and range objects from other browsers. Here is some code for doing this that will work in all major browsers, including older IE. Note that it won't restore the original direction of the selection, however. If you need to do that, the solution becomes more complicated and can't be done in IE.
Before opening your dialog:
After closing the dialog:
Alternatively, if you were doing a lot of selection/range manipulation you could use the selection save/restore module of my Rangy library, but that would be overkill for just this one use.
I take it this is an "inline" dialog (much like StackOverflow's link dialog, for instance), which moves the focus and therefore the selection. The solution seems to be to save and restore the selection. Call getSelection() to get a reference to the selection and save the
anchorNode
,anchorOffset
,focusNode
andfocusOffset
properties and then usecollapse(anchorNode, anchorOffset)
andextend(focusNode, focusOffset)
to restore the selection once you have focused the contenteditable div. (If you're not interested in both nodes you could justcollapse(focusNode, focusOffset)
.)Probably suggested above wouldnt work properly, because of two reasons: first, need to use window.getSelection().getRangeAt(i).cloneRange(), and there is second, - selection and range are living data and refreshes automatically with user related actions, like new selections and focuses.