代码之家  ›  专栏  ›  技术社区  ›  Ran

如何在包含iframe或仅包含框架的HTML文档中查找所选内容

  •  4
  • Ran  · 技术社区  · 15 年前

    如果文本可能在某个框架(或iframes)内,是否有方法在HTML文档中查找所选文本?

    如果文档没有框架,那么很简单:

    var text;
    if (document.getSelection) {
     // Firefox and friends
     text = document.getSelection();
    } else if (document.selection) {
     // IE 
     text = document.selection.createRange();
    }
    if (text == undefined || text == '') {
     // Iterate over all textarea elements and see if one of them has selection
     var areas = document.getElementsByTagName('textarea');
     for(var i = 0; i = areas.length; i++) {
      if(areas[i].selectionStart != undefined && 
         areas[i].selectionStart != areas[i].selectionEnd){
       text = areas[i].value.substring(areas[i].selectionStart, a[i].selectionEnd);
       break;
      }   
     }   
    }
    // Now if document has selected text, it's in text
    

    所以这可以跨浏览器工作(虽然不是很漂亮)。

    问题在于文档包含框架或iframes时。 框架有自己的文档,所以仅仅使用上面的代码是不够的。 人们可能会在框架树上迭代并在其中一个框架中搜索选定的文本,但是一般来说,框架可以具有来自不同域的内容,因此即使我在搜索选定文本时遍历根文档的所有框架和所有子框架等,我也没有访问其HTML的权限,对吗?所以我无法获取他们选定的文本。

    是否有(简单)可靠的方法在网页上查找所选文本,即使网页包含框架?

    谢谢

    2 回复  |  直到 11 年前
        1
  •  3
  •   Ran    15 年前

    为了回答我自己的问题,经过一番调查: 因此,如果帧来自不同的域,那么您就无法对其进行任何处理,因为您没有访问它们的DOM的权限。但是,在所有帧都位于同一域(例如gmail)的常见情况下,只需像树一样迭代主题。下面是实现这一点的代码:

    下面的代码用于统计所选文本的字符和单词的书签:

    javascript:(function(){
      // Function: finds selected text on document d.
      // @return the selected text or null
      function f(d){
        var t;
        if (d.getSelection) t = d.getSelection();
        else if(d.selection) t = d.selection.createRange();
        if (t.text != undefined) t = t.text;
        if (!t || t=='') {
          var a = d.getElementsByTagName('textarea');
          for (var i = 0; i < a.length; ++i) {
            if (a[i].selectionStart != undefined && a[i].selectionStart != a[i].selectionEnd) {
              t = a[i].value.substring(a[i].selectionStart, a[i].selectionEnd);
              break;
            }
          }
        }
        return t;
      };
      // Function: finds selected text in document d and frames and subframes of d
      // @return the selected text or null
      function g(d){
        var t;
        try{t = f(d);}catch(e){};
        if (!t || t == '') {
          var fs = d.getElementsByTagName('frame');
          for (var i = 0; i < fs.length; ++i){
            t = g(fs[i].contentDocument);
            if(t && t.toString() != '') break;
          }
          if (!t || t.toString() == '') {
            fs = d.getElementsByTagName('iframe');
            for (var i = 0; i < fs.length; ++i){
              t = g(fs[i].contentDocument);
              if(t && t.toString() != '') break;
            }
          }
        }
        return t;
      };
      var t= g(document);
      if (!t || t == '') alert('please select some text');
      else alert('Chars: '+t.toString().length+'\nWords: '+t.toString().match(/(\S+)/g).length);
    })()
    
        2
  •  0
  •   Ohhh    12 年前

    @对你自己的问题有很好的回答。但是,如果iframe文档未定义,那么函数将失败。我添加了一个条件来检查这个,现在它在我尝试过的每个站点上都有效,包括Gmail。 if ((!t || t == '') && d) 再次感谢伟大的代码。

    var getSelectedText = function(){
      // Function: finds selected text on document d.
      // @return the selected text or null
      function f(d){
        var t;
        if (d.getSelection) t = d.getSelection();
        else if(d.selection) t = d.selection.createRange();
        if (t.text != undefined) t = t.text;
        if (!t || t=='') {
          var a = d.getElementsByTagName('textarea');
          for (var i = 0; i < a.length; ++i) {
            if (a[i].selectionStart != undefined && a[i].selectionStart != a[i].selectionEnd) {
              t = a[i].value.substring(a[i].selectionStart, a[i].selectionEnd);
              break;
            }
          }
        }
        return t;
      };
      // Function: finds selected text in document d and frames and subframes of d
      // @return the selected text or null
      function g(d){
        var t;
        try{t = f(d);}catch(e){console.log('ERROR: ',e);};
        if ((!t || t == '') && d){
          var fs = d.getElementsByTagName('frame');
          for (var i = 0; i < fs.length; ++i){
            t = g(fs[i].contentDocument);
            if(t && t.toString() != '') break;
          }
          if (!t || t.toString() == '') {
            fs = d.getElementsByTagName('iframe');
            for (var i = 0; i < fs.length; ++i){
              t = g(fs[i].contentDocument);
              if(t && t.toString() != '') break;
            }
          }
        }
        return t;
      };
      var t= g(document);
      if (!t || t == '') ;
      else return t.toString();
    }