問題は、Rangyの保存/復元選択モジュールが、選択境界があるDOMに非表示のマーカー要素を挿入することによって機能し、コードがRangyのマーカー要素を含むすべてのHTMLタグを削除することです(エラーメッセージが示すように)。2つのオプションがあります。
- ではなく数値に色を付けるためのDOMトラバーサルソリューションに移動します
innerHTML
。これはより信頼性が高くなりますが、より複雑になります。
- 代替の文字インデックスベースの選択の保存と復元を実装します。これは一般的に壊れやすいでしょうが、この場合あなたが望むことをします。
アップデート
Rangy(上記のオプション2)の文字インデックスベースの選択の保存/復元をノックアップしました。少しラフですが、この場合はうまくいきます。これは、テキストノードをトラバースすることによって機能します。これをなんらかの形でRangyに追加するかもしれません。(2012年6月5日更新: Rangyのために、より信頼性の高い方法でこれを実装しました。)
jsFiddle: http: //jsfiddle.net/2rTA5/2/
コード:
function saveSelection(containerEl) {
var charIndex = 0, start = 0, end = 0, foundStart = false, stop = {};
var sel = rangy.getSelection(), range;
function traverseTextNodes(node, range) {
if (node.nodeType == 3) {
if (!foundStart && node == range.startContainer) {
start = charIndex + range.startOffset;
foundStart = true;
}
if (foundStart && node == range.endContainer) {
end = charIndex + range.endOffset;
throw stop;
}
charIndex += node.length;
} else {
for (var i = 0, len = node.childNodes.length; i < len; ++i) {
traverseTextNodes(node.childNodes[i], range);
}
}
}
if (sel.rangeCount) {
try {
traverseTextNodes(containerEl, sel.getRangeAt(0));
} catch (ex) {
if (ex != stop) {
throw ex;
}
}
}
return {
start: start,
end: end
};
}
function restoreSelection(containerEl, savedSel) {
var charIndex = 0, range = rangy.createRange(), foundStart = false, stop = {};
range.collapseToPoint(containerEl, 0);
function traverseTextNodes(node) {
if (node.nodeType == 3) {
var nextCharIndex = charIndex + node.length;
if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {
range.setStart(node, savedSel.start - charIndex);
foundStart = true;
}
if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {
range.setEnd(node, savedSel.end - charIndex);
throw stop;
}
charIndex = nextCharIndex;
} else {
for (var i = 0, len = node.childNodes.length; i < len; ++i) {
traverseTextNodes(node.childNodes[i]);
}
}
}
try {
traverseTextNodes(containerEl);
} catch (ex) {
if (ex == stop) {
rangy.getSelection().setSingleRange(range);
} else {
throw ex;
}
}
}
function formatText() {
var el = document.getElementById('pad');
var savedSel = saveSelection(el);
el.innerHTML = el.innerHTML.replace(/(<([^>]+)>)/ig,"");
el.innerHTML = el.innerHTML.replace(/([0-9])/ig,"<font color='red'>$1</font>");
// Restore the original selection
restoreSelection(el, savedSel);
}