Die folgende Lösung funktioniert nicht für den IE, Sie müssen dafür TextRange-Objekte usw. verwenden. Da hierbei Auswahlen verwendet werden, sollte der HTML-Code in normalen Fällen nicht beschädigt werden, zum Beispiel:
<div>abcd<span>efg</span>hij</div>
Mit highlight(3,6);
Ausgänge:
<div>abc<em>d<span>ef</span></em><span>g</span>hij</div>
Beachten Sie, wie das erste Zeichen außerhalb der Spanne in eine em
und dann der Rest innerhalb der span
in eine neue. Wenn es nur bei Zeichen 3 geöffnet und bei Zeichen 6 beendet würde, würde es ungültiges Markup wie:
<div>abc<em>d<span>ef</em>g</span>hij</div>
Der Code:
~~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);
Exemple : http://jsfiddle.net/niklasvh/4NDb9/
modifier Sieht so aus, als hätte zumindest mein FF4 einige Probleme mit
s.modify("move", "backward", "documentboundary");
aber gleichzeitig scheint es auch ohne zu funktionieren, also habe ich es einfach geändert in
if ($.browser.webkit) {
s.modify("move", "backward", "documentboundary");
}~~
modifier Tim wies darauf hin, dass "modify" erst ab FF4 verfügbar ist, also habe ich einen anderen Ansatz gewählt, um die Auswahl zu erhalten, der die "modify"-Methode nicht benötigt, in der Hoffnung, sie etwas browser-kompatibler zu machen (IE braucht immer noch seine eigene Lösung).
Der Code:
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);
Beispiel: http://jsfiddle.net/niklasvh/4NDb9/