admin管理员组文章数量:1344207
In Javascript, I'd like determine whether an element, say an A
element, exists inside a given range/textRange. The aim is to determine if the user's current selection contains a link. I am building a rich text editor control.
The range object has a monAncestorContainer (W3C) or parentElement() (Microsoft) method which returns the closest mon anscestor of all elements in the range. However, looking inside this element for A
elements won't work, since this mon ancestor may also contain elements that aren't in the range, since the range can start or end part way through a parent.
How would you achieve this?
In Javascript, I'd like determine whether an element, say an A
element, exists inside a given range/textRange. The aim is to determine if the user's current selection contains a link. I am building a rich text editor control.
The range object has a monAncestorContainer (W3C) or parentElement() (Microsoft) method which returns the closest mon anscestor of all elements in the range. However, looking inside this element for A
elements won't work, since this mon ancestor may also contain elements that aren't in the range, since the range can start or end part way through a parent.
How would you achieve this?
Share Improve this question asked May 19, 2011 at 1:36 thomasrutterthomasrutter 117k31 gold badges153 silver badges170 bronze badges5 Answers
Reset to default 5How about selection.containsNode? https://developer.mozilla/en/DOM/Selection/containsNode
something like:
var selection = window.getSelection();
var range = selection.getRangeAt(0);
var result = $('a', range.monAncestorContainer).filter(function() {
return selection.containsNode(this);
});
console.log(result);
I ended up going with a solution like this:
var findinselection = function(tagname, container) {
var
i, len, el,
rng = getrange(),
prng,
selparent;
if (rng) {
selparent = rng.monAncestorContainer || rng.parentElement();
// Look for an element *around* the selected range
for (el = selparent; el !== container; el = el.parentNode) {
if (el.tagName && el.tagName.toLowerCase() === tagname) {
return el;
}
}
// Look for an element *within* the selected range
if (!rng.collapsed && (rng.text === undefined || rng.text) &&
selparent.getElementsByTagName) {
el = selparent.getElementsByTagName(tagname);
prng = document.createRange ?
document.createRange() : document.body.createTextRange();
for (i = 0, len = el.length; i < len; i++) {
// determine if element el[i] is within the range
if (document.createRange) { // w3c
prng.selectNodeContents(el[i]);
if (rng.pareBoundaryPoints(Range.END_TO_START, prng) < 0 &&
rng.pareBoundaryPoints(Range.START_TO_END, prng) > 0) {
return el[i];
}
}
else { // microsoft
prng.moveToElementText(el[i]);
if (rng.pareEndPoints("StartToEnd", prng) < 0 &&
rng.pareEndPoints("EndToStart", prng) > 0) {
return el[i];
}
}
}
}
}
};
Where getrange() is another function of mine to get the current selection as a range object.
To use, call it like
var link = findselection('a', editor);
Where editor is the contenteditable element, or body in a designmode iframe.
This is a bit of a pain in the bum to do cross-browser. You could use my Rangy library, which is probably overkill for just this task but does make it more straightforward and works in all major browsers. The following code assumes only one Range is selected:
var sel = rangy.getSelection();
if (sel.rangeCount) {
var range = sel.getRangeAt(0);
var links = range.getNodes([1], function(node) {
return node.tagName.toLowerCase() == "a" && range.containsNode(node);
});
}
I'm using this code that works with IE / Chrome / FF: (I'm using it to select rows <tr> in a table)
// yourLink is the DOM element you want to check
var selection = window.getSelection().getRangeAt(0)
var node = document.createRange()
node.selectNode(yourLink)
var s2s = selection.pareBoundaryPoints(Range.START_TO_END, node)
var s2e = selection.pareBoundaryPoints(Range.START_TO_START, node)
var e2s = selection.pareBoundaryPoints(Range.END_TO_START, node)
var e2e = selection.pareBoundaryPoints(Range.END_TO_END, node)
if ((s2s != s2e) || (e2s != e2e) || (s2s!=e2e))
console.log("your node is inside selection")
In the case of the range on the searched element, what I wrote is useful. (But only for that case!)
First I wrote a function that returns the Node found in the range: getNodeFromRange(rangeObject). Using this function it was already easy to write the function that returns the desired Node: findTagInRange(tagName).
function getNodeFromRange(range) {
if(range.endContainer.nodeType==Node.ELEMENT_NODE) {
return range.endContainer;
}
if(range.endContainer.nodeType==Node.TEXT_NODE) {
return range.endContainer.parentNode;
}
else {
// the 'startContainer' it isn't on an Element (<p>, <div>, etc...)
return;
}
}
function findTagInRange(tagName, range) {
var node = getNodeFromRange(range);
if(node && typeof(node.tagName)!='undefiend' && node.tagName.toLowerCase()==tagName.toLowerCase()) {
return $(node);
}
return;
}
And then I can use it as follows:
var link = findTagInRange('A', range);
And I see the determination of the range you've already solved. :)
本文标签: javascriptHow to know if there is a link element within the selectionStack Overflow
版权声明:本文标题:javascript - How to know if there is a link element within the selection - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743743569a2531343.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论