Character offset in an Internet Explorer TextRange

后端 未结 4 996
醉话见心
醉话见心 2020-12-11 07:42

As far as I can tell there\'s no simple way of retrieving a character offset from a TextRange object in Internet Explorer. The W3C Range object has a node, and the offset in

相关标签:
4条回答
  • 2020-12-11 07:59

    I used a slightly simpler solution using the offset values of a textRange:

    function getIECharOffset() {
      var offset = 0;
    
      // get the users selection - this handles empty selections
      var userSelection = document.selection.createRange();
    
      // get a selection from the contents of the parent element
      var parentSelection = userSelection.parentElement().createTextRange();
    
      // loop - moving the parent selection on a character at a time until the offsets match
      while (!offsetEqual(parentSelection, userSelection)) {
        parentSelection.move('character');
        offset++;
      }
    
      // return the number of char you have moved through
      return offset;
    }
    
    function offsetEqual(arg1, arg2) {
      if (arg1.offsetLeft == arg2.offsetLeft && arg1.offsetTop == arg2.offsetTop) {
        return true;
      }
      return false;
    }
    
    0 讨论(0)
  • 2020-12-11 08:02

    I use a method based on this caret position trick:

    // Assume r is a range:
    var offsetFromBody = Math.abs( r.moveEnd('character', -1000000) );
    

    Since moveEnd returns the number of characters actually moved, offset should now be the offset from the start of the document. This works fine for testing primitive caret movement, but for expanded selections and for getting the exact node that holds the range anchor you'll need something more complex:

    // where paramter r is a range:
    function getRangeOffsetIE( r ) {
      var end = Math.abs( r.duplicate().moveEnd('character', -1000000) );
      // find the anchor element's offset
      var range = r.duplicate();
      r.collapse( false );
      var parentElm = range.parentElement();
      var children = parentElm.getElementsByTagName('*');
      for (var i = children.length - 1; i >= 0; i--) {
        range.moveToElementText( children[i] );
        if ( range.inRange(r) ) {
          parentElm = children[i];
          break;
        }
      }
      range.moveToElementText( parentElm );
      return end - Math.abs( range.moveStart('character', -1000000) );
    }
    

    This should return the correct caret text offset. Of course, if you know the target node already, or are able to provide a context, then you can skip the whole looping search mess.

    0 讨论(0)
  • 2020-12-11 08:08

    I'd suggest IERange, or just the TextRange-to-DOM Range algorithm from it.

    Update, 9 August 2011

    I'd now suggest using my own Rangy library, which is similar in idea to IERange but much more fully realized and supported.

    0 讨论(0)
  • 2020-12-11 08:20

    You can iterate through the body element's TextRange.text property using String.substring() to compare against the TextRange for which you want the character offset.

    function charOffset(textRange, parentTextRange)
     { var parentTxt = parentTextRange.text;
       var txt       = textRange.text;
       var parentLen = parentTxt.length;
    
       for(int i=0; i < parentLen ; ++i) 
        { if (parentTxt.substring(i, txt.length+i) == txt) 
           { var originalPosition = textRange.getBookmark();
    
             //moves back one and searches backwards for same text
             textRange.moveStart("character",-1);
             var foundOther = textRange.findText(textRange.text,-parentLen,1);
    
             //if no others were found return offset
             if (!foundOther) return i;
    
             //returns to original position to try next offset
             else textRange.moveToBookmark(originalPosition);
           }
        }
    
       return -1;
     }
    

    [Reference for findText()]

    0 讨论(0)
提交回复
热议问题