123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- // Copyright 2007 The Closure Library Authors. All Rights Reserved.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS-IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- /**
- * @fileoverview Iterator between two DOM text range positions.
- *
- * @author robbyw@google.com (Robby Walker)
- */
- goog.provide('goog.dom.TextRangeIterator');
- goog.require('goog.array');
- goog.require('goog.dom');
- goog.require('goog.dom.NodeType');
- goog.require('goog.dom.RangeIterator');
- goog.require('goog.dom.TagName');
- goog.require('goog.iter.StopIteration');
- /**
- * Subclass of goog.dom.TagIterator that iterates over a DOM range. It
- * adds functions to determine the portion of each text node that is selected.
- *
- * @param {Node} startNode The starting node position.
- * @param {number} startOffset The offset in to startNode. If startNode is
- * an element, indicates an offset in to childNodes. If startNode is a
- * text node, indicates an offset in to nodeValue.
- * @param {Node} endNode The ending node position.
- * @param {number} endOffset The offset in to endNode. If endNode is
- * an element, indicates an offset in to childNodes. If endNode is a
- * text node, indicates an offset in to nodeValue.
- * @param {boolean=} opt_reverse Whether to traverse nodes in reverse.
- * @constructor
- * @extends {goog.dom.RangeIterator}
- * @final
- */
- goog.dom.TextRangeIterator = function(
- startNode, startOffset, endNode, endOffset, opt_reverse) {
- /**
- * The first node in the selection.
- * @private {Node}
- */
- this.startNode_ = null;
- /**
- * The last node in the selection.
- * @private {Node}
- */
- this.endNode_ = null;
- /**
- * The offset within the first node in the selection.
- * @private {number}
- */
- this.startOffset_ = 0;
- /**
- * The offset within the last node in the selection.
- * @private {number}
- */
- this.endOffset_ = 0;
- var goNext;
- if (startNode) {
- this.startNode_ = startNode;
- this.startOffset_ = startOffset;
- this.endNode_ = endNode;
- this.endOffset_ = endOffset;
- // Skip to the offset nodes - being careful to special case BRs since these
- // have no children but still can appear as the startContainer of a range.
- if (startNode.nodeType == goog.dom.NodeType.ELEMENT &&
- /** @type {!Element} */ (startNode).tagName != goog.dom.TagName.BR) {
- var startChildren = startNode.childNodes;
- var candidate = startChildren[startOffset];
- if (candidate) {
- this.startNode_ = candidate;
- this.startOffset_ = 0;
- } else {
- if (startChildren.length) {
- this.startNode_ =
- /** @type {Node} */ (goog.array.peek(startChildren));
- }
- goNext = true;
- }
- }
- if (endNode.nodeType == goog.dom.NodeType.ELEMENT) {
- this.endNode_ = endNode.childNodes[endOffset];
- if (this.endNode_) {
- this.endOffset_ = 0;
- } else {
- // The offset was past the last element.
- this.endNode_ = endNode;
- }
- }
- }
- goog.dom.TextRangeIterator.base(
- this, 'constructor', opt_reverse ? this.endNode_ : this.startNode_,
- opt_reverse);
- if (goNext) {
- try {
- this.next();
- } catch (e) {
- if (e != goog.iter.StopIteration) {
- throw e;
- }
- }
- }
- };
- goog.inherits(goog.dom.TextRangeIterator, goog.dom.RangeIterator);
- /** @override */
- goog.dom.TextRangeIterator.prototype.getStartTextOffset = function() {
- // Offsets only apply to text nodes. If our current node is the start node,
- // return the saved offset. Otherwise, return 0.
- return this.node.nodeType != goog.dom.NodeType.TEXT ?
- -1 :
- this.node == this.startNode_ ? this.startOffset_ : 0;
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.getEndTextOffset = function() {
- // Offsets only apply to text nodes. If our current node is the end node,
- // return the saved offset. Otherwise, return the length of the node.
- return this.node.nodeType != goog.dom.NodeType.TEXT ?
- -1 :
- this.node == this.endNode_ ? this.endOffset_ : this.node.nodeValue.length;
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.getStartNode = function() {
- return this.startNode_;
- };
- /**
- * Change the start node of the iterator.
- * @param {Node} node The new start node.
- */
- goog.dom.TextRangeIterator.prototype.setStartNode = function(node) {
- if (!this.isStarted()) {
- this.setPosition(node);
- }
- this.startNode_ = node;
- this.startOffset_ = 0;
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.getEndNode = function() {
- return this.endNode_;
- };
- /**
- * Change the end node of the iterator.
- * @param {Node} node The new end node.
- */
- goog.dom.TextRangeIterator.prototype.setEndNode = function(node) {
- this.endNode_ = node;
- this.endOffset_ = 0;
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.isLast = function() {
- return this.isStarted() && this.node == this.endNode_ &&
- (!this.endOffset_ || !this.isStartTag());
- };
- /**
- * Move to the next position in the selection.
- * Throws {@code goog.iter.StopIteration} when it passes the end of the range.
- * @return {Node} The node at the next position.
- * @override
- */
- goog.dom.TextRangeIterator.prototype.next = function() {
- if (this.isLast()) {
- throw goog.iter.StopIteration;
- }
- // Call the super function.
- return goog.dom.TextRangeIterator.superClass_.next.call(this);
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.skipTag = function() {
- goog.dom.TextRangeIterator.superClass_.skipTag.apply(this);
- // If the node we are skipping contains the end node, we just skipped past
- // the end, so we stop the iteration.
- if (goog.dom.contains(this.node, this.endNode_)) {
- throw goog.iter.StopIteration;
- }
- };
- /** @override */
- goog.dom.TextRangeIterator.prototype.copyFrom = function(other) {
- this.startNode_ = other.startNode_;
- this.endNode_ = other.endNode_;
- this.startOffset_ = other.startOffset_;
- this.endOffset_ = other.endOffset_;
- this.isReversed_ = other.isReversed_;
- goog.dom.TextRangeIterator.superClass_.copyFrom.call(this, other);
- };
- /**
- * @return {!goog.dom.TextRangeIterator} An identical iterator.
- * @override
- */
- goog.dom.TextRangeIterator.prototype.clone = function() {
- var copy = new goog.dom.TextRangeIterator(
- this.startNode_, this.startOffset_, this.endNode_, this.endOffset_,
- this.isReversed_);
- copy.copyFrom(this);
- return copy;
- };
|