Найти все текстовые узлы в HTML-странице
Возможный дубликат:
эквивалент getElementsByTagName() для textNodes
Для этот вопрос Мне нужно было найти все текстовые узлы под определенным node. Я могу сделать это так:
function textNodesUnder(root){
var textNodes = [];
addTextNodes(root);
[].forEach.call(root.querySelectorAll('*'),addTextNodes);
return textNodes;
function addTextNodes(el){
textNodes = textNodes.concat(
[].filter.call(el.childNodes,function(k){
return k.nodeType==Node.TEXT_NODE;
})
);
}
}
Однако это кажется неэлегантным в свете того, что с XPath можно просто запросить .//text()
и сделать с ним.
Самый простой способ получить все текстовые узлы под определенным элементом в документе HTML, который работает на IE9 +, Safari5 +, Chrome19 +, Firefox12 +, Opera11 +?
"Простейший" определяется свободно как "эффективный и короткий, без игры в гольф".
Ответы
Ответ 1
Основываясь на ответе @kennebec, немного более жесткая реализация той же логики:
function textNodesUnder(node){
var all = [];
for (node=node.firstChild;node;node=node.nextSibling){
if (node.nodeType==3) all.push(node);
else all = all.concat(textNodesUnder(node));
}
return all;
}
Однако гораздо более быстрый, более элегантный и элегантный интерфейс использует createTreeWalker
, чтобы браузер отфильтровывал все, кроме текстовых узлов:
function textNodesUnder(el){
var n, a=[], walk=document.createTreeWalker(el,NodeFilter.SHOW_TEXT,null,false);
while(n=walk.nextNode()) a.push(n);
return a;
}
Ответ 2
function deepText(node){
var A= [];
if(node){
node= node.firstChild;
while(node!= null){
if(node.nodeType== 3) A[A.length]=node;
else A= A.concat(deepText(node));
node= node.nextSibling;
}
}
return A;
}