Следующее решение не работает для IE, для этого вам необходимо применить объекты TextRange и т. Д.Поскольку для этого используются выборки, он не должен нарушать HTML в обычных случаях, например:
<div>abcd<span>efg</span>hij</div>
с highlight(3,6);
выходными данными:
<div>abc<em>d<span>ef</span></em><span>g</span>hij</div>
Обратите внимание, как первый символ за пределами диапазона превращается в em
, а затем остальные символы в пределах span
в новый.Где, как если бы он просто открывал его в символе 3 и заканчивал в символе 6, он бы выдал недопустимую разметку, например:
<div>abc<em>d<span>ef</em>g</span>hij</div>
Код:
var r = document.createRange();
var s = window.getSelection()
r.selectNode($('div')[0]);
s.removeAllRanges();
s.addRange(r);
// not quite sure why firefox has problems with this
if ($.browser.webkit) {
s.modify("move", "backward", "documentboundary");
}
function highlight(start,end){
for(var st=0;st<start;st++){
s.modify("move", "forward", "character");
}
for(var st=0;st<(end-start);st++){
s.modify("extend", "forward", "character");
}
}
highlight(2,6);
var ra = s.getRangeAt(0);
var newNode = document.createElement("em");
newNode.appendChild(ra.extractContents());
ra.insertNode(newNode);
Пример:http://jsfiddle.net/niklasvh/4NDb9/
edit Похоже, по крайней мере у моего FF4 были некоторые проблемы с
s.modify("move", "backward", "documentboundary");
, но в то же время, похоже, он работает без него, поэтому япросто изменил его на
if ($.browser.webkit) {
s.modify("move", "backward", "documentboundary");
}
edit как указал Тим, модификация доступна только с FF4 и далее, поэтому я выбрал другой подход к получениювыбор, который не нуждается в методе модификации, в надежде сделать его немного более совместимым с браузером (IE все еще нуждается в собственном решении).
код:
var r = document.createRange();
var s = window.getSelection()
var pos = 0;
function dig(el){
$(el).contents().each(function(i,e){
if (e.nodeType==1){
// not a textnode
dig(e);
}else{
if (pos<start){
if (pos+e.length>=start){
range.setStart(e, start-pos);
}
}
if (pos<end){
if (pos+e.length>=end){
range.setEnd(e, end-pos);
}
}
pos = pos+e.length;
}
});
}
var start,end, range;
function highlight(element,st,en){
range = document.createRange();
start = st;
end = en;
dig(element);
s.addRange(range);
}
highlight($('div'),3,6);
var ra = s.getRangeAt(0);
var newNode = document.createElement("em");
newNode.appendChild(ra.extractContents());
ra.insertNode(newNode);
пример: http://jsfiddle.net/niklasvh/4NDb9/