如何使用 jQuery 选择文本节点?

Posted

技术标签:

【中文标题】如何使用 jQuery 选择文本节点?【英文标题】:How do I select text nodes with jQuery? 【发布时间】:2010-09-22 20:40:33 【问题描述】:

我想获取一个元素的所有后代文本节点,作为一个 jQuery 集合。最好的方法是什么?

【问题讨论】:

【参考方案1】:

jQuery 没有为此提供方便的功能。您需要将contents()find() 结合起来,find() 将仅提供子节点但包含文本节点,而find() 将提供所有后代元素但不提供文本节点。这是我想出的:

var getTextNodesIn = function(el) 
    return $(el).find(":not(iframe)").addBack().contents().filter(function() 
        return this.nodeType == 3;
    );
;

getTextNodesIn(el);

注意:如果您使用的是 jQuery 1.7 或更早版本,上面的代码将不起作用。要解决此问题,请将 addBack() 替换为 andSelf()。从 1.8 开始,andSelf() 已被弃用,取而代之的是 addBack()

与纯 DOM 方法相比,这有点低效,并且必须包含 ugly workaround for jQuery's overloading of its contents() function(感谢 cmets 中的 @rabidsnail 指出),所以这里是使用简单递归函数的非 jQuery 解决方案。 includeWhitespaceNodes 参数控制空白文本节点是否包含在输出中(在 jQuery 中它们会被自动过滤掉)。

更新:修复 includeWhitespaceNodes 错误时的错误。

function getTextNodesIn(node, includeWhitespaceNodes) 
    var textNodes = [], nonWhitespaceMatcher = /\S/;

    function getTextNodes(node) 
        if (node.nodeType == 3) 
            if (includeWhitespaceNodes || nonWhitespaceMatcher.test(node.nodeValue)) 
                textNodes.push(node);
            
         else 
            for (var i = 0, len = node.childNodes.length; i < len; ++i) 
                getTextNodes(node.childNodes[i]);
            
        
    

    getTextNodes(node);
    return textNodes;


getTextNodesIn(el);

【讨论】:

传入的元素可以是一个div的名字吗? @crosenblum:如果你是这个意思,你可以先打电话给document.getElementById()var div = document.getElementById("foo"); var textNodes = getTextNodesIn(div); 由于 jQuery 中的一个错误,如果您在 el 中有任何 iframe,您将需要使用 .find(':not(iframe)') 而不是 .find('*') 。跨度> @rabidsnail:我认为,.contents() 的使用意味着它也会搜索 iframe。我不明白它怎么可能是一个错误。 bugs.jquery.com/ticket/11275 这是否真的是一个错误似乎还有待商榷,但如果你在包含 iframe 的节点上调用 find('*').contents() '未添加到 dom 中,您将在未定义的点收到异常。【参考方案2】:

Jauco 在评论中发布了一个很好的解决方案,所以我在这里复制它:

$(elem)
  .contents()
  .filter(function() 
    return this.nodeType === 3; //Node.TEXT_NODE
  );

【讨论】:

其实 $(elem) .contents() .filter(function() return this.nodeType == Node.TEXT_NODE; );够了 IE7 没有定义 Node 全局,所以你必须使用 this.nodeType == 3,不幸的是:***.com/questions/1423599/node-textnode-and-ie7 这是否不仅返回作为元素的直接子元素的文本节点,而不是 OP 请求的元素的后代? 当文本节点深深嵌套在其他元素中时,这将不起作用,因为 contents() 方法只返回直接子节点,api.jquery.com/contents @Jauco,不,还不够!因为 .contents() 只返回直接子节点【参考方案3】:
$('body').find('*').contents().filter(function ()  return this.nodeType === 3; );

【讨论】:

【参考方案4】:

jQuery.contents() 可以与jQuery.filter 一起使用以查找所有子文本节点。稍加改动,您也可以找到孙子文本节点。不需要递归:

$(function() 
  var $textNodes = $("#test, #test *").contents().filter(function() 
    return this.nodeType === Node.TEXT_NODE;
  );
  /*
   * for testing
   */
  $textNodes.each(function() 
    console.log(this);
  );
);
div  margin-left: 1em; 
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>

<div id="test">
  child text 1<br>
  child text 2
  <div>
    grandchild text 1
    <div>grand-grandchild text 1</div>
    grandchild text 2
  </div>
  child text 3<br>
  child text 4
</div>

jsFiddle

【讨论】:

我试过这个。它乱序打印标签名称。有没有办法按它们出现的顺序打印标签名称?我在这里问了一个单独的问题***.com/questions/63276378/…【参考方案5】:

我得到了很多带有接受过滤功能的空文本节点。如果您只对选择包含非空格的文本节点感兴趣,请尝试将 nodeValue 条件添加到您的 filter 函数中,例如简单的 $.trim(this.nodevalue) !== ''

$('element')
    .contents()
    .filter(function()
        return this.nodeType === 3 && $.trim(this.nodeValue) !== '';
    );

http://jsfiddle.net/ptp6m97v/

或者为了避免内容看起来像空格但不是的奇怪情况(例如软连字符&amp;shy; 字符、换行符\n、制表符等),您可以尝试使用正则表达式。例如,\S 将匹配任何非空白字符:

$('element')
        .contents()
        .filter(function()
            return this.nodeType === 3 && /\S/.test(this.nodeValue);
        );

【讨论】:

我试过这个。它乱序打印标签名称。有没有办法按它们出现的顺序打印标签名称?我在这里问了一个单独的问题***.com/questions/63276378/…【参考方案6】:

如果您可以假设所有子节点都是元素节点或文本节点,那么这是一种解决方案。

将所有子文本节点作为一个 jquery 集合获取:

$('selector').clone().children().remove().end().contents();

要获得原始元素的副本,其中非文本子元素已删除:

$('selector').clone().children().remove().end();

【讨论】:

刚刚注意到 Tim Down 对另一个答案的评论。此解决方案仅获取直接子代,而不是所有后代。【参考方案7】:

由于某种原因,contents() 对我不起作用,所以如果它对您不起作用,这是我制定的解决方案,我创建了 jQuery.fn.descendants,并选择是否包含文本节点

用法


获取所有后代,包括文本节点和元素节点

jQuery('body').descendants('all');

获取所有仅返回文本节点的后代

jQuery('body').descendants(true);

获取所有仅返回元素节点的后代

jQuery('body').descendants();

Coffeescript 原创

jQuery.fn.descendants = ( textNodes ) ->

    # if textNodes is 'all' then textNodes and elementNodes are allowed
    # if textNodes if true then only textNodes will be returned
    # if textNodes is not provided as an argument then only element nodes
    # will be returned

    allowedTypes = if textNodes is 'all' then [1,3] else if textNodes then [3] else [1]

    # nodes we find
    nodes = []


    dig = (node) ->

        # loop through children
        for child in node.childNodes

            # push child to collection if has allowed type
            nodes.push(child) if child.nodeType in allowedTypes

            # dig through child if has children
            dig child if child.childNodes.length


    # loop and dig through nodes in the current
    # jQuery object
    dig node for node in this


    # wrap with jQuery
    return jQuery(nodes)

加入 Javascript 版本

var __indexOf=[].indexOf||function(e)for(var t=0,n=this.length;t<n;t++)if(t in this&&this[t]===e)return treturn-1; /* indexOf polyfill ends here*/ jQuery.fn.descendants=function(e)var t,n,r,i,s,o;t=e==="all"?[1,3]:e?[3]:[1];i=[];n=function(e)var r,s,o,u,a,f;u=e.childNodes;f=[];for(s=0,o=u.length;s<o;s++)r=u[s];if(a=r.nodeType,__indexOf.call(t,a)>=0)i.push(r)if(r.childNodes.length)f.push(n(r))elsef.push(void 0)return f;for(s=0,o=this.length;s<o;s++)r=this[s];n(r)return jQuery(i)

未缩小的 javascript 版本:http://pastebin.com/cX3jMfuD

这是跨浏览器,代码中包含一个小的Array.indexOf polyfill。

【讨论】:

【参考方案8】:

也可以这样:

var textContents = $(document.getElementById("ElementId").childNodes).filter(function()
        return this.nodeType == 3;
);

以上代码从给定元素的直接子子节点中过滤 textNodes。

【讨论】:

...但不是所有 descendant 子节点(例如,作为原始元素子元素的子元素的文本节点)。【参考方案9】:

如果你想去掉所有标签,那么试试这个

功能:

String.prototype.stripTags=function()
var rtag=/<.*?[^>]>/g;
return this.replace(rtag,'');

用法:

var newText=$('selector').html().stripTags();

【讨论】:

【参考方案10】:

对我来说,普通的旧 .contents() 似乎可以返回文本节点,只需要小心你的选择器,这样你就知道它们将是文本节点。

例如,这用pre标签包裹了我表中所有TD的文本内容,没有问题。

jQuery("#resultTable td").content().wrap("<pre/>")

【讨论】:

【参考方案11】:

我遇到了同样的问题并解决了:

代码:

$.fn.nextNode = function()
  var contents = $(this).parent().contents();
  return contents.get(contents.index(this)+1);

用法:

$('#my_id').nextNode();

类似于next(),但也返回文本节点。

【讨论】:

.nextSibling 来自 Dom 规范:developer.mozilla.org/en/Document_Object_Model_(DOM)/…

以上是关于如何使用 jQuery 选择文本节点?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 jquery 选择某些文本节点?

如何在 jQuery 中实现任意文本节点选择

如何使用jQuery在元素的内部文本的最后一个单词周围包裹一个范围?

如何使用 jQuery 禁用文本选择?

如何使用 jQuery 禁用文本选择?

如何使用 jQuery 选择带有文本的按钮