ブログ投稿画面などのエディタで、選択文字列を<strong>強調</strong>
とかって囲んだりする機能がありますが、それをIEでやるときの話です。
例えばこのはてなブログの見出しボタンとかもそうですね。
h1とかで囲んでくれます。
BとかIとかもです。
検索すると参考サイトとして上のようなページがあるのですが、2007年の記事でして、しかもうまく動きません。
「筍子」を押してstrongボタンを押すと、その後ろの「。学」が囲まれちゃうようです。
テキストエリアの1行目だと動いていて、2行目、3行目と1文字ずつずれてしまうようです。
ということは改行コードですね。
そこで検索してこの記事を見つけました。2006年の記事で、IEに対比するものとしてNNとか書いてあります。
NNに続くMozilla系だとテキストエリアの改行コードが¥nなんですが、IEだと¥r¥nになっちゃうようです。
ということで、1件目の記事のコードを2件目の記事を参考に書き換えてみました。
変更箇所が下記になります。選択文字列の地点を取るgetAreaRangeメソッドの、if(isIE)の辺りです。
...
obj.focus();
var range = document.selection.createRange();
var len = range.text.replace(/\r/g, "").length;
var clone = range.duplicate();
clone.moveToElementText(obj);
var all_len = clone.text.replace(/\r/g, "").length;
clone.setEndPoint( 'StartToStart', range );
var s = all_len - clone.text.replace(/\r/g, "").length;
pos.start = s;
pos.end = s + len;
...
変更点としては、text.lengthを全て、text.replace(/\r/g, "").lengthにしたのと、
setEndPoint("EndToEnd" をsetEndPoint("StartToStart" にしました。
デモのHTML全文は下記のようになります。
<html>
<head>
<script type="text/javascript">
window.onload = function() {
var enc = document.getElementById("enclose");
enc.onclick = function() {
surroundHTML("div", "txt");
};
function getAreaRange(obj) {
var pos = new Object();
if (isIE) {
obj.focus();
var range = document.selection.createRange();
var len = range.text.replace(/\r/g, "").length;
var clone = range.duplicate();
clone.moveToElementText(obj);
var all_len = clone.text.replace(/\r/g, "").length;
clone.setEndPoint( 'StartToStart', range );
var s = all_len - clone.text.replace(/\r/g, "").length;
pos.start = s;
pos.end = s + len;
}
else if(window.getSelection()) {
pos.start = obj.selectionStart;
pos.end = obj.selectionEnd;
}
return pos;
}
var isIE = (navigator.appName.toLowerCase().indexOf('internet explorer')+1?1:0);
function surroundHTML(tag, obj) {
var target = document.getElementById(obj);
var pos = getAreaRange(target);
var val = target.value;
var range = val.slice(pos.start, pos.end);
var beforeNode = val.slice(0, pos.start);
var afterNode = val.slice(pos.end);
var insertNode;
if (range || pos.start != pos.end) {
insertNode = '<' + tag + '>' + range + '</' + tag + '>';
target.value = beforeNode + insertNode + afterNode;
}
else if (pos.start == pos.end) {
insertNode = '<' + tag + '>' + '</' + tag + '>';
target.value = beforeNode + insertNode + afterNode;
}
}
}
</script>
</head>
<body>
<div>
<input type="button" id="enclose" value="enclose" />
</div>
<div>
<textarea id="txt" rows="20" cols="100">
即興ソング
カメルーン
ブランチ
発送商品の場合、良品と交換いたします。不良品の返品にかかる送料は、弊社が負担いたします。ただし、良品在庫のない場合及び数量限定品の場合については個別にご相談させていただきます。 デジタルコンテンツ(PDF版書籍等)の場合は、再ダウンロード方法についてご案内いたします。
</textarea>
</div>
</body>
</html>