UNPKG

google-closure-library

Version:
295 lines (272 loc) 12.2 kB
/** * @license * Copyright The Closure Library Authors. * SPDX-License-Identifier: Apache-2.0 */ /** * @fileoverview Testing utilities for editor specific DOM related tests. */ goog.setTestOnly('goog.testing.editor.dom'); goog.provide('goog.testing.editor.dom'); goog.require('goog.dom.AbstractRange'); goog.require('goog.dom.NodeType'); goog.require('goog.dom.TagIterator'); goog.require('goog.dom.TagWalkType'); goog.require('goog.iter'); goog.require('goog.string'); goog.require('goog.testing.asserts'); /** * Returns the previous (in document order) node from the given node that is a * non-empty text node, or null if none is found or opt_stopAt is not an * ancestor of node. Note that if the given node has children, the search will * start from the end tag of the node, meaning all its descendants will be * included in the search, unless opt_skipDescendants is true. * @param {Node} node Node to start searching from. * @param {Node=} opt_stopAt Node to stop searching at (search will be * restricted to this node's subtree), defaults to the body of the document * containing node. * @param {boolean=} opt_skipDescendants Whether to skip searching the given * node's descentants. * @return {Text} The previous (in document order) node from the given node * that is a non-empty text node, or null if none is found. */ goog.testing.editor.dom.getPreviousNonEmptyTextNode = function( node, opt_stopAt, opt_skipDescendants) { 'use strict'; return goog.testing.editor.dom.getPreviousNextNonEmptyTextNodeHelper_( node, opt_stopAt, opt_skipDescendants, true); }; /** * Returns the next (in document order) node from the given node that is a * non-empty text node, or null if none is found or opt_stopAt is not an * ancestor of node. Note that if the given node has children, the search will * start from the start tag of the node, meaning all its descendants will be * included in the search, unless opt_skipDescendants is true. * @param {Node} node Node to start searching from. * @param {Node=} opt_stopAt Node to stop searching at (search will be * restricted to this node's subtree), defaults to the body of the document * containing node. * @param {boolean=} opt_skipDescendants Whether to skip searching the given * node's descentants. * @return {Text} The next (in document order) node from the given node that * is a non-empty text node, or null if none is found or opt_stopAt is not * an ancestor of node. */ goog.testing.editor.dom.getNextNonEmptyTextNode = function( node, opt_stopAt, opt_skipDescendants) { 'use strict'; return goog.testing.editor.dom.getPreviousNextNonEmptyTextNodeHelper_( node, opt_stopAt, opt_skipDescendants, false); }; /** * Helper that returns the previous or next (in document order) node from the * given node that is a non-empty text node, or null if none is found or * opt_stopAt is not an ancestor of node. Note that if the given node has * children, the search will start from the end or start tag of the node * (depending on whether it's searching for the previous or next node), meaning * all its descendants will be included in the search, unless * opt_skipDescendants is true. * @param {Node} node Node to start searching from. * @param {Node=} opt_stopAt Node to stop searching at (search will be * restricted to this node's subtree), defaults to the body of the document * containing node. * @param {boolean=} opt_skipDescendants Whether to skip searching the given * node's descentants. * @param {boolean=} opt_isPrevious Whether to search for the previous non-empty * text node instead of the next one. * @return {Text} The next (in document order) node from the given node that * is a non-empty text node, or null if none is found or opt_stopAt is not * an ancestor of node. * @private */ goog.testing.editor.dom.getPreviousNextNonEmptyTextNodeHelper_ = function( node, opt_stopAt, opt_skipDescendants, opt_isPrevious) { 'use strict'; opt_stopAt = opt_stopAt || node.ownerDocument.body; // Initializing the iterator to iterate over the children of opt_stopAt // makes it stop only when it finishes iterating through all of that // node's children, even though we will start at a different node and exit // that starting node's subtree in the process. const iter = new goog.dom.TagIterator(opt_stopAt, opt_isPrevious); // TODO(user): Move this logic to a new method in TagIterator such as // skipToNode(). // Then we set the iterator to start at the given start node, not opt_stopAt. let walkType; // Let TagIterator set the initial walk type by default. let depth = goog.testing.editor.dom.getRelativeDepth_(node, opt_stopAt); if (depth == -1) { return null; // Fail because opt_stopAt is not an ancestor of node. } if (node.nodeType == goog.dom.NodeType.ELEMENT) { if (opt_skipDescendants) { // Specifically set the initial walk type so that we skip the descendant // subtree by starting at the start if going backwards or at the end if // going forwards. walkType = opt_isPrevious ? goog.dom.TagWalkType.START_TAG : goog.dom.TagWalkType.END_TAG; } else { // We're starting "inside" an element node so the depth needs to be one // deeper than the node's actual depth. That's how TagIterator works! depth++; } } iter.setPosition(node, walkType, depth); // Advance the iterator so it skips the start node. let it = iter.next(); if (it.done) return null; // Now just get the first non-empty text node the iterator finds. const filter = goog.iter.filter(iter, goog.testing.editor.dom.isNonEmptyTextNode_); it = filter.next(); return it.done ? null : /** @type {!Text} */ (it.value); }; /** * Returns whether the given node is a non-empty text node. * @param {Node} node Node to be checked. * @return {boolean} Whether the given node is a non-empty text node. * @private */ goog.testing.editor.dom.isNonEmptyTextNode_ = function(node) { 'use strict'; if (node && node.nodeType == goog.dom.NodeType.TEXT) { node = /** @type {!Text} */ (node); return node.length > 0; } return false; }; /** * Returns the depth of the given node relative to the given parent node, or -1 * if the given node is not a descendant of the given parent node. E.g. if * node == parentNode returns 0, if node.parentNode == parentNode returns 1, * etc. * @param {Node} node Node whose depth to get. * @param {Node} parentNode Node relative to which the depth should be * calculated. * @return {number} The depth of the given node relative to the given parent * node, or -1 if the given node is not a descendant of the given parent * node. * @private */ goog.testing.editor.dom.getRelativeDepth_ = function(node, parentNode) { 'use strict'; let depth = 0; while (node) { if (node == parentNode) { return depth; } node = node.parentNode; depth++; } return -1; }; /** * Assert that the range is surrounded by the given strings. This is useful * because different browsers can place the range endpoints inside different * nodes even when visually the range looks the same. Also, there may be empty * text nodes in the way (again depending on the browser) making it difficult to * use assertRangeEquals. * @param {string} before String that should occur immediately before the start * point of the range. If this is the empty string, assert will only succeed * if there is no text before the start point of the range. * @param {string} after String that should occur immediately after the end * point of the range. If this is the empty string, assert will only succeed * if there is no text after the end point of the range. * @param {goog.dom.AbstractRange} range The range to be tested. * @param {Node=} opt_stopAt Node to stop searching at (search will be * restricted to this node's subtree). */ goog.testing.editor.dom.assertRangeBetweenText = function( before, after, range, opt_stopAt) { 'use strict'; const previousText = goog.testing.editor.dom.getTextFollowingRange_(range, true, opt_stopAt); if (before == '') { assertNull( 'Expected nothing before range but found <' + previousText + '>', previousText); } else { assertNotNull( 'Expected <' + before + '> before range but found nothing', previousText); assertTrue( 'Expected <' + before + '> before range but found <' + previousText + '>', goog.string.endsWith( /** @type {string} */ (previousText), before)); } const nextText = goog.testing.editor.dom.getTextFollowingRange_(range, false, opt_stopAt); if (after == '') { assertNull( 'Expected nothing after range but found <' + nextText + '>', nextText); } else { assertNotNull( 'Expected <' + after + '> after range but found nothing', nextText); assertTrue( 'Expected <' + after + '> after range but found <' + nextText + '>', goog.string.startsWith( /** @type {string} */ (nextText), after)); } }; /** * Returns the text that follows the given range, where the term "follows" means * "comes immediately before the start of the range" if isBefore is true, and * "comes immediately after the end of the range" if isBefore is false, or null * if no non-empty text node is found. * @param {goog.dom.AbstractRange} range The range to search from. * @param {boolean} isBefore Whether to search before the range instead of * after it. * @param {Node=} opt_stopAt Node to stop searching at (search will be * restricted to this node's subtree). * @return {?string} The text that follows the given range, or null if no * non-empty text node is found. * @private */ goog.testing.editor.dom.getTextFollowingRange_ = function( range, isBefore, opt_stopAt) { 'use strict'; let followingTextNode; const endpointNode = isBefore ? range.getStartNode() : range.getEndNode(); const endpointOffset = isBefore ? range.getStartOffset() : range.getEndOffset(); const getFollowingTextNode = isBefore ? goog.testing.editor.dom.getPreviousNonEmptyTextNode : goog.testing.editor.dom.getNextNonEmptyTextNode; if (endpointNode.nodeType == goog.dom.NodeType.TEXT) { // Range endpoint is in a text node. const endText = endpointNode.nodeValue; if (isBefore ? endpointOffset > 0 : endpointOffset < endText.length) { // There is text in this node following the endpoint so return the portion // that follows the endpoint. return isBefore ? endText.slice(0, endpointOffset) : endText.slice(endpointOffset); } else { // There is no text following the endpoint so look for the follwing text // node. followingTextNode = getFollowingTextNode(endpointNode, opt_stopAt); return followingTextNode && followingTextNode.nodeValue; } } else { // Range endpoint is in an element node. const numChildren = endpointNode.childNodes.length; if (isBefore ? endpointOffset > 0 : endpointOffset < numChildren) { // There is at least one child following the endpoint. const followingChild = endpointNode .childNodes[isBefore ? endpointOffset - 1 : endpointOffset]; if (goog.testing.editor.dom.isNonEmptyTextNode_(followingChild)) { // The following child has text so return that. return followingChild.nodeValue; } else { // The following child has no text so look for the following text node. followingTextNode = getFollowingTextNode(followingChild, opt_stopAt); return followingTextNode && followingTextNode.nodeValue; } } else { // There is no child following the endpoint, so search from the endpoint // node, but don't search its children because they are not following the // endpoint! followingTextNode = getFollowingTextNode(endpointNode, opt_stopAt, true); return followingTextNode && followingTextNode.nodeValue; } } };