You can do this in current versions of all browsers. These browsers have at least one of the following:
- The standards-based approach, implemented by Firefox >= 20, from the CSSOM View spec:
document.caretPositionFromPoint()
- WebKit's proprietary version of the same:
document.caretRangeFromPoint()
.
- IE's proprietary
TextRange
object, which has a moveToPoint()
method that takes pixel coordinates. However, it seems that moveToPoint()
, which is used in all version of IE, can be buggy (see here and here, for example); I've simply been lucky that has worked in all the documents I've used it in.
However, Mozilla does not yet implement any of these and neither does Opera, so this can't be done in those browsers yet.
Firefox 20 and later supports document.caretPositionFromPoint()
. Opera 15 supports document.caretRangeFromPoint()
Here's some example code. It works in IE 5+, WebKit from around 2010 onwards, Firefox >= 20 and Opera >= 15.
Live demo: http://jsfiddle.net/timdown/ABjQP/
Code:
function createSelectionFromPoint(startX, startY, endX, endY) {
var doc = document;
var start, end, range = null;
if (typeof doc.caretPositionFromPoint != "undefined") {
start = doc.caretPositionFromPoint(startX, startY);
end = doc.caretPositionFromPoint(endX, endY);
range = doc.createRange();
range.setStart(start.offsetNode, start.offset);
range.setEnd(end.offsetNode, end.offset);
} else if (typeof doc.caretRangeFromPoint != "undefined") {
start = doc.caretRangeFromPoint(startX, startY);
end = doc.caretRangeFromPoint(endX, endY);
range = doc.createRange();
range.setStart(start.startContainer, start.startOffset);
range.setEnd(end.startContainer, end.startOffset);
}
if (range !== null && typeof window.getSelection != "undefined") {
var sel = window.getSelection();
sel.removeAllRanges();
sel.addRange(range);
} else if (typeof doc.body.createTextRange != "undefined") {
range = doc.body.createTextRange();
range.moveToPoint(startX, startY);
var endRange = range.duplicate();
endRange.moveToPoint(endX, endY);
range.setEndPoint("EndToEnd", endRange);
range.select();
}
}
与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…